Completed
Branch BUG/11302/correct-error-messag... (694f28)
by
unknown
29:59 queued 17:10
created

EE_Base_Class::ID()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 10
Code Lines 6

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 2
eloc 6
nc 2
nop 0
dl 0
loc 10
rs 9.4285
c 0
b 0
f 0
1
<?php if ( ! defined('EVENT_ESPRESSO_VERSION')) {
2
    exit('No direct script access allowed');
3
}
4
do_action('AHEE_log', __FILE__, ' FILE LOADED', '');
5
6
7
8
/**
9
 * Event Espresso
10
 * Event Registration and Management Plugin for WordPress
11
 * @ package            Event Espresso
12
 * @ author                Seth Shoultes
13
 * @ copyright        (c) 2008-2011 Event Espresso  All Rights Reserved.
14
 * @ license            http://eventespresso.com/support/terms-conditions/   * see Plugin Licensing *
15
 * @ link                    http://www.eventespresso.com
16
 * @ version            4.0
17
 * ------------------------------------------------------------------------
18
 * EE_Base_Class class
19
 *
20
 * @package                   Event Espresso
21
 * @subpackage                includes/classes/EE_Base_Class.class.php
22
 * @author                    Michael Nelson
23
 *                            ------------------------------------------------------------------------
24
 */
25
abstract class EE_Base_Class
26
{
27
28
    /**
29
     * This is an array of the original properties and values provided during construction
30
     * of this model object. (keys are model field names, values are their values).
31
     * This list is important to remember so that when we are merging data from the db, we know
32
     * which values to override and which to not override.
33
     *
34
     * @var array
35
     */
36
    protected $_props_n_values_provided_in_constructor;
37
38
    /**
39
     * Timezone
40
     * This gets set by the "set_timezone()" method so that we know what timezone incoming strings|timestamps are in.
41
     * This can also be used before a get to set what timezone you want strings coming out of the object to be in.  NOT
42
     * all EE_Base_Class child classes use this property but any that use a EE_Datetime_Field data type will have
43
     * access to it.
44
     *
45
     * @var string
46
     */
47
    protected $_timezone;
48
49
50
51
    /**
52
     * date format
53
     * pattern or format for displaying dates
54
     *
55
     * @var string $_dt_frmt
56
     */
57
    protected $_dt_frmt;
58
59
60
61
    /**
62
     * time format
63
     * pattern or format for displaying time
64
     *
65
     * @var string $_tm_frmt
66
     */
67
    protected $_tm_frmt;
68
69
70
71
    /**
72
     * This property is for holding a cached array of object properties indexed by property name as the key.
73
     * The purpose of this is for setting a cache on properties that may have calculated values after a
74
     * prepare_for_get.  That way the cache can be checked first and the calculated property returned instead of having
75
     * to recalculate. Used by _set_cached_property() and _get_cached_property() methods.
76
     *
77
     * @var array
78
     */
79
    protected $_cached_properties = array();
80
81
    /**
82
     * An array containing keys of the related model, and values are either an array of related mode objects or a
83
     * single
84
     * related model object. see the model's _model_relations. The keys should match those specified. And if the
85
     * relation is of type EE_Belongs_To (or one of its children), then there should only be ONE related model object,
86
     * all others have an array)
87
     *
88
     * @var array
89
     */
90
    protected $_model_relations = array();
91
92
    /**
93
     * Array where keys are field names (see the model's _fields property) and values are their values. To see what
94
     * their types should be, look at what that field object returns on its prepare_for_get and prepare_for_set methods)
95
     *
96
     * @var array
97
     */
98
    protected $_fields = array();
99
100
    /**
101
     * @var boolean indicating whether or not this model object is intended to ever be saved
102
     * For example, we might create model objects intended to only be used for the duration
103
     * of this request and to be thrown away, and if they were accidentally saved
104
     * it would be a bug.
105
     */
106
    protected $_allow_persist = true;
107
108
    /**
109
     * @var boolean indicating whether or not this model object's properties have changed since construction
110
     */
111
    protected $_has_changes = false;
112
113
    /**
114
     * @var EEM_Base
115
     */
116
    protected $_model;
117
118
119
120
    /**
121
     * basic constructor for Event Espresso classes, performs any necessary initialization, and verifies it's children
122
     * play nice
123
     *
124
     * @param array   $fieldValues                             where each key is a field (ie, array key in the 2nd
125
     *                                                         layer of the model's _fields array, (eg, EVT_ID,
126
     *                                                         TXN_amount, QST_name, etc) and values are their values
127
     * @param boolean $bydb                                    a flag for setting if the class is instantiated by the
128
     *                                                         corresponding db model or not.
129
     * @param string  $timezone                                indicate what timezone you want any datetime fields to
130
     *                                                         be in when instantiating a EE_Base_Class object.
131
     * @param array   $date_formats                            An array of date formats to set on construct where first
132
     *                                                         value is the date_format and second value is the time
133
     *                                                         format.
134
     * @throws EE_Error
135
     */
136
    protected function __construct($fieldValues = array(), $bydb = false, $timezone = '', $date_formats = array())
137
    {
138
        $className = get_class($this);
139
        do_action("AHEE__{$className}__construct", $this, $fieldValues);
140
        $model = $this->get_model();
141
        $model_fields = $model->field_settings(false);
142
        // ensure $fieldValues is an array
143
        $fieldValues = is_array($fieldValues) ? $fieldValues : array($fieldValues);
144
        // EEH_Debug_Tools::printr( $fieldValues, '$fieldValues  <br /><span style="font-size:10px;font-weight:normal;">' . __FILE__ . '<br />line no: ' . __LINE__ . '</span>', 'auto' );
145
        // verify client code has not passed any invalid field names
146
        foreach ($fieldValues as $field_name => $field_value) {
147 View Code Duplication
            if ( ! isset($model_fields[$field_name])) {
148
                throw new EE_Error(sprintf(__("Invalid field (%s) passed to constructor of %s. Allowed fields are :%s",
149
                    "event_espresso"), $field_name, get_class($this), implode(", ", array_keys($model_fields))));
150
            }
151
        }
152
        // EEH_Debug_Tools::printr( $model_fields, '$model_fields  <br /><span style="font-size:10px;font-weight:normal;">' . __FILE__ . '<br />line no: ' . __LINE__ . '</span>', 'auto' );
153
        $this->_timezone = EEH_DTT_Helper::get_valid_timezone_string($timezone);
154
        if ( ! empty($date_formats) && is_array($date_formats)) {
155
            list($this->_dt_frmt, $this->_tm_frmt) = $date_formats;
156
        } else {
157
            //set default formats for date and time
158
            $this->_dt_frmt = (string)get_option('date_format', 'Y-m-d');
159
            $this->_tm_frmt = (string)get_option('time_format', 'g:i a');
160
        }
161
        //if db model is instantiating
162
        if ($bydb) {
163
            //client code has indicated these field values are from the database
164 View Code Duplication
            foreach ($model_fields as $fieldName => $field) {
165
                $this->set_from_db($fieldName, isset($fieldValues[$fieldName]) ? $fieldValues[$fieldName] : null);
166
            }
167
        } else {
168
            //we're constructing a brand
169
            //new instance of the model object. Generally, this means we'll need to do more field validation
170 View Code Duplication
            foreach ($model_fields as $fieldName => $field) {
171
                $this->set($fieldName, isset($fieldValues[$fieldName]) ? $fieldValues[$fieldName] : null, true);
172
            }
173
        }
174
        //remember what values were passed to this constructor
175
        $this->_props_n_values_provided_in_constructor = $fieldValues;
176
        //remember in entity mapper
177
        if ( ! $bydb && $model->has_primary_key_field() && $this->ID()) {
178
            $model->add_to_entity_map($this);
179
        }
180
        //setup all the relations
181
        foreach ($model->relation_settings() as $relation_name => $relation_obj) {
182
            if ($relation_obj instanceof EE_Belongs_To_Relation) {
183
                $this->_model_relations[$relation_name] = null;
184
            } else {
185
                $this->_model_relations[$relation_name] = array();
186
            }
187
        }
188
        /**
189
         * Action done at the end of each model object construction
190
         *
191
         * @param EE_Base_Class $this the model object just created
192
         */
193
        do_action('AHEE__EE_Base_Class__construct__finished', $this);
194
    }
195
196
197
198
    /**
199
     * Gets whether or not this model object is allowed to persist/be saved to the database.
200
     *
201
     * @return boolean
202
     */
203
    public function allow_persist()
204
    {
205
        return $this->_allow_persist;
206
    }
207
208
209
210
    /**
211
     * Sets whether or not this model object should be allowed to be saved to the DB.
212
     * Normally once this is set to FALSE you wouldn't set it back to TRUE, unless
213
     * you got new information that somehow made you change your mind.
214
     *
215
     * @param boolean $allow_persist
216
     * @return boolean
217
     */
218
    public function set_allow_persist($allow_persist)
219
    {
220
        return $this->_allow_persist = $allow_persist;
221
    }
222
223
224
225
    /**
226
     * Gets the field's original value when this object was constructed during this request.
227
     * This can be helpful when determining if a model object has changed or not
228
     *
229
     * @param string $field_name
230
     * @return mixed|null
231
     * @throws \EE_Error
232
     */
233
    public function get_original($field_name)
234
    {
235
        if (isset($this->_props_n_values_provided_in_constructor[$field_name])
236
            && $field_settings = $this->get_model()->field_settings_for($field_name)
237
        ) {
238
            return $field_settings->prepare_for_get($this->_props_n_values_provided_in_constructor[$field_name]);
239
        } else {
240
            return null;
241
        }
242
    }
243
244
245
246
    /**
247
     * @param EE_Base_Class $obj
248
     * @return string
249
     */
250
    public function get_class($obj)
251
    {
252
        return get_class($obj);
253
    }
254
255
256
257
    /**
258
     * Overrides parent because parent expects old models.
259
     * This also doesn't do any validation, and won't work for serialized arrays
260
     *
261
     * @param    string $field_name
262
     * @param    mixed  $field_value
263
     * @param bool      $use_default
264
     * @throws \EE_Error
265
     */
266
    public function set($field_name, $field_value, $use_default = false)
267
    {
268
        // if not using default and nothing has changed, and object has already been setup (has ID),
269
        // then don't do anything
270
        if (
271
            ! $use_default
272
            && $this->_fields[$field_name] === $field_value
273
            && $this->ID()
274
        ) {
275
            return;
276
        }
277
        $model = $this->get_model();
278
        $this->_has_changes = true;
279
        $field_obj = $model->field_settings_for($field_name);
280
        if ($field_obj instanceof EE_Model_Field_Base) {
281
            //			if ( method_exists( $field_obj, 'set_timezone' )) {
282
            if ($field_obj instanceof EE_Datetime_Field) {
283
                $field_obj->set_timezone($this->_timezone);
284
                $field_obj->set_date_format($this->_dt_frmt);
285
                $field_obj->set_time_format($this->_tm_frmt);
286
            }
287
            $holder_of_value = $field_obj->prepare_for_set($field_value);
288
            //should the value be null?
289
            if (($field_value === null || $holder_of_value === null || $holder_of_value === '') && $use_default) {
290
                $this->_fields[$field_name] = $field_obj->get_default_value();
291
                /**
292
                 * To save having to refactor all the models, if a default value is used for a
293
                 * EE_Datetime_Field, and that value is not null nor is it a DateTime
294
                 * object.  Then let's do a set again to ensure that it becomes a DateTime
295
                 * object.
296
                 *
297
                 * @since 4.6.10+
298
                 */
299
                if (
300
                    $field_obj instanceof EE_Datetime_Field
301
                    && $this->_fields[$field_name] !== null
302
                    && ! $this->_fields[$field_name] instanceof DateTime
303
                ) {
304
                    empty($this->_fields[$field_name])
305
                        ? $this->set($field_name, time())
306
                        : $this->set($field_name, $this->_fields[$field_name]);
307
                }
308
            } else {
309
                $this->_fields[$field_name] = $holder_of_value;
310
            }
311
            //if we're not in the constructor...
312
            //now check if what we set was a primary key
313
            if (
314
                //note: props_n_values_provided_in_constructor is only set at the END of the constructor
315
                $this->_props_n_values_provided_in_constructor
0 ignored issues
show
Bug Best Practice introduced by
The expression $this->_props_n_values_provided_in_constructor of type array is implicitly converted to a boolean; are you sure this is intended? If so, consider using ! empty($expr) instead to make it clear that you intend to check for an array without elements.

This check marks implicit conversions of arrays to boolean values in a comparison. While in PHP an empty array is considered to be equal (but not identical) to false, this is not always apparent.

Consider making the comparison explicit by using empty(..) or ! empty(...) instead.

Loading history...
316
                && $field_value
317
                && $field_name === $model->primary_key_name()
318
            ) {
319
                //if so, we want all this object's fields to be filled either with
320
                //what we've explicitly set on this model
321
                //or what we have in the db
322
                // echo "setting primary key!";
323
                $fields_on_model = self::_get_model(get_class($this))->field_settings();
324
                $obj_in_db = self::_get_model(get_class($this))->get_one_by_ID($field_value);
325
                foreach ($fields_on_model as $field_obj) {
326
                    if ( ! array_key_exists($field_obj->get_name(), $this->_props_n_values_provided_in_constructor)
327
                         && $field_obj->get_name() !== $field_name
328
                    ) {
329
                        $this->set($field_obj->get_name(), $obj_in_db->get($field_obj->get_name()));
330
                    }
331
                }
332
                //oh this model object has an ID? well make sure its in the entity mapper
333
                $model->add_to_entity_map($this);
334
            }
335
            //let's unset any cache for this field_name from the $_cached_properties property.
336
            $this->_clear_cached_property($field_name);
337
        } else {
338
            throw new EE_Error(sprintf(__("A valid EE_Model_Field_Base could not be found for the given field name: %s",
339
                "event_espresso"), $field_name));
340
        }
341
    }
342
343
344
345
    /**
346
     * This sets the field value on the db column if it exists for the given $column_name or
347
     * saves it to EE_Extra_Meta if the given $column_name does not match a db column.
348
     *
349
     * @see EE_message::get_column_value for related documentation on the necessity of this method.
350
     * @param string $field_name  Must be the exact column name.
351
     * @param mixed  $field_value The value to set.
352
     * @return int|bool @see EE_Base_Class::update_extra_meta() for return docs.
353
     * @throws \EE_Error
354
     */
355
    public function set_field_or_extra_meta($field_name, $field_value)
356
    {
357
        if ($this->get_model()->has_field($field_name)) {
358
            $this->set($field_name, $field_value);
359
            return true;
360
        } else {
361
            //ensure this object is saved first so that extra meta can be properly related.
362
            $this->save();
363
            return $this->update_extra_meta($field_name, $field_value);
364
        }
365
    }
366
367
368
369
    /**
370
     * This retrieves the value of the db column set on this class or if that's not present
371
     * it will attempt to retrieve from extra_meta if found.
372
     * Example Usage:
373
     * Via EE_Message child class:
374
     * Due to the dynamic nature of the EE_messages system, EE_messengers will always have a "to",
375
     * "from", "subject", and "content" field (as represented in the EE_Message schema), however they may
376
     * also have additional main fields specific to the messenger.  The system accommodates those extra
377
     * fields through the EE_Extra_Meta table.  This method allows for EE_messengers to retrieve the
378
     * value for those extra fields dynamically via the EE_message object.
379
     *
380
     * @param  string $field_name expecting the fully qualified field name.
381
     * @return mixed|null  value for the field if found.  null if not found.
382
     * @throws \EE_Error
383
     */
384
    public function get_field_or_extra_meta($field_name)
385
    {
386
        if ($this->get_model()->has_field($field_name)) {
387
            $column_value = $this->get($field_name);
388
        } else {
389
            //This isn't a column in the main table, let's see if it is in the extra meta.
390
            $column_value = $this->get_extra_meta($field_name, true, null);
391
        }
392
        return $column_value;
393
    }
394
395
396
397
    /**
398
     * See $_timezone property for description of what the timezone property is for.  This SETS the timezone internally
399
     * for being able to reference what timezone we are running conversions on when converting TO the internal timezone
400
     * (UTC Unix Timestamp) for the object OR when converting FROM the internal timezone (UTC Unix Timestamp). This is
401
     * available to all child classes that may be using the EE_Datetime_Field for a field data type.
402
     *
403
     * @access public
404
     * @param string $timezone A valid timezone string as described by @link http://www.php.net/manual/en/timezones.php
405
     * @return void
406
     * @throws \EE_Error
407
     */
408
    public function set_timezone($timezone = '')
409
    {
410
        $this->_timezone = EEH_DTT_Helper::get_valid_timezone_string($timezone);
411
        //make sure we clear all cached properties because they won't be relevant now
412
        $this->_clear_cached_properties();
413
        //make sure we update field settings and the date for all EE_Datetime_Fields
414
        $model_fields = $this->get_model()->field_settings(false);
415
        foreach ($model_fields as $field_name => $field_obj) {
416
            if ($field_obj instanceof EE_Datetime_Field) {
417
                $field_obj->set_timezone($this->_timezone);
418
                if (isset($this->_fields[$field_name]) && $this->_fields[$field_name] instanceof DateTime) {
419
                    $this->_fields[$field_name]->setTimezone(new DateTimeZone($this->_timezone));
420
                }
421
            }
422
        }
423
    }
424
425
426
427
    /**
428
     * This just returns whatever is set for the current timezone.
429
     *
430
     * @access public
431
     * @return string timezone string
432
     */
433
    public function get_timezone()
434
    {
435
        return $this->_timezone;
436
    }
437
438
439
440
    /**
441
     * This sets the internal date format to what is sent in to be used as the new default for the class
442
     * internally instead of wp set date format options
443
     *
444
     * @since 4.6
445
     * @param string $format should be a format recognizable by PHP date() functions.
446
     */
447
    public function set_date_format($format)
448
    {
449
        $this->_dt_frmt = $format;
450
        //clear cached_properties because they won't be relevant now.
451
        $this->_clear_cached_properties();
452
    }
453
454
455
456
    /**
457
     * This sets the internal time format string to what is sent in to be used as the new default for the
458
     * class internally instead of wp set time format options.
459
     *
460
     * @since 4.6
461
     * @param string $format should be a format recognizable by PHP date() functions.
462
     */
463
    public function set_time_format($format)
464
    {
465
        $this->_tm_frmt = $format;
466
        //clear cached_properties because they won't be relevant now.
467
        $this->_clear_cached_properties();
468
    }
469
470
471
472
    /**
473
     * This returns the current internal set format for the date and time formats.
474
     *
475
     * @param bool $full           if true (default), then return the full format.  Otherwise will return an array
476
     *                             where the first value is the date format and the second value is the time format.
477
     * @return mixed string|array
478
     */
479
    public function get_format($full = true)
480
    {
481
        return $full ? $this->_dt_frmt . ' ' . $this->_tm_frmt : array($this->_dt_frmt, $this->_tm_frmt);
482
    }
483
484
485
486
    /**
487
     * cache
488
     * stores the passed model object on the current model object.
489
     * In certain circumstances, we can use this cached model object instead of querying for another one entirely.
490
     *
491
     * @param string        $relationName    one of the keys in the _model_relations array on the model. Eg
492
     *                                       'Registration' associated with this model object
493
     * @param EE_Base_Class $object_to_cache that has a relation to this model object. (Eg, if this is a Transaction,
494
     *                                       that could be a payment or a registration)
495
     * @param null          $cache_id        a string or number that will be used as the key for any Belongs_To_Many
496
     *                                       items which will be stored in an array on this object
497
     * @throws EE_Error
498
     * @return mixed    index into cache, or just TRUE if the relation is of type Belongs_To (because there's only one
499
     *                  related thing, no array)
500
     */
501
    public function cache($relationName = '', $object_to_cache = null, $cache_id = null)
502
    {
503
        // its entirely possible that there IS no related object yet in which case there is nothing to cache.
504
        if ( ! $object_to_cache instanceof EE_Base_Class) {
505
            return false;
506
        }
507
        // also get "how" the object is related, or throw an error
508
        if ( ! $relationship_to_model = $this->get_model()->related_settings_for($relationName)) {
509
            throw new EE_Error(sprintf(__('There is no relationship to %s on a %s. Cannot cache it', 'event_espresso'),
510
                $relationName, get_class($this)));
511
        }
512
        // how many things are related ?
513
        if ($relationship_to_model instanceof EE_Belongs_To_Relation) {
514
            // if it's a "belongs to" relationship, then there's only one related model object  eg, if this is a registration, there's only 1 attendee for it
515
            // so for these model objects just set it to be cached
516
            $this->_model_relations[$relationName] = $object_to_cache;
517
            $return = true;
518
        } else {
519
            // otherwise, this is the "many" side of a one to many relationship, so we'll add the object to the array of related objects for that type.
520
            // eg: if this is an event, there are many registrations for that event, so we cache the registrations in an array
521
            if ( ! is_array($this->_model_relations[$relationName])) {
522
                // if for some reason, the cached item is a model object, then stick that in the array, otherwise start with an empty array
523
                $this->_model_relations[$relationName] = $this->_model_relations[$relationName] instanceof EE_Base_Class
524
                    ? array($this->_model_relations[$relationName]) : array();
525
            }
526
            // first check for a cache_id which is normally empty
527
            if ( ! empty($cache_id)) {
528
                // if the cache_id exists, then it means we are purposely trying to cache this with a known key that can then be used to retrieve the object later on
529
                $this->_model_relations[$relationName][$cache_id] = $object_to_cache;
530
                $return = $cache_id;
531
            } elseif ($object_to_cache->ID()) {
532
                // OR the cached object originally came from the db, so let's just use it's PK for an ID
533
                $this->_model_relations[$relationName][$object_to_cache->ID()] = $object_to_cache;
534
                $return = $object_to_cache->ID();
535
            } else {
536
                // OR it's a new object with no ID, so just throw it in the array with an auto-incremented ID
537
                $this->_model_relations[$relationName][] = $object_to_cache;
538
                // move the internal pointer to the end of the array
539
                end($this->_model_relations[$relationName]);
540
                // and grab the key so that we can return it
541
                $return = key($this->_model_relations[$relationName]);
542
            }
543
        }
544
        return $return;
545
    }
546
547
548
549
    /**
550
     * For adding an item to the cached_properties property.
551
     *
552
     * @access protected
553
     * @param string      $fieldname the property item the corresponding value is for.
554
     * @param mixed       $value     The value we are caching.
555
     * @param string|null $cache_type
556
     * @return void
557
     * @throws \EE_Error
558
     */
559
    protected function _set_cached_property($fieldname, $value, $cache_type = null)
560
    {
561
        //first make sure this property exists
562
        $this->get_model()->field_settings_for($fieldname);
563
        $cache_type = empty($cache_type) ? 'standard' : $cache_type;
564
        $this->_cached_properties[$fieldname][$cache_type] = $value;
565
    }
566
567
568
569
    /**
570
     * This returns the value cached property if it exists OR the actual property value if the cache doesn't exist.
571
     * This also SETS the cache if we return the actual property!
572
     *
573
     * @param string $fieldname        the name of the property we're trying to retrieve
574
     * @param bool   $pretty
575
     * @param string $extra_cache_ref  This allows the user to specify an extra cache ref for the given property
576
     *                                 (in cases where the same property may be used for different outputs
577
     *                                 - i.e. datetime, money etc.)
578
     *                                 It can also accept certain pre-defined "schema" strings
579
     *                                 to define how to output the property.
580
     *                                 see the field's prepare_for_pretty_echoing for what strings can be used
581
     * @return mixed                   whatever the value for the property is we're retrieving
582
     * @throws \EE_Error
583
     */
584
    protected function _get_cached_property($fieldname, $pretty = false, $extra_cache_ref = null)
585
    {
586
        //verify the field exists
587
        $model = $this->get_model();
588
        $model->field_settings_for($fieldname);
589
        $cache_type = $pretty ? 'pretty' : 'standard';
590
        $cache_type .= ! empty($extra_cache_ref) ? '_' . $extra_cache_ref : '';
591
        if (isset($this->_cached_properties[$fieldname][$cache_type])) {
592
            return $this->_cached_properties[$fieldname][$cache_type];
593
        }
594
        $value = $this->_get_fresh_property($fieldname, $pretty, $extra_cache_ref);
595
        $this->_set_cached_property($fieldname, $value, $cache_type);
596
        return $value;
597
    }
598
599
600
601
    /**
602
     * If the cache didn't fetch the needed item, this fetches it.
603
     * @param string $fieldname
604
     * @param bool $pretty
605
     * @param string $extra_cache_ref
606
     * @return mixed
607
     */
608
    protected function _get_fresh_property($fieldname, $pretty = false, $extra_cache_ref = null)
609
    {
610
        $field_obj = $this->get_model()->field_settings_for($fieldname);
611
        // If this is an EE_Datetime_Field we need to make sure timezone, formats, and output are correct
612
        if ($field_obj instanceof EE_Datetime_Field) {
613
            $this->_prepare_datetime_field($field_obj, $pretty, $extra_cache_ref);
614
        }
615
        if ( ! isset($this->_fields[$fieldname])) {
616
            $this->_fields[$fieldname] = null;
617
        }
618
        $value = $pretty
619
            ? $field_obj->prepare_for_pretty_echoing($this->_fields[$fieldname], $extra_cache_ref)
620
            : $field_obj->prepare_for_get($this->_fields[$fieldname]);
621
        return $value;
622
    }
623
624
625
626
    /**
627
     * set timezone, formats, and output for EE_Datetime_Field objects
628
     *
629
     * @param \EE_Datetime_Field $datetime_field
630
     * @param bool               $pretty
631
     * @param null $date_or_time
632
     * @return void
633
     * @throws \EE_Error
634
     */
635
    protected function _prepare_datetime_field(
636
        EE_Datetime_Field $datetime_field,
637
        $pretty = false,
638
        $date_or_time = null
639
    ) {
640
        $datetime_field->set_timezone($this->_timezone);
641
        $datetime_field->set_date_format($this->_dt_frmt, $pretty);
642
        $datetime_field->set_time_format($this->_tm_frmt, $pretty);
643
        //set the output returned
644
        switch ($date_or_time) {
645
            case 'D' :
646
                $datetime_field->set_date_time_output('date');
647
                break;
648
            case 'T' :
649
                $datetime_field->set_date_time_output('time');
650
                break;
651
            default :
652
                $datetime_field->set_date_time_output();
653
        }
654
    }
655
656
657
658
    /**
659
     * This just takes care of clearing out the cached_properties
660
     *
661
     * @return void
662
     */
663
    protected function _clear_cached_properties()
664
    {
665
        $this->_cached_properties = array();
666
    }
667
668
669
670
    /**
671
     * This just clears out ONE property if it exists in the cache
672
     *
673
     * @param  string $property_name the property to remove if it exists (from the _cached_properties array)
674
     * @return void
675
     */
676
    protected function _clear_cached_property($property_name)
677
    {
678
        if (isset($this->_cached_properties[$property_name])) {
679
            unset($this->_cached_properties[$property_name]);
680
        }
681
    }
682
683
684
685
    /**
686
     * Ensures that this related thing is a model object.
687
     *
688
     * @param mixed  $object_or_id EE_base_Class/int/string either a related model object, or its ID
689
     * @param string $model_name   name of the related thing, eg 'Attendee',
690
     * @return EE_Base_Class
691
     * @throws \EE_Error
692
     */
693
    protected function ensure_related_thing_is_model_obj($object_or_id, $model_name)
694
    {
695
        $other_model_instance = self::_get_model_instance_with_name(
696
            self::_get_model_classname($model_name),
697
            $this->_timezone
698
        );
699
        return $other_model_instance->ensure_is_obj($object_or_id);
700
    }
701
702
703
704
    /**
705
     * Forgets the cached model of the given relation Name. So the next time we request it,
706
     * we will fetch it again from the database. (Handy if you know it's changed somehow).
707
     * If a specific object is supplied, and the relationship to it is either a HasMany or HABTM,
708
     * then only remove that one object from our cached array. Otherwise, clear the entire list
709
     *
710
     * @param string $relationName                         one of the keys in the _model_relations array on the model.
711
     *                                                     Eg 'Registration'
712
     * @param mixed  $object_to_remove_or_index_into_array or an index into the array of cached things, or NULL
713
     *                                                     if you intend to use $clear_all = TRUE, or the relation only
714
     *                                                     has 1 object anyways (ie, it's a BelongsToRelation)
715
     * @param bool   $clear_all                            This flags clearing the entire cache relation property if
716
     *                                                     this is HasMany or HABTM.
717
     * @throws EE_Error
718
     * @return EE_Base_Class | boolean from which was cleared from the cache, or true if we requested to remove a
719
     *                       relation from all
720
     */
721
    public function clear_cache($relationName, $object_to_remove_or_index_into_array = null, $clear_all = false)
722
    {
723
        $relationship_to_model = $this->get_model()->related_settings_for($relationName);
724
        $index_in_cache = '';
725
        if ( ! $relationship_to_model) {
726
            throw new EE_Error(
727
                sprintf(
728
                    __("There is no relationship to %s on a %s. Cannot clear that cache", 'event_espresso'),
729
                    $relationName,
730
                    get_class($this)
731
                )
732
            );
733
        }
734
        if ($clear_all) {
735
            $obj_removed = true;
736
            $this->_model_relations[$relationName] = null;
737
        } elseif ($relationship_to_model instanceof EE_Belongs_To_Relation) {
738
            $obj_removed = $this->_model_relations[$relationName];
739
            $this->_model_relations[$relationName] = null;
740
        } else {
741
            if ($object_to_remove_or_index_into_array instanceof EE_Base_Class
742
                && $object_to_remove_or_index_into_array->ID()
743
            ) {
744
                $index_in_cache = $object_to_remove_or_index_into_array->ID();
745
                if (is_array($this->_model_relations[$relationName])
746
                    && ! isset($this->_model_relations[$relationName][$index_in_cache])
747
                ) {
748
                    $index_found_at = null;
749
                    //find this object in the array even though it has a different key
750
                    foreach ($this->_model_relations[$relationName] as $index => $obj) {
751
                        if (
752
                            $obj instanceof EE_Base_Class
753
                            && (
754
                                $obj == $object_to_remove_or_index_into_array
755
                                || $obj->ID() === $object_to_remove_or_index_into_array->ID()
756
                            )
757
                        ) {
758
                            $index_found_at = $index;
759
                            break;
760
                        }
761
                    }
762
                    if ($index_found_at) {
763
                        $index_in_cache = $index_found_at;
764
                    } else {
765
                        //it wasn't found. huh. well obviously it doesn't need to be removed from teh cache
766
                        //if it wasn't in it to begin with. So we're done
767
                        return $object_to_remove_or_index_into_array;
768
                    }
769
                }
770
            } elseif ($object_to_remove_or_index_into_array instanceof EE_Base_Class) {
771
                //so they provided a model object, but it's not yet saved to the DB... so let's go hunting for it!
772
                foreach ($this->get_all_from_cache($relationName) as $index => $potentially_obj_we_want) {
773
                    if ($potentially_obj_we_want == $object_to_remove_or_index_into_array) {
774
                        $index_in_cache = $index;
775
                    }
776
                }
777
            } else {
778
                $index_in_cache = $object_to_remove_or_index_into_array;
779
            }
780
            //supposedly we've found it. But it could just be that the client code
781
            //provided a bad index/object
782
            if (
783
            isset(
784
                $this->_model_relations[$relationName],
785
                $this->_model_relations[$relationName][$index_in_cache]
786
            )
787
            ) {
788
                $obj_removed = $this->_model_relations[$relationName][$index_in_cache];
789
                unset($this->_model_relations[$relationName][$index_in_cache]);
790
            } else {
791
                //that thing was never cached anyways.
792
                $obj_removed = null;
793
            }
794
        }
795
        return $obj_removed;
796
    }
797
798
799
800
    /**
801
     * update_cache_after_object_save
802
     * Allows a cached item to have it's cache ID (within the array of cached items) reset using the new ID it has
803
     * obtained after being saved to the db
804
     *
805
     * @param string         $relationName       - the type of object that is cached
806
     * @param \EE_Base_Class $newly_saved_object - the newly saved object to be re-cached
807
     * @param string         $current_cache_id   - the ID that was used when originally caching the object
808
     * @return boolean TRUE on success, FALSE on fail
809
     * @throws \EE_Error
810
     */
811
    public function update_cache_after_object_save(
812
        $relationName,
813
        EE_Base_Class $newly_saved_object,
814
        $current_cache_id = ''
815
    ) {
816
        // verify that incoming object is of the correct type
817
        $obj_class = 'EE_' . $relationName;
818
        if ($newly_saved_object instanceof $obj_class) {
819
            /* @type EE_Base_Class $newly_saved_object */
820
            // now get the type of relation
821
            $relationship_to_model = $this->get_model()->related_settings_for($relationName);
822
            // if this is a 1:1 relationship
823
            if ($relationship_to_model instanceof EE_Belongs_To_Relation) {
824
                // then just replace the cached object with the newly saved object
825
                $this->_model_relations[$relationName] = $newly_saved_object;
826
                return true;
827
                // or if it's some kind of sordid feral polyamorous relationship...
828
            } elseif (is_array($this->_model_relations[$relationName])
829
                      && isset($this->_model_relations[$relationName][$current_cache_id])
830
            ) {
831
                // then remove the current cached item
832
                unset($this->_model_relations[$relationName][$current_cache_id]);
833
                // and cache the newly saved object using it's new ID
834
                $this->_model_relations[$relationName][$newly_saved_object->ID()] = $newly_saved_object;
835
                return true;
836
            }
837
        }
838
        return false;
839
    }
840
841
842
843
    /**
844
     * Fetches a single EE_Base_Class on that relation. (If the relation is of type
845
     * BelongsTo, it will only ever have 1 object. However, other relations could have an array of objects)
846
     *
847
     * @param string $relationName
848
     * @return EE_Base_Class
849
     */
850
    public function get_one_from_cache($relationName)
851
    {
852
        $cached_array_or_object = isset($this->_model_relations[$relationName]) ? $this->_model_relations[$relationName]
853
            : null;
854
        if (is_array($cached_array_or_object)) {
855
            return array_shift($cached_array_or_object);
856
        } else {
857
            return $cached_array_or_object;
858
        }
859
    }
860
861
862
863
    /**
864
     * Fetches a single EE_Base_Class on that relation. (If the relation is of type
865
     * BelongsTo, it will only ever have 1 object. However, other relations could have an array of objects)
866
     *
867
     * @param string $relationName
868
     * @throws \EE_Error
869
     * @return EE_Base_Class[] NOT necessarily indexed by primary keys
870
     */
871
    public function get_all_from_cache($relationName)
872
    {
873
        $objects = isset($this->_model_relations[$relationName]) ? $this->_model_relations[$relationName] : array();
874
        // if the result is not an array, but exists, make it an array
875
        $objects = is_array($objects) ? $objects : array($objects);
876
        //bugfix for https://events.codebasehq.com/projects/event-espresso/tickets/7143
877
        //basically, if this model object was stored in the session, and these cached model objects
878
        //already have IDs, let's make sure they're in their model's entity mapper
879
        //otherwise we will have duplicates next time we call
880
        // EE_Registry::instance()->load_model( $relationName )->get_one_by_ID( $result->ID() );
881
        $model = EE_Registry::instance()->load_model($relationName);
882
        foreach ($objects as $model_object) {
883
            if ($model instanceof EEM_Base && $model_object instanceof EE_Base_Class) {
884
                //ensure its in the map if it has an ID; otherwise it will be added to the map when its saved
885
                if ($model_object->ID()) {
886
                    $model->add_to_entity_map($model_object);
887
                }
888
            } else {
889
                throw new EE_Error(
890
                    sprintf(
891
                        __(
892
                            'Error retrieving related model objects. Either $1%s is not a model or $2%s is not a model object',
893
                            'event_espresso'
894
                        ),
895
                        $relationName,
896
                        gettype($model_object)
897
                    )
898
                );
899
            }
900
        }
901
        return $objects;
902
    }
903
904
905
906
    /**
907
     * Returns the next x number of EE_Base_Class objects in sequence from this object as found in the database
908
     * matching the given query conditions.
909
     *
910
     * @param null  $field_to_order_by  What field is being used as the reference point.
911
     * @param int   $limit              How many objects to return.
912
     * @param array $query_params       Any additional conditions on the query.
913
     * @param null  $columns_to_select  If left null, then an array of EE_Base_Class objects is returned, otherwise
914
     *                                  you can indicate just the columns you want returned
915
     * @return array|EE_Base_Class[]
916
     * @throws \EE_Error
917
     */
918 View Code Duplication
    public function next_x($field_to_order_by = null, $limit = 1, $query_params = array(), $columns_to_select = null)
919
    {
920
        $model = $this->get_model();
921
        $field = empty($field_to_order_by) && $model->has_primary_key_field()
922
            ? $model->get_primary_key_field()->get_name()
923
            : $field_to_order_by;
924
        $current_value = ! empty($field) ? $this->get($field) : null;
925
        if (empty($field) || empty($current_value)) {
926
            return array();
927
        }
928
        return $model->next_x($current_value, $field, $limit, $query_params, $columns_to_select);
929
    }
930
931
932
933
    /**
934
     * Returns the previous x number of EE_Base_Class objects in sequence from this object as found in the database
935
     * matching the given query conditions.
936
     *
937
     * @param null  $field_to_order_by  What field is being used as the reference point.
938
     * @param int   $limit              How many objects to return.
939
     * @param array $query_params       Any additional conditions on the query.
940
     * @param null  $columns_to_select  If left null, then an array of EE_Base_Class objects is returned, otherwise
941
     *                                  you can indicate just the columns you want returned
942
     * @return array|EE_Base_Class[]
943
     * @throws \EE_Error
944
     */
945 View Code Duplication
    public function previous_x(
946
        $field_to_order_by = null,
947
        $limit = 1,
948
        $query_params = array(),
949
        $columns_to_select = null
950
    ) {
951
        $model = $this->get_model();
952
        $field = empty($field_to_order_by) && $model->has_primary_key_field()
953
            ? $model->get_primary_key_field()->get_name()
954
            : $field_to_order_by;
955
        $current_value = ! empty($field) ? $this->get($field) : null;
956
        if (empty($field) || empty($current_value)) {
957
            return array();
958
        }
959
        return $model->previous_x($current_value, $field, $limit, $query_params, $columns_to_select);
960
    }
961
962
963
964
    /**
965
     * Returns the next EE_Base_Class object in sequence from this object as found in the database
966
     * matching the given query conditions.
967
     *
968
     * @param null  $field_to_order_by  What field is being used as the reference point.
969
     * @param array $query_params       Any additional conditions on the query.
970
     * @param null  $columns_to_select  If left null, then an array of EE_Base_Class objects is returned, otherwise
971
     *                                  you can indicate just the columns you want returned
972
     * @return array|EE_Base_Class
973
     * @throws \EE_Error
974
     */
975 View Code Duplication
    public function next($field_to_order_by = null, $query_params = array(), $columns_to_select = null)
976
    {
977
        $model = $this->get_model();
978
        $field = empty($field_to_order_by) && $model->has_primary_key_field()
979
            ? $model->get_primary_key_field()->get_name()
980
            : $field_to_order_by;
981
        $current_value = ! empty($field) ? $this->get($field) : null;
982
        if (empty($field) || empty($current_value)) {
983
            return array();
984
        }
985
        return $model->next($current_value, $field, $query_params, $columns_to_select);
986
    }
987
988
989
990
    /**
991
     * Returns the previous EE_Base_Class object in sequence from this object as found in the database
992
     * matching the given query conditions.
993
     *
994
     * @param null  $field_to_order_by  What field is being used as the reference point.
995
     * @param array $query_params       Any additional conditions on the query.
996
     * @param null  $columns_to_select  If left null, then an EE_Base_Class object is returned, otherwise
997
     *                                  you can indicate just the column you want returned
998
     * @return array|EE_Base_Class
999
     * @throws \EE_Error
1000
     */
1001 View Code Duplication
    public function previous($field_to_order_by = null, $query_params = array(), $columns_to_select = null)
1002
    {
1003
        $model = $this->get_model();
1004
        $field = empty($field_to_order_by) && $model->has_primary_key_field()
1005
            ? $model->get_primary_key_field()->get_name()
1006
            : $field_to_order_by;
1007
        $current_value = ! empty($field) ? $this->get($field) : null;
1008
        if (empty($field) || empty($current_value)) {
1009
            return array();
1010
        }
1011
        return $model->previous($current_value, $field, $query_params, $columns_to_select);
1012
    }
1013
1014
1015
1016
    /**
1017
     * Overrides parent because parent expects old models.
1018
     * This also doesn't do any validation, and won't work for serialized arrays
1019
     *
1020
     * @param string $field_name
1021
     * @param mixed  $field_value_from_db
1022
     * @throws \EE_Error
1023
     */
1024
    public function set_from_db($field_name, $field_value_from_db)
1025
    {
1026
        $field_obj = $this->get_model()->field_settings_for($field_name);
1027
        if ($field_obj instanceof EE_Model_Field_Base) {
1028
            //you would think the DB has no NULLs for non-null label fields right? wrong!
1029
            //eg, a CPT model object could have an entry in the posts table, but no
1030
            //entry in the meta table. Meaning that all its columns in the meta table
1031
            //are null! yikes! so when we find one like that, use defaults for its meta columns
1032
            if ($field_value_from_db === null) {
1033
                if ($field_obj->is_nullable()) {
1034
                    //if the field allows nulls, then let it be null
1035
                    $field_value = null;
1036
                } else {
1037
                    $field_value = $field_obj->get_default_value();
1038
                }
1039
            } else {
1040
                $field_value = $field_obj->prepare_for_set_from_db($field_value_from_db);
1041
            }
1042
            $this->_fields[$field_name] = $field_value;
1043
            $this->_clear_cached_property($field_name);
1044
        }
1045
    }
1046
1047
1048
1049
    /**
1050
     * verifies that the specified field is of the correct type
1051
     *
1052
     * @param string $field_name
1053
     * @param string $extra_cache_ref This allows the user to specify an extra cache ref for the given property
1054
     *                                (in cases where the same property may be used for different outputs
1055
     *                                - i.e. datetime, money etc.)
1056
     * @return mixed
1057
     * @throws \EE_Error
1058
     */
1059
    public function get($field_name, $extra_cache_ref = null)
1060
    {
1061
        return $this->_get_cached_property($field_name, false, $extra_cache_ref);
1062
    }
1063
1064
1065
1066
    /**
1067
     * This method simply returns the RAW unprocessed value for the given property in this class
1068
     *
1069
     * @param  string $field_name A valid fieldname
1070
     * @return mixed              Whatever the raw value stored on the property is.
1071
     * @throws EE_Error if fieldSettings is misconfigured or the field doesn't exist.
1072
     */
1073
    public function get_raw($field_name)
1074
    {
1075
        $field_settings = $this->get_model()->field_settings_for($field_name);
1076
        return $field_settings instanceof EE_Datetime_Field && $this->_fields[$field_name] instanceof DateTime
1077
            ? $this->_fields[$field_name]->format('U')
1078
            : $this->_fields[$field_name];
1079
    }
1080
1081
1082
1083
    /**
1084
     * This is used to return the internal DateTime object used for a field that is a
1085
     * EE_Datetime_Field.
1086
     *
1087
     * @param string $field_name               The field name retrieving the DateTime object.
1088
     * @return mixed null | false | DateTime  If the requested field is NOT a EE_Datetime_Field then
1089
     * @throws \EE_Error
1090
     *                                         an error is set and false returned.  If the field IS an
1091
     *                                         EE_Datetime_Field and but the field value is null, then
1092
     *                                         just null is returned (because that indicates that likely
1093
     *                                         this field is nullable).
1094
     */
1095
    public function get_DateTime_object($field_name)
1096
    {
1097
        $field_settings = $this->get_model()->field_settings_for($field_name);
1098 View Code Duplication
        if ( ! $field_settings instanceof EE_Datetime_Field) {
1099
            EE_Error::add_error(
1100
                sprintf(
1101
                    __(
1102
                        'The field %s is not an EE_Datetime_Field field.  There is no DateTime object stored on this field type.',
1103
                        'event_espresso'
1104
                    ),
1105
                    $field_name
1106
                ),
1107
                __FILE__,
1108
                __FUNCTION__,
1109
                __LINE__
1110
            );
1111
            return false;
1112
        }
1113
        return $this->_fields[$field_name];
1114
    }
1115
1116
1117
1118
    /**
1119
     * To be used in template to immediately echo out the value, and format it for output.
1120
     * Eg, should call stripslashes and whatnot before echoing
1121
     *
1122
     * @param string $field_name      the name of the field as it appears in the DB
1123
     * @param string $extra_cache_ref This allows the user to specify an extra cache ref for the given property
1124
     *                                (in cases where the same property may be used for different outputs
1125
     *                                - i.e. datetime, money etc.)
1126
     * @return void
1127
     * @throws \EE_Error
1128
     */
1129
    public function e($field_name, $extra_cache_ref = null)
1130
    {
1131
        echo $this->get_pretty($field_name, $extra_cache_ref);
1132
    }
1133
1134
1135
1136
    /**
1137
     * Exactly like e(), echoes out the field, but sets its schema to 'form_input', so that it
1138
     * can be easily used as the value of form input.
1139
     *
1140
     * @param string $field_name
1141
     * @return void
1142
     * @throws \EE_Error
1143
     */
1144
    public function f($field_name)
1145
    {
1146
        $this->e($field_name, 'form_input');
1147
    }
1148
1149
    /**
1150
     * Same as `f()` but just returns the value instead of echoing it
1151
     * @param string $field_name
1152
     * @return string
1153
     */
1154
    public function get_f($field_name)
1155
    {
1156
        return (string)$this->get_pretty($field_name,'form_input');
1157
    }
1158
1159
1160
1161
    /**
1162
     * Gets a pretty view of the field's value. $extra_cache_ref can specify different formats for this.
1163
     * The $extra_cache_ref will be passed to the model field's prepare_for_pretty_echoing, so consult the field's class
1164
     * to see what options are available.
1165
     * @param string $field_name
1166
     * @param string $extra_cache_ref This allows the user to specify an extra cache ref for the given property
1167
     *                                (in cases where the same property may be used for different outputs
1168
     *                                - i.e. datetime, money etc.)
1169
     * @return mixed
1170
     * @throws \EE_Error
1171
     */
1172
    public function get_pretty($field_name, $extra_cache_ref = null)
1173
    {
1174
        return $this->_get_cached_property($field_name, true, $extra_cache_ref);
1175
    }
1176
1177
1178
1179
    /**
1180
     * This simply returns the datetime for the given field name
1181
     * Note: this protected function is called by the wrapper get_date or get_time or get_datetime functions
1182
     * (and the equivalent e_date, e_time, e_datetime).
1183
     *
1184
     * @access   protected
1185
     * @param string   $field_name   Field on the instantiated EE_Base_Class child object
1186
     * @param string   $dt_frmt      valid datetime format used for date
1187
     *                               (if '' then we just use the default on the field,
1188
     *                               if NULL we use the last-used format)
1189
     * @param string   $tm_frmt      Same as above except this is for time format
1190
     * @param string   $date_or_time if NULL then both are returned, otherwise "D" = only date and "T" = only time.
1191
     * @param  boolean $echo         Whether the dtt is echoing using pretty echoing or just returned using vanilla get
1192
     * @return string|bool|EE_Error string on success, FALSE on fail, or EE_Error Exception is thrown
1193
     *                               if field is not a valid dtt field, or void if echoing
1194
     * @throws \EE_Error
1195
     */
1196
    protected function _get_datetime($field_name, $dt_frmt = '', $tm_frmt = '', $date_or_time = '', $echo = false)
1197
    {
1198
        // clear cached property
1199
        $this->_clear_cached_property($field_name);
1200
        //reset format properties because they are used in get()
1201
        $this->_dt_frmt = $dt_frmt !== '' ? $dt_frmt : $this->_dt_frmt;
1202
        $this->_tm_frmt = $tm_frmt !== '' ? $tm_frmt : $this->_tm_frmt;
1203
        if ($echo) {
1204
            $this->e($field_name, $date_or_time);
1205
            return '';
1206
        }
1207
        return $this->get($field_name, $date_or_time);
1208
    }
1209
1210
1211
1212
    /**
1213
     * below are wrapper functions for the various datetime outputs that can be obtained for JUST returning the date
1214
     * portion of a datetime value. (note the only difference between get_ and e_ is one returns the value and the
1215
     * other echoes the pretty value for dtt)
1216
     *
1217
     * @param  string $field_name name of model object datetime field holding the value
1218
     * @param  string $format     format for the date returned (if NULL we use default in dt_frmt property)
1219
     * @return string            datetime value formatted
1220
     * @throws \EE_Error
1221
     */
1222
    public function get_date($field_name, $format = '')
1223
    {
1224
        return $this->_get_datetime($field_name, $format, null, 'D');
1225
    }
1226
1227
1228
1229
    /**
1230
     * @param      $field_name
1231
     * @param string $format
1232
     * @throws \EE_Error
1233
     */
1234
    public function e_date($field_name, $format = '')
1235
    {
1236
        $this->_get_datetime($field_name, $format, null, 'D', true);
1237
    }
1238
1239
1240
1241
    /**
1242
     * below are wrapper functions for the various datetime outputs that can be obtained for JUST returning the time
1243
     * portion of a datetime value. (note the only difference between get_ and e_ is one returns the value and the
1244
     * other echoes the pretty value for dtt)
1245
     *
1246
     * @param  string $field_name name of model object datetime field holding the value
1247
     * @param  string $format     format for the time returned ( if NULL we use default in tm_frmt property)
1248
     * @return string             datetime value formatted
1249
     * @throws \EE_Error
1250
     */
1251
    public function get_time($field_name, $format = '')
1252
    {
1253
        return $this->_get_datetime($field_name, null, $format, 'T');
1254
    }
1255
1256
1257
1258
    /**
1259
     * @param      $field_name
1260
     * @param string $format
1261
     * @throws \EE_Error
1262
     */
1263
    public function e_time($field_name, $format = '')
1264
    {
1265
        $this->_get_datetime($field_name, null, $format, 'T', true);
1266
    }
1267
1268
1269
1270
    /**
1271
     * below are wrapper functions for the various datetime outputs that can be obtained for returning the date AND
1272
     * time portion of a datetime value. (note the only difference between get_ and e_ is one returns the value and the
1273
     * other echoes the pretty value for dtt)
1274
     *
1275
     * @param  string $field_name name of model object datetime field holding the value
1276
     * @param  string $dt_frmt    format for the date returned (if NULL we use default in dt_frmt property)
1277
     * @param  string $tm_frmt    format for the time returned (if NULL we use default in tm_frmt property)
1278
     * @return string             datetime value formatted
1279
     * @throws \EE_Error
1280
     */
1281
    public function get_datetime($field_name, $dt_frmt = '', $tm_frmt = '')
1282
    {
1283
        return $this->_get_datetime($field_name, $dt_frmt, $tm_frmt);
1284
    }
1285
1286
1287
1288
    /**
1289
     * @param string $field_name
1290
     * @param string $dt_frmt
1291
     * @param string $tm_frmt
1292
     * @throws \EE_Error
1293
     */
1294
    public function e_datetime($field_name, $dt_frmt = '', $tm_frmt = '')
1295
    {
1296
        $this->_get_datetime($field_name, $dt_frmt, $tm_frmt, null, true);
1297
    }
1298
1299
1300
1301
    /**
1302
     * Get the i8ln value for a date using the WordPress @see date_i18n function.
1303
     *
1304
     * @param string $field_name The EE_Datetime_Field reference for the date being retrieved.
1305
     * @param string $format     PHP valid date/time string format.  If none is provided then the internal set format
1306
     *                           on the object will be used.
1307
     * @return string Date and time string in set locale or false if no field exists for the given
1308
     * @throws \EE_Error
1309
     *                           field name.
1310
     */
1311
    public function get_i18n_datetime($field_name, $format = '')
1312
    {
1313
        $format = empty($format) ? $this->_dt_frmt . ' ' . $this->_tm_frmt : $format;
1314
        return date_i18n(
1315
            $format,
1316
            EEH_DTT_Helper::get_timestamp_with_offset($this->get_raw($field_name), $this->_timezone)
1317
        );
1318
    }
1319
1320
1321
1322
    /**
1323
     * This method validates whether the given field name is a valid field on the model object as well as it is of a
1324
     * type EE_Datetime_Field.  On success there will be returned the field settings.  On fail an EE_Error exception is
1325
     * thrown.
1326
     *
1327
     * @param  string $field_name The field name being checked
1328
     * @throws EE_Error
1329
     * @return EE_Datetime_Field
1330
     */
1331
    protected function _get_dtt_field_settings($field_name)
1332
    {
1333
        $field = $this->get_model()->field_settings_for($field_name);
1334
        //check if field is dtt
1335
        if ($field instanceof EE_Datetime_Field) {
1336
            return $field;
1337
        } else {
1338
            throw new EE_Error(sprintf(__('The field name "%s" has been requested for the EE_Base_Class datetime functions and it is not a valid EE_Datetime_Field.  Please check the spelling of the field and make sure it has been setup as a EE_Datetime_Field in the %s model constructor',
1339
                'event_espresso'), $field_name, self::_get_model_classname(get_class($this))));
1340
        }
1341
    }
1342
1343
1344
1345
1346
    /**
1347
     * NOTE ABOUT BELOW:
1348
     * These convenience date and time setters are for setting date and time independently.  In other words you might
1349
     * want to change the time on a datetime_field but leave the date the same (or vice versa). IF on the other hand
1350
     * you want to set both date and time at the same time, you can just use the models default set($fieldname,$value)
1351
     * method and make sure you send the entire datetime value for setting.
1352
     */
1353
    /**
1354
     * sets the time on a datetime property
1355
     *
1356
     * @access protected
1357
     * @param string|Datetime $time      a valid time string for php datetime functions (or DateTime object)
1358
     * @param string          $fieldname the name of the field the time is being set on (must match a EE_Datetime_Field)
1359
     * @throws \EE_Error
1360
     */
1361
    protected function _set_time_for($time, $fieldname)
1362
    {
1363
        $this->_set_date_time('T', $time, $fieldname);
1364
    }
1365
1366
1367
1368
    /**
1369
     * sets the date on a datetime property
1370
     *
1371
     * @access protected
1372
     * @param string|DateTime $date      a valid date string for php datetime functions ( or DateTime object)
1373
     * @param string          $fieldname the name of the field the date is being set on (must match a EE_Datetime_Field)
1374
     * @throws \EE_Error
1375
     */
1376
    protected function _set_date_for($date, $fieldname)
1377
    {
1378
        $this->_set_date_time('D', $date, $fieldname);
1379
    }
1380
1381
1382
1383
    /**
1384
     * This takes care of setting a date or time independently on a given model object property. This method also
1385
     * verifies that the given fieldname matches a model object property and is for a EE_Datetime_Field field
1386
     *
1387
     * @access protected
1388
     * @param string          $what           "T" for time, 'B' for both, 'D' for Date.
1389
     * @param string|DateTime $datetime_value A valid Date or Time string (or DateTime object)
1390
     * @param string          $fieldname      the name of the field the date OR time is being set on (must match a
1391
     *                                        EE_Datetime_Field property)
1392
     * @throws \EE_Error
1393
     */
1394
    protected function _set_date_time($what = 'T', $datetime_value, $fieldname)
1395
    {
1396
        $field = $this->_get_dtt_field_settings($fieldname);
1397
        $field->set_timezone($this->_timezone);
1398
        $field->set_date_format($this->_dt_frmt);
1399
        $field->set_time_format($this->_tm_frmt);
1400
        switch ($what) {
1401
            case 'T' :
1402
                $this->_fields[$fieldname] = $field->prepare_for_set_with_new_time(
1403
                    $datetime_value,
1404
                    $this->_fields[$fieldname]
1405
                );
1406
                break;
1407
            case 'D' :
1408
                $this->_fields[$fieldname] = $field->prepare_for_set_with_new_date(
1409
                    $datetime_value,
1410
                    $this->_fields[$fieldname]
1411
                );
1412
                break;
1413
            case 'B' :
1414
                $this->_fields[$fieldname] = $field->prepare_for_set($datetime_value);
1415
                break;
1416
        }
1417
        $this->_clear_cached_property($fieldname);
1418
    }
1419
1420
1421
1422
    /**
1423
     * This will return a timestamp for the website timezone but ONLY when the current website timezone is different
1424
     * than the timezone set for the website. NOTE, this currently only works well with methods that return values.  If
1425
     * you use it with methods that echo values the $_timestamp property may not get reset to its original value and
1426
     * that could lead to some unexpected results!
1427
     *
1428
     * @access public
1429
     * @param string               $field_name This is the name of the field on the object that contains the date/time
1430
     *                                         value being returned.
1431
     * @param string               $callback   must match a valid method in this class (defaults to get_datetime)
1432
     * @param mixed (array|string) $args       This is the arguments that will be passed to the callback.
1433
     * @param string               $prepend    You can include something to prepend on the timestamp
1434
     * @param string               $append     You can include something to append on the timestamp
1435
     * @throws EE_Error
1436
     * @return string timestamp
1437
     */
1438
    public function display_in_my_timezone(
1439
        $field_name,
1440
        $callback = 'get_datetime',
1441
        $args = null,
1442
        $prepend = '',
1443
        $append = ''
1444
    ) {
1445
        $timezone = EEH_DTT_Helper::get_timezone();
1446
        if ($timezone === $this->_timezone) {
1447
            return '';
1448
        }
1449
        $original_timezone = $this->_timezone;
1450
        $this->set_timezone($timezone);
1451
        $fn = (array)$field_name;
1452
        $args = array_merge($fn, (array)$args);
1453 View Code Duplication
        if ( ! method_exists($this, $callback)) {
1454
            throw new EE_Error(
1455
                sprintf(
1456
                    __(
1457
                        'The method named "%s" given as the callback param in "display_in_my_timezone" does not exist.  Please check your spelling',
1458
                        'event_espresso'
1459
                    ),
1460
                    $callback
1461
                )
1462
            );
1463
        }
1464
        $args = (array)$args;
1465
        $return = $prepend . call_user_func_array(array($this, $callback), $args) . $append;
1466
        $this->set_timezone($original_timezone);
1467
        return $return;
1468
    }
1469
1470
1471
1472
    /**
1473
     * Deletes this model object.
1474
     * This calls the `EE_Base_Class::_delete` method.  Child classes wishing to change default behaviour should
1475
     * override
1476
     * `EE_Base_Class::_delete` NOT this class.
1477
     *
1478
     * @return boolean | int
1479
     * @throws \EE_Error
1480
     */
1481
    public function delete()
1482
    {
1483
        /**
1484
         * Called just before the `EE_Base_Class::_delete` method call.
1485
         * Note: `EE_Base_Class::_delete` might be overridden by child classes so any client code hooking into these actions
1486
         * should be aware that `_delete` may not always result in a permanent delete.  For example, `EE_Soft_Delete_Base_Class::_delete`
1487
         * soft deletes (trash) the object and does not permanently delete it.
1488
         *
1489
         * @param EE_Base_Class $model_object about to be 'deleted'
1490
         */
1491
        do_action('AHEE__EE_Base_Class__delete__before', $this);
1492
        $result = $this->_delete();
1493
        /**
1494
         * Called just after the `EE_Base_Class::_delete` method call.
1495
         * Note: `EE_Base_Class::_delete` might be overridden by child classes so any client code hooking into these actions
1496
         * should be aware that `_delete` may not always result in a permanent delete.  For example `EE_Soft_Base_Class::_delete`
1497
         * soft deletes (trash) the object and does not permanently delete it.
1498
         *
1499
         * @param EE_Base_Class $model_object that was just 'deleted'
1500
         * @param boolean       $result
1501
         */
1502
        do_action('AHEE__EE_Base_Class__delete__end', $this, $result);
1503
        return $result;
1504
    }
1505
1506
1507
1508
    /**
1509
     * Calls the specific delete method for the instantiated class.
1510
     * This method is called by the public `EE_Base_Class::delete` method.  Any child classes desiring to override
1511
     * default functionality for "delete" (which is to call `permanently_delete`) should override this method NOT
1512
     * `EE_Base_Class::delete`
1513
     *
1514
     * @return bool|int
1515
     * @throws \EE_Error
1516
     */
1517
    protected function _delete()
1518
    {
1519
        return $this->delete_permanently();
1520
    }
1521
1522
1523
1524
    /**
1525
     * Deletes this model object permanently from db (but keep in mind related models my block the delete and return an
1526
     * error)
1527
     *
1528
     * @return bool | int
1529
     * @throws \EE_Error
1530
     */
1531
    public function delete_permanently()
1532
    {
1533
        /**
1534
         * Called just before HARD deleting a model object
1535
         *
1536
         * @param EE_Base_Class $model_object about to be 'deleted'
1537
         */
1538
        do_action('AHEE__EE_Base_Class__delete_permanently__before', $this);
1539
        $model = $this->get_model();
1540
        $result = $model->delete_permanently_by_ID($this->ID());
1541
        $this->refresh_cache_of_related_objects();
1542
        /**
1543
         * Called just after HARD deleting a model object
1544
         *
1545
         * @param EE_Base_Class $model_object that was just 'deleted'
1546
         * @param boolean       $result
1547
         */
1548
        do_action('AHEE__EE_Base_Class__delete_permanently__end', $this, $result);
1549
        return $result;
1550
    }
1551
1552
1553
1554
    /**
1555
     * When this model object is deleted, it may still be cached on related model objects. This clears the cache of
1556
     * related model objects
1557
     *
1558
     * @throws \EE_Error
1559
     */
1560
    public function refresh_cache_of_related_objects()
1561
    {
1562
        $model = $this->get_model();
1563
        foreach ($model->relation_settings() as $relation_name => $relation_obj) {
1564
            if ( ! empty($this->_model_relations[$relation_name])) {
1565
                $related_objects = $this->_model_relations[$relation_name];
1566
                if ($relation_obj instanceof EE_Belongs_To_Relation) {
1567
                    //this relation only stores a single model object, not an array
1568
                    //but let's make it consistent
1569
                    $related_objects = array($related_objects);
1570
                }
1571
                foreach ($related_objects as $related_object) {
1572
                    //only refresh their cache if they're in memory
1573
                    if ($related_object instanceof EE_Base_Class) {
1574
                        $related_object->clear_cache($model->get_this_model_name(), $this);
1575
                    }
1576
                }
1577
            }
1578
        }
1579
    }
1580
1581
1582
1583
    /**
1584
     *        Saves this object to the database. An array may be supplied to set some values on this
1585
     * object just before saving.
1586
     *
1587
     * @access public
1588
     * @param array $set_cols_n_values keys are field names, values are their new values,
1589
     *                                 if provided during the save() method (often client code will change the fields'
1590
     *                                 values before calling save)
1591
     * @throws \EE_Error
1592
     * @return int , 1 on a successful update, the ID of the new entry on insert; 0 on failure or if the model object
1593
     *                                 isn't allowed to persist (as determined by EE_Base_Class::allow_persist())
1594
     */
1595
    public function save($set_cols_n_values = array())
1596
    {
1597
        $model = $this->get_model();
1598
        /**
1599
         * Filters the fields we're about to save on the model object
1600
         *
1601
         * @param array         $set_cols_n_values
1602
         * @param EE_Base_Class $model_object
1603
         */
1604
        $set_cols_n_values = (array)apply_filters('FHEE__EE_Base_Class__save__set_cols_n_values', $set_cols_n_values,
1605
            $this);
1606
        //set attributes as provided in $set_cols_n_values
1607
        foreach ($set_cols_n_values as $column => $value) {
1608
            $this->set($column, $value);
1609
        }
1610
        // no changes ? then don't do anything
1611
        if (! $this->_has_changes && $this->ID() && $model->get_primary_key_field()->is_auto_increment()) {
1612
            return 0;
1613
        }
1614
        /**
1615
         * Saving a model object.
1616
         * Before we perform a save, this action is fired.
1617
         *
1618
         * @param EE_Base_Class $model_object the model object about to be saved.
1619
         */
1620
        do_action('AHEE__EE_Base_Class__save__begin', $this);
1621
        if ( ! $this->allow_persist()) {
1622
            return 0;
1623
        }
1624
        //now get current attribute values
1625
        $save_cols_n_values = $this->_fields;
1626
        //if the object already has an ID, update it. Otherwise, insert it
1627
        //also: change the assumption about values passed to the model NOT being prepare dby the model object. They have been
1628
        $old_assumption_concerning_value_preparation = $model
1629
                                                            ->get_assumption_concerning_values_already_prepared_by_model_object();
1630
        $model->assume_values_already_prepared_by_model_object(true);
1631
        //does this model have an autoincrement PK?
1632
        if ($model->has_primary_key_field()) {
1633
            if ($model->get_primary_key_field()->is_auto_increment()) {
1634
                //ok check if it's set, if so: update; if not, insert
1635
                if ( ! empty($save_cols_n_values[$model->primary_key_name()])) {
1636
                    $results = $model->update_by_ID($save_cols_n_values, $this->ID());
1637
                } else {
1638
                    unset($save_cols_n_values[$model->primary_key_name()]);
1639
                    $results = $model->insert($save_cols_n_values);
1640
                    if ($results) {
1641
                        //if successful, set the primary key
1642
                        //but don't use the normal SET method, because it will check if
1643
                        //an item with the same ID exists in the mapper & db, then
1644
                        //will find it in the db (because we just added it) and THAT object
1645
                        //will get added to the mapper before we can add this one!
1646
                        //but if we just avoid using the SET method, all that headache can be avoided
1647
                        $pk_field_name = $model->primary_key_name();
1648
                        $this->_fields[$pk_field_name] = $results;
1649
                        $this->_clear_cached_property($pk_field_name);
1650
                        $model->add_to_entity_map($this);
1651
                        $this->_update_cached_related_model_objs_fks();
1652
                    }
1653
                }
1654
            } else {//PK is NOT auto-increment
1655
                //so check if one like it already exists in the db
1656
                if ($model->exists_by_ID($this->ID())) {
1657
                    if (WP_DEBUG && ! $this->in_entity_map()) {
1658
                        throw new EE_Error(
1659
                            sprintf(
1660
                                __('Using a model object %1$s that is NOT in the entity map, can lead to unexpected errors. You should either: %4$s 1. Put it in the entity mapper by calling %2$s %4$s 2. Discard this model object and use what is in the entity mapper %4$s 3. Fetch from the database using %3$s',
1661
                                    'event_espresso'),
1662
                                get_class($this),
1663
                                get_class($model) . '::instance()->add_to_entity_map()',
1664
                                get_class($model) . '::instance()->get_one_by_ID()',
1665
                                '<br />'
1666
                            )
1667
                        );
1668
                    }
1669
                    $results = $model->update_by_ID($save_cols_n_values, $this->ID());
1670
                } else {
1671
                    $results = $model->insert($save_cols_n_values);
1672
                    $this->_update_cached_related_model_objs_fks();
1673
                }
1674
            }
1675
        } else {//there is NO primary key
1676
            $already_in_db = false;
1677
            foreach ($model->unique_indexes() as $index) {
1678
                $uniqueness_where_params = array_intersect_key($save_cols_n_values, $index->fields());
1679
                if ($model->exists(array($uniqueness_where_params))) {
1680
                    $already_in_db = true;
1681
                }
1682
            }
1683
            if ($already_in_db) {
1684
                $combined_pk_fields_n_values = array_intersect_key($save_cols_n_values,
1685
                    $model->get_combined_primary_key_fields());
1686
                $results = $model->update($save_cols_n_values, $combined_pk_fields_n_values);
1687
            } else {
1688
                $results = $model->insert($save_cols_n_values);
1689
            }
1690
        }
1691
        //restore the old assumption about values being prepared by the model object
1692
        $model
1693
             ->assume_values_already_prepared_by_model_object($old_assumption_concerning_value_preparation);
1694
        /**
1695
         * After saving the model object this action is called
1696
         *
1697
         * @param EE_Base_Class $model_object which was just saved
1698
         * @param boolean|int   $results      if it were updated, TRUE or FALSE; if it were newly inserted
1699
         *                                    the new ID (or 0 if an error occurred and it wasn't updated)
1700
         */
1701
        do_action('AHEE__EE_Base_Class__save__end', $this, $results);
1702
        $this->_has_changes = false;
1703
        return $results;
1704
    }
1705
1706
1707
1708
    /**
1709
     * Updates the foreign key on related models objects pointing to this to have this model object's ID
1710
     * as their foreign key.  If the cached related model objects already exist in the db, saves them (so that the DB
1711
     * is consistent) Especially useful in case we JUST added this model object ot the database and we want to let its
1712
     * cached relations with foreign keys to it know about that change. Eg: we've created a transaction but haven't
1713
     * saved it to the db. We also create a registration and don't save it to the DB, but we DO cache it on the
1714
     * transaction. Now, when we save the transaction, the registration's TXN_ID will be automatically updated, whether
1715
     * or not they exist in the DB (if they do, their DB records will be automatically updated)
1716
     *
1717
     * @return void
1718
     * @throws \EE_Error
1719
     */
1720
    protected function _update_cached_related_model_objs_fks()
1721
    {
1722
        $model = $this->get_model();
1723
        foreach ($model->relation_settings() as $relation_name => $relation_obj) {
1724
            if ($relation_obj instanceof EE_Has_Many_Relation) {
1725
                foreach ($this->get_all_from_cache($relation_name) as $related_model_obj_in_cache) {
1726
                    $fk_to_this = $related_model_obj_in_cache->get_model()->get_foreign_key_to(
1727
                        $model->get_this_model_name()
1728
                    );
1729
                    $related_model_obj_in_cache->set($fk_to_this->get_name(), $this->ID());
1730
                    if ($related_model_obj_in_cache->ID()) {
1731
                        $related_model_obj_in_cache->save();
1732
                    }
1733
                }
1734
            }
1735
        }
1736
    }
1737
1738
1739
1740
    /**
1741
     * Saves this model object and its NEW cached relations to the database.
1742
     * (Meaning, for now, IT DOES NOT WORK if the cached items already exist in the DB.
1743
     * In order for that to work, we would need to mark model objects as dirty/clean...
1744
     * because otherwise, there's a potential for infinite looping of saving
1745
     * Saves the cached related model objects, and ensures the relation between them
1746
     * and this object and properly setup
1747
     *
1748
     * @return int ID of new model object on save; 0 on failure+
1749
     * @throws \EE_Error
1750
     */
1751
    public function save_new_cached_related_model_objs()
1752
    {
1753
        //make sure this has been saved
1754
        if ( ! $this->ID()) {
1755
            $id = $this->save();
1756
        } else {
1757
            $id = $this->ID();
1758
        }
1759
        //now save all the NEW cached model objects  (ie they don't exist in the DB)
1760
        foreach ($this->get_model()->relation_settings() as $relationName => $relationObj) {
1761
            if ($this->_model_relations[$relationName]) {
1762
                //is this a relation where we should expect just ONE related object (ie, EE_Belongs_To_relation)
1763
                //or MANY related objects (ie, EE_HABTM_Relation or EE_Has_Many_Relation)?
1764
                if ($relationObj instanceof EE_Belongs_To_Relation) {
1765
                    //add a relation to that relation type (which saves the appropriate thing in the process)
1766
                    //but ONLY if it DOES NOT exist in the DB
1767
                    /* @var $related_model_obj EE_Base_Class */
1768
                    $related_model_obj = $this->_model_relations[$relationName];
1769
                    //					if( ! $related_model_obj->ID()){
1770
                    $this->_add_relation_to($related_model_obj, $relationName);
1771
                    $related_model_obj->save_new_cached_related_model_objs();
1772
                    //					}
1773
                } else {
1774
                    foreach ($this->_model_relations[$relationName] as $related_model_obj) {
1775
                        //add a relation to that relation type (which saves the appropriate thing in the process)
1776
                        //but ONLY if it DOES NOT exist in the DB
1777
                        //						if( ! $related_model_obj->ID()){
1778
                        $this->_add_relation_to($related_model_obj, $relationName);
1779
                        $related_model_obj->save_new_cached_related_model_objs();
1780
                        //						}
1781
                    }
1782
                }
1783
            }
1784
        }
1785
        return $id;
1786
    }
1787
1788
1789
1790
    /**
1791
     * for getting a model while instantiated.
1792
     *
1793
     * @return \EEM_Base | \EEM_CPT_Base
1794
     */
1795
    public function get_model()
1796
    {
1797
        if( ! $this->_model){
1798
            $modelName = self::_get_model_classname(get_class($this));
1799
            $this->_model = self::_get_model_instance_with_name($modelName, $this->_timezone);
1800
        } else {
1801
            $this->_model->set_timezone($this->_timezone);
1802
        }
1803
1804
        return $this->_model;
1805
    }
1806
1807
1808
1809
    /**
1810
     * @param $props_n_values
1811
     * @param $classname
1812
     * @return mixed bool|EE_Base_Class|EEM_CPT_Base
1813
     * @throws \EE_Error
1814
     */
1815
    protected static function _get_object_from_entity_mapper($props_n_values, $classname)
1816
    {
1817
        //TODO: will not work for Term_Relationships because they have no PK!
1818
        $primary_id_ref = self::_get_primary_key_name($classname);
1819
        if (array_key_exists($primary_id_ref, $props_n_values) && ! empty($props_n_values[$primary_id_ref])) {
1820
            $id = $props_n_values[$primary_id_ref];
1821
            return self::_get_model($classname)->get_from_entity_map($id);
1822
        }
1823
        return false;
1824
    }
1825
1826
1827
1828
    /**
1829
     * This is called by child static "new_instance" method and we'll check to see if there is an existing db entry for
1830
     * the primary key (if present in incoming values). If there is a key in the incoming array that matches the
1831
     * primary key for the model AND it is not null, then we check the db. If there's a an object we return it.  If not
1832
     * we return false.
1833
     *
1834
     * @param  array  $props_n_values   incoming array of properties and their values
1835
     * @param  string $classname        the classname of the child class
1836
     * @param null    $timezone
1837
     * @param array   $date_formats     incoming date_formats in an array where the first value is the
1838
     *                                  date_format and the second value is the time format
1839
     * @return mixed (EE_Base_Class|bool)
1840
     * @throws \EE_Error
1841
     */
1842
    protected static function _check_for_object($props_n_values, $classname, $timezone = null, $date_formats = array())
1843
    {
1844
        $existing = null;
1845
        $model = self::_get_model($classname, $timezone);
1846
        if ($model->has_primary_key_field()) {
1847
            $primary_id_ref = self::_get_primary_key_name($classname);
1848
            if (array_key_exists($primary_id_ref, $props_n_values)
1849
                && ! empty($props_n_values[$primary_id_ref])
1850
            ) {
1851
                $existing = $model->get_one_by_ID(
1852
                    $props_n_values[$primary_id_ref]
1853
                );
1854
            }
1855
        } elseif ($model->has_all_combined_primary_key_fields($props_n_values)) {
1856
            //no primary key on this model, but there's still a matching item in the DB
1857
            $existing = self::_get_model($classname, $timezone)->get_one_by_ID(
1858
                self::_get_model($classname, $timezone)->get_index_primary_key_string($props_n_values)
1859
            );
1860
        }
1861
        if ($existing) {
1862
            //set date formats if present before setting values
1863
            if ( ! empty($date_formats) && is_array($date_formats)) {
1864
                $existing->set_date_format($date_formats[0]);
1865
                $existing->set_time_format($date_formats[1]);
1866
            } else {
1867
                //set default formats for date and time
1868
                $existing->set_date_format(get_option('date_format'));
1869
                $existing->set_time_format(get_option('time_format'));
1870
            }
1871
            foreach ($props_n_values as $property => $field_value) {
1872
                $existing->set($property, $field_value);
1873
            }
1874
            return $existing;
1875
        } else {
1876
            return false;
1877
        }
1878
    }
1879
1880
1881
1882
    /**
1883
     * Gets the EEM_*_Model for this class
1884
     *
1885
     * @access public now, as this is more convenient
1886
     * @param      $classname
1887
     * @param null $timezone
1888
     * @throws EE_Error
1889
     * @return EEM_Base
1890
     */
1891
    protected static function _get_model($classname, $timezone = null)
1892
    {
1893
        //find model for this class
1894
        if ( ! $classname) {
1895
            throw new EE_Error(
1896
                sprintf(
1897
                    __(
1898
                        "What were you thinking calling _get_model(%s)?? You need to specify the class name",
1899
                        "event_espresso"
1900
                    ),
1901
                    $classname
1902
                )
1903
            );
1904
        }
1905
        $modelName = self::_get_model_classname($classname);
1906
        return self::_get_model_instance_with_name($modelName, $timezone);
1907
    }
1908
1909
1910
1911
    /**
1912
     * Gets the model instance (eg instance of EEM_Attendee) given its classname (eg EE_Attendee)
1913
     *
1914
     * @param string $model_classname
1915
     * @param null   $timezone
1916
     * @return EEM_Base
1917
     */
1918
    protected static function _get_model_instance_with_name($model_classname, $timezone = null)
1919
    {
1920
        $model_classname = str_replace('EEM_', '', $model_classname);
1921
        $model = EE_Registry::instance()->load_model($model_classname);
1922
        $model->set_timezone($timezone);
1923
        return $model;
1924
    }
1925
1926
1927
1928
    /**
1929
     * If a model name is provided (eg Registration), gets the model classname for that model.
1930
     * Also works if a model class's classname is provided (eg EE_Registration).
1931
     *
1932
     * @param null $model_name
1933
     * @return string like EEM_Attendee
1934
     */
1935
    private static function _get_model_classname($model_name = null)
1936
    {
1937
        if (strpos($model_name, "EE_") === 0) {
1938
            $model_classname = str_replace("EE_", "EEM_", $model_name);
1939
        } else {
1940
            $model_classname = "EEM_" . $model_name;
1941
        }
1942
        return $model_classname;
1943
    }
1944
1945
1946
1947
    /**
1948
     * returns the name of the primary key attribute
1949
     *
1950
     * @param null $classname
1951
     * @throws EE_Error
1952
     * @return string
1953
     */
1954
    protected static function _get_primary_key_name($classname = null)
1955
    {
1956
        if ( ! $classname) {
1957
            throw new EE_Error(
1958
                sprintf(
1959
                    __("What were you thinking calling _get_primary_key_name(%s)", "event_espresso"),
1960
                    $classname
1961
                )
1962
            );
1963
        }
1964
        return self::_get_model($classname)->get_primary_key_field()->get_name();
1965
    }
1966
1967
1968
1969
    /**
1970
     * Gets the value of the primary key.
1971
     * If the object hasn't yet been saved, it should be whatever the model field's default was
1972
     * (eg, if this were the EE_Event class, look at the primary key field on EEM_Event and see what its default value
1973
     * is. Usually defaults for integer primary keys are 0; string primary keys are usually NULL).
1974
     *
1975
     * @return mixed, if the primary key is of type INT it'll be an int. Otherwise it could be a string
1976
     * @throws \EE_Error
1977
     */
1978
    public function ID()
1979
    {
1980
        $model = $this->get_model();
1981
        //now that we know the name of the variable, use a variable variable to get its value and return its
1982
        if ($model->has_primary_key_field()) {
1983
            return $this->_fields[$model->primary_key_name()];
1984
        } else {
1985
            return $model->get_index_primary_key_string($this->_fields);
1986
        }
1987
    }
1988
1989
1990
1991
    /**
1992
     * Adds a relationship to the specified EE_Base_Class object, given the relationship's name. Eg, if the current
1993
     * model is related to a group of events, the $relationName should be 'Event', and should be a key in the EE
1994
     * Model's $_model_relations array. If this model object doesn't exist in the DB, just caches the related thing
1995
     *
1996
     * @param mixed  $otherObjectModelObjectOrID       EE_Base_Class or the ID of the other object
1997
     * @param string $relationName                     eg 'Events','Question',etc.
1998
     *                                                 an attendee to a group, you also want to specify which role they
1999
     *                                                 will have in that group. So you would use this parameter to
2000
     *                                                 specify array('role-column-name'=>'role-id')
2001
     * @param array  $extra_join_model_fields_n_values You can optionally include an array of key=>value pairs that
2002
     *                                                 allow you to further constrict the relation to being added.
2003
     *                                                 However, keep in mind that the columns (keys) given must match a
2004
     *                                                 column on the JOIN table and currently only the HABTM models
2005
     *                                                 accept these additional conditions.  Also remember that if an
2006
     *                                                 exact match isn't found for these extra cols/val pairs, then a
2007
     *                                                 NEW row is created in the join table.
2008
     * @param null   $cache_id
2009
     * @throws EE_Error
2010
     * @return EE_Base_Class the object the relation was added to
2011
     */
2012
    public function _add_relation_to(
2013
        $otherObjectModelObjectOrID,
2014
        $relationName,
2015
        $extra_join_model_fields_n_values = array(),
2016
        $cache_id = null
2017
    ) {
2018
        $model = $this->get_model();
2019
        //if this thing exists in the DB, save the relation to the DB
2020
        if ($this->ID()) {
2021
            $otherObject = $model
2022
                                ->add_relationship_to($this, $otherObjectModelObjectOrID, $relationName,
2023
                                    $extra_join_model_fields_n_values);
2024
            //clear cache so future get_many_related and get_first_related() return new results.
2025
            $this->clear_cache($relationName, $otherObject, true);
2026
            if ($otherObject instanceof EE_Base_Class) {
2027
                $otherObject->clear_cache($model->get_this_model_name(), $this);
2028
            }
2029 View Code Duplication
        } else {
2030
            //this thing doesn't exist in the DB,  so just cache it
2031
            if ( ! $otherObjectModelObjectOrID instanceof EE_Base_Class) {
2032
                throw new EE_Error(sprintf(
2033
                    __('Before a model object is saved to the database, calls to _add_relation_to must be passed an actual object, not just an ID. You provided %s as the model object to a %s',
2034
                        'event_espresso'),
2035
                    $otherObjectModelObjectOrID,
2036
                    get_class($this)
2037
                ));
2038
            } else {
2039
                $otherObject = $otherObjectModelObjectOrID;
2040
            }
2041
            $this->cache($relationName, $otherObjectModelObjectOrID, $cache_id);
2042
        }
2043
        if ($otherObject instanceof EE_Base_Class) {
2044
            //fix the reciprocal relation too
2045
            if ($otherObject->ID()) {
2046
                //its saved so assumed relations exist in the DB, so we can just
2047
                //clear the cache so future queries use the updated info in the DB
2048
                $otherObject->clear_cache($model->get_this_model_name(), null, true);
2049
            } else {
2050
                //it's not saved, so it caches relations like this
2051
                $otherObject->cache($model->get_this_model_name(), $this);
2052
            }
2053
        }
2054
        return $otherObject;
2055
    }
2056
2057
2058
2059
    /**
2060
     * Removes a relationship to the specified EE_Base_Class object, given the relationships' name. Eg, if the current
2061
     * model is related to a group of events, the $relationName should be 'Events', and should be a key in the EE
2062
     * Model's $_model_relations array. If this model object doesn't exist in the DB, just removes the related thing
2063
     * from the cache
2064
     *
2065
     * @param mixed  $otherObjectModelObjectOrID
2066
     *                EE_Base_Class or the ID of the other object, OR an array key into the cache if this isn't saved
2067
     *                to the DB yet
2068
     * @param string $relationName
2069
     * @param array  $where_query
2070
     *                You can optionally include an array of key=>value pairs that allow you to further constrict the
2071
     *                relation to being added. However, keep in mind that the columns (keys) given must match a column
2072
     *                on the JOIN table and currently only the HABTM models accept these additional conditions. Also
2073
     *                remember that if an exact match isn't found for these extra cols/val pairs, then a NEW row is
2074
     *                created in the join table.
2075
     * @return EE_Base_Class the relation was removed from
2076
     * @throws \EE_Error
2077
     */
2078
    public function _remove_relation_to($otherObjectModelObjectOrID, $relationName, $where_query = array())
2079
    {
2080
        if ($this->ID()) {
2081
            //if this exists in the DB, save the relation change to the DB too
2082
            $otherObject = $this->get_model()
2083
                                ->remove_relationship_to($this, $otherObjectModelObjectOrID, $relationName,
2084
                                    $where_query);
2085
            $this->clear_cache($relationName, $otherObject);
2086
        } else {
2087
            //this doesn't exist in the DB, just remove it from the cache
2088
            $otherObject = $this->clear_cache($relationName, $otherObjectModelObjectOrID);
2089
        }
2090
        if ($otherObject instanceof EE_Base_Class) {
2091
            $otherObject->clear_cache($this->get_model()->get_this_model_name(), $this);
2092
        }
2093
        return $otherObject;
2094
    }
2095
2096
2097
2098
    /**
2099
     * Removes ALL the related things for the $relationName.
2100
     *
2101
     * @param string $relationName
2102
     * @param array  $where_query_params like EEM_Base::get_all's $query_params[0] (where conditions)
2103
     * @return EE_Base_Class
2104
     * @throws \EE_Error
2105
     */
2106
    public function _remove_relations($relationName, $where_query_params = array())
2107
    {
2108
        if ($this->ID()) {
2109
            //if this exists in the DB, save the relation change to the DB too
2110
            $otherObjects = $this->get_model()->remove_relations($this, $relationName, $where_query_params);
2111
            $this->clear_cache($relationName, null, true);
2112
        } else {
2113
            //this doesn't exist in the DB, just remove it from the cache
2114
            $otherObjects = $this->clear_cache($relationName, null, true);
2115
        }
2116
        if (is_array($otherObjects)) {
2117
            foreach ($otherObjects as $otherObject) {
2118
                $otherObject->clear_cache($this->get_model()->get_this_model_name(), $this);
2119
            }
2120
        }
2121
        return $otherObjects;
2122
    }
2123
2124
2125
2126
    /**
2127
     * Gets all the related model objects of the specified type. Eg, if the current class if
2128
     * EE_Event, you could call $this->get_many_related('Registration') to get an array of all the
2129
     * EE_Registration objects which related to this event. Note: by default, we remove the "default query params"
2130
     * because we want to get even deleted items etc.
2131
     *
2132
     * @param string $relationName key in the model's _model_relations array
2133
     * @param array  $query_params like EEM_Base::get_all
2134
     * @return EE_Base_Class[] Results not necessarily indexed by IDs, because some results might not have primary keys
2135
     * @throws \EE_Error
2136
     *                             or might not be saved yet. Consider using EEM_Base::get_IDs() on these results if
2137
     *                             you want IDs
2138
     */
2139
    public function get_many_related($relationName, $query_params = array())
2140
    {
2141
        if ($this->ID()) {
2142
            //this exists in the DB, so get the related things from either the cache or the DB
2143
            //if there are query parameters, forget about caching the related model objects.
2144
            if ($query_params) {
0 ignored issues
show
Bug Best Practice introduced by
The expression $query_params of type array is implicitly converted to a boolean; are you sure this is intended? If so, consider using ! empty($expr) instead to make it clear that you intend to check for an array without elements.

This check marks implicit conversions of arrays to boolean values in a comparison. While in PHP an empty array is considered to be equal (but not identical) to false, this is not always apparent.

Consider making the comparison explicit by using empty(..) or ! empty(...) instead.

Loading history...
2145
                $related_model_objects = $this->get_model()->get_all_related($this, $relationName, $query_params);
2146
            } else {
2147
                //did we already cache the result of this query?
2148
                $cached_results = $this->get_all_from_cache($relationName);
2149
                if ( ! $cached_results) {
0 ignored issues
show
Bug Best Practice introduced by
The expression $cached_results of type EE_Base_Class[] is implicitly converted to a boolean; are you sure this is intended? If so, consider using empty($expr) instead to make it clear that you intend to check for an array without elements.

This check marks implicit conversions of arrays to boolean values in a comparison. While in PHP an empty array is considered to be equal (but not identical) to false, this is not always apparent.

Consider making the comparison explicit by using empty(..) or ! empty(...) instead.

Loading history...
2150
                    $related_model_objects = $this->get_model()->get_all_related($this, $relationName, $query_params);
2151
                    //if no query parameters were passed, then we got all the related model objects
2152
                    //for that relation. We can cache them then.
2153
                    foreach ($related_model_objects as $related_model_object) {
2154
                        $this->cache($relationName, $related_model_object);
2155
                    }
2156
                } else {
2157
                    $related_model_objects = $cached_results;
2158
                }
2159
            }
2160
        } else {
2161
            //this doesn't exist in the DB, so just get the related things from the cache
2162
            $related_model_objects = $this->get_all_from_cache($relationName);
2163
        }
2164
        return $related_model_objects;
2165
    }
2166
2167
2168
2169
    /**
2170
     * Instead of getting the related model objects, simply counts them. Ignores default_where_conditions by default,
2171
     * unless otherwise specified in the $query_params
2172
     *
2173
     * @param string $relation_name  model_name like 'Event', or 'Registration'
2174
     * @param array  $query_params   like EEM_Base::get_all's
2175
     * @param string $field_to_count name of field to count by. By default, uses primary key
2176
     * @param bool   $distinct       if we want to only count the distinct values for the column then you can trigger
2177
     *                               that by the setting $distinct to TRUE;
2178
     * @return int
2179
     */
2180
    public function count_related($relation_name, $query_params = array(), $field_to_count = null, $distinct = false)
2181
    {
2182
        return $this->get_model()->count_related($this, $relation_name, $query_params, $field_to_count, $distinct);
2183
    }
2184
2185
2186
2187
    /**
2188
     * Instead of getting the related model objects, simply sums up the values of the specified field.
2189
     * Note: ignores default_where_conditions by default, unless otherwise specified in the $query_params
2190
     *
2191
     * @param string $relation_name model_name like 'Event', or 'Registration'
2192
     * @param array  $query_params  like EEM_Base::get_all's
2193
     * @param string $field_to_sum  name of field to count by.
2194
     *                              By default, uses primary key (which doesn't make much sense, so you should probably
2195
     *                              change it)
2196
     * @return int
2197
     */
2198
    public function sum_related($relation_name, $query_params = array(), $field_to_sum = null)
2199
    {
2200
        return $this->get_model()->sum_related($this, $relation_name, $query_params, $field_to_sum);
2201
    }
2202
2203
2204
2205
    /**
2206
     * Gets the first (ie, one) related model object of the specified type.
2207
     *
2208
     * @param string $relationName key in the model's _model_relations array
2209
     * @param array  $query_params like EEM_Base::get_all
2210
     * @return EE_Base_Class (not an array, a single object)
2211
     * @throws \EE_Error
2212
     */
2213
    public function get_first_related($relationName, $query_params = array())
2214
    {
2215
        $model = $this->get_model();
2216
        if ($this->ID()) {//this exists in the DB, get from the cache OR the DB
2217
            //if they've provided some query parameters, don't bother trying to cache the result
2218
            //also make sure we're not caching the result of get_first_related
2219
            //on a relation which should have an array of objects (because the cache might have an array of objects)
2220
            if ($query_params
0 ignored issues
show
Bug Best Practice introduced by
The expression $query_params of type array is implicitly converted to a boolean; are you sure this is intended? If so, consider using ! empty($expr) instead to make it clear that you intend to check for an array without elements.

This check marks implicit conversions of arrays to boolean values in a comparison. While in PHP an empty array is considered to be equal (but not identical) to false, this is not always apparent.

Consider making the comparison explicit by using empty(..) or ! empty(...) instead.

Loading history...
2221
                || ! $model->related_settings_for($relationName)
2222
                     instanceof
2223
                     EE_Belongs_To_Relation
2224
            ) {
2225
                $related_model_object = $model->get_first_related($this, $relationName, $query_params);
2226
            } else {
2227
                //first, check if we've already cached the result of this query
2228
                $cached_result = $this->get_one_from_cache($relationName);
2229
                if ( ! $cached_result) {
2230
                    $related_model_object = $model->get_first_related($this, $relationName, $query_params);
2231
                    $this->cache($relationName, $related_model_object);
2232
                } else {
2233
                    $related_model_object = $cached_result;
2234
                }
2235
            }
2236
        } else {
2237
            $related_model_object = null;
2238
            //this doesn't exist in the Db, but maybe the relation is of type belongs to, and so the related thing might
2239
            if ($model->related_settings_for($relationName) instanceof EE_Belongs_To_Relation) {
2240
                $related_model_object = $model->get_first_related($this, $relationName, $query_params);
2241
            }
2242
            //this doesn't exist in the DB and apparently the thing it belongs to doesn't either, just get what's cached on this object
2243
            if ( ! $related_model_object) {
2244
                $related_model_object = $this->get_one_from_cache($relationName);
2245
            }
2246
        }
2247
        return $related_model_object;
2248
    }
2249
2250
2251
2252
    /**
2253
     * Does a delete on all related objects of type $relationName and removes
2254
     * the current model object's relation to them. If they can't be deleted (because
2255
     * of blocking related model objects) does nothing. If the related model objects are
2256
     * soft-deletable, they will be soft-deleted regardless of related blocking model objects.
2257
     * If this model object doesn't exist yet in the DB, just removes its related things
2258
     *
2259
     * @param string $relationName
2260
     * @param array  $query_params like EEM_Base::get_all's
2261
     * @return int how many deleted
2262
     * @throws \EE_Error
2263
     */
2264 View Code Duplication
    public function delete_related($relationName, $query_params = array())
2265
    {
2266
        if ($this->ID()) {
2267
            $count = $this->get_model()->delete_related($this, $relationName, $query_params);
2268
        } else {
2269
            $count = count($this->get_all_from_cache($relationName));
2270
            $this->clear_cache($relationName, null, true);
2271
        }
2272
        return $count;
2273
    }
2274
2275
2276
2277
    /**
2278
     * Does a hard delete (ie, removes the DB row) on all related objects of type $relationName and removes
2279
     * the current model object's relation to them. If they can't be deleted (because
2280
     * of blocking related model objects) just does a soft delete on it instead, if possible.
2281
     * If the related thing isn't a soft-deletable model object, this function is identical
2282
     * to delete_related(). If this model object doesn't exist in the DB, just remove its related things
2283
     *
2284
     * @param string $relationName
2285
     * @param array  $query_params like EEM_Base::get_all's
2286
     * @return int how many deleted (including those soft deleted)
2287
     * @throws \EE_Error
2288
     */
2289 View Code Duplication
    public function delete_related_permanently($relationName, $query_params = array())
2290
    {
2291
        if ($this->ID()) {
2292
            $count = $this->get_model()->delete_related_permanently($this, $relationName, $query_params);
2293
        } else {
2294
            $count = count($this->get_all_from_cache($relationName));
2295
        }
2296
        $this->clear_cache($relationName, null, true);
2297
        return $count;
2298
    }
2299
2300
2301
2302
    /**
2303
     * is_set
2304
     * Just a simple utility function children can use for checking if property exists
2305
     *
2306
     * @access  public
2307
     * @param  string $field_name property to check
2308
     * @return bool                              TRUE if existing,FALSE if not.
2309
     */
2310
    public function is_set($field_name)
2311
    {
2312
        return isset($this->_fields[$field_name]);
2313
    }
2314
2315
2316
2317
    /**
2318
     * Just a simple utility function children can use for checking if property (or properties) exists and throwing an
2319
     * EE_Error exception if they don't
2320
     *
2321
     * @param  mixed (string|array) $properties properties to check
2322
     * @throws EE_Error
2323
     * @return bool                              TRUE if existing, throw EE_Error if not.
2324
     */
2325
    protected function _property_exists($properties)
2326
    {
2327
        foreach ((array)$properties as $property_name) {
2328
            //first make sure this property exists
2329
            if ( ! $this->_fields[$property_name]) {
2330
                throw new EE_Error(
2331
                    sprintf(
2332
                        __(
2333
                            'Trying to retrieve a non-existent property (%s).  Double check the spelling please',
2334
                            'event_espresso'
2335
                        ),
2336
                        $property_name
2337
                    )
2338
                );
2339
            }
2340
        }
2341
        return true;
2342
    }
2343
2344
2345
2346
    /**
2347
     * This simply returns an array of model fields for this object
2348
     *
2349
     * @return array
2350
     * @throws \EE_Error
2351
     */
2352
    public function model_field_array()
2353
    {
2354
        $fields = $this->get_model()->field_settings(false);
2355
        $properties = array();
2356
        //remove prepended underscore
2357
        foreach ($fields as $field_name => $settings) {
2358
            $properties[$field_name] = $this->get($field_name);
2359
        }
2360
        return $properties;
2361
    }
2362
2363
2364
2365
    /**
2366
     * Very handy general function to allow for plugins to extend any child of EE_Base_Class.
2367
     * If a method is called on a child of EE_Base_Class that doesn't exist, this function is called
2368
     * (http://www.garfieldtech.com/blog/php-magic-call) and passed the method's name and arguments. Instead of
2369
     * requiring a plugin to extend the EE_Base_Class (which works fine is there's only 1 plugin, but when will that
2370
     * happen?) they can add a hook onto 'filters_hook_espresso__{className}__{methodName}' (eg,
2371
     * filters_hook_espresso__EE_Answer__my_great_function) and accepts 2 arguments: the object on which the function
2372
     * was called, and an array of the original arguments passed to the function. Whatever their callback function
2373
     * returns will be returned by this function. Example: in functions.php (or in a plugin):
2374
     * add_filter('FHEE__EE_Answer__my_callback','my_callback',10,3); function
2375
     * my_callback($previousReturnValue,EE_Base_Class $object,$argsArray){
2376
     * $returnString= "you called my_callback! and passed args:".implode(",",$argsArray);
2377
     *        return $previousReturnValue.$returnString;
2378
     * }
2379
     * require('EE_Answer.class.php');
2380
     * $answer= EE_Answer::new_instance(array('REG_ID' => 2,'QST_ID' => 3,'ANS_value' => The answer is 42'));
2381
     * echo $answer->my_callback('monkeys',100);
2382
     * //will output "you called my_callback! and passed args:monkeys,100"
2383
     *
2384
     * @param string $methodName name of method which was called on a child of EE_Base_Class, but which
2385
     * @param array  $args       array of original arguments passed to the function
2386
     * @throws EE_Error
2387
     * @return mixed whatever the plugin which calls add_filter decides
2388
     */
2389
    public function __call($methodName, $args)
2390
    {
2391
        $className = get_class($this);
2392
        $tagName = "FHEE__{$className}__{$methodName}";
2393
        if ( ! has_filter($tagName)) {
2394
            throw new EE_Error(
2395
                sprintf(
2396
                    __(
2397
                        "Method %s on class %s does not exist! You can create one with the following code in functions.php or in a plugin: add_filter('%s','my_callback',10,3);function my_callback(\$previousReturnValue,EE_Base_Class \$object, \$argsArray){/*function body*/return \$whatever;}",
2398
                        "event_espresso"
2399
                    ),
2400
                    $methodName,
2401
                    $className,
2402
                    $tagName
2403
                )
2404
            );
2405
        }
2406
        return apply_filters($tagName, null, $this, $args);
2407
    }
2408
2409
2410
2411
    /**
2412
     * Similar to insert_post_meta, adds a record in the Extra_Meta model's table with the given key and value.
2413
     * A $previous_value can be specified in case there are many meta rows with the same key
2414
     *
2415
     * @param string $meta_key
2416
     * @param mixed  $meta_value
2417
     * @param mixed  $previous_value
2418
     * @return bool|int # of records updated (or BOOLEAN if we actually ended up inserting the extra meta row)
2419
     * @throws \EE_Error
2420
     * NOTE: if the values haven't changed, returns 0
2421
     */
2422
    public function update_extra_meta($meta_key, $meta_value, $previous_value = null)
2423
    {
2424
        $query_params = array(
2425
            array(
2426
                'EXM_key'  => $meta_key,
2427
                'OBJ_ID'   => $this->ID(),
2428
                'EXM_type' => $this->get_model()->get_this_model_name(),
2429
            ),
2430
        );
2431
        if ($previous_value !== null) {
2432
            $query_params[0]['EXM_value'] = $meta_value;
2433
        }
2434
        $existing_rows_like_that = EEM_Extra_Meta::instance()->get_all($query_params);
2435
        if ( ! $existing_rows_like_that) {
0 ignored issues
show
Bug Best Practice introduced by
The expression $existing_rows_like_that of type EE_Base_Class[] is implicitly converted to a boolean; are you sure this is intended? If so, consider using empty($expr) instead to make it clear that you intend to check for an array without elements.

This check marks implicit conversions of arrays to boolean values in a comparison. While in PHP an empty array is considered to be equal (but not identical) to false, this is not always apparent.

Consider making the comparison explicit by using empty(..) or ! empty(...) instead.

Loading history...
2436
            return $this->add_extra_meta($meta_key, $meta_value);
2437
        }
2438
        foreach ($existing_rows_like_that as $existing_row) {
2439
            $existing_row->save(array('EXM_value' => $meta_value));
2440
        }
2441
        return count($existing_rows_like_that);
2442
    }
2443
2444
2445
2446
    /**
2447
     * Adds a new extra meta record. If $unique is set to TRUE, we'll first double-check
2448
     * no other extra meta for this model object have the same key. Returns TRUE if the
2449
     * extra meta row was entered, false if not
2450
     *
2451
     * @param string  $meta_key
2452
     * @param mixed   $meta_value
2453
     * @param boolean $unique
2454
     * @return boolean
2455
     * @throws \EE_Error
2456
     */
2457
    public function add_extra_meta($meta_key, $meta_value, $unique = false)
2458
    {
2459
        if ($unique) {
2460
            $existing_extra_meta = EEM_Extra_Meta::instance()->get_one(
2461
                array(
2462
                    array(
2463
                        'EXM_key'  => $meta_key,
2464
                        'OBJ_ID'   => $this->ID(),
2465
                        'EXM_type' => $this->get_model()->get_this_model_name(),
2466
                    ),
2467
                )
2468
            );
2469
            if ($existing_extra_meta) {
2470
                return false;
2471
            }
2472
        }
2473
        $new_extra_meta = EE_Extra_Meta::new_instance(
2474
            array(
2475
                'EXM_key'   => $meta_key,
2476
                'EXM_value' => $meta_value,
2477
                'OBJ_ID'    => $this->ID(),
2478
                'EXM_type'  => $this->get_model()->get_this_model_name(),
2479
            )
2480
        );
2481
        $new_extra_meta->save();
2482
        return true;
2483
    }
2484
2485
2486
2487
    /**
2488
     * Deletes all the extra meta rows for this record as specified by key. If $meta_value
2489
     * is specified, only deletes extra meta records with that value.
2490
     *
2491
     * @param string $meta_key
2492
     * @param mixed  $meta_value
2493
     * @return int number of extra meta rows deleted
2494
     * @throws \EE_Error
2495
     */
2496
    public function delete_extra_meta($meta_key, $meta_value = null)
2497
    {
2498
        $query_params = array(
2499
            array(
2500
                'EXM_key'  => $meta_key,
2501
                'OBJ_ID'   => $this->ID(),
2502
                'EXM_type' => $this->get_model()->get_this_model_name(),
2503
            ),
2504
        );
2505
        if ($meta_value !== null) {
2506
            $query_params[0]['EXM_value'] = $meta_value;
2507
        }
2508
        return EEM_Extra_Meta::instance()->delete($query_params);
2509
    }
2510
2511
2512
2513
    /**
2514
     * Gets the extra meta with the given meta key. If you specify "single" we just return 1, otherwise
2515
     * an array of everything found. Requires that this model actually have a relation of type EE_Has_Many_Any_Relation.
2516
     * You can specify $default is case you haven't found the extra meta
2517
     *
2518
     * @param string  $meta_key
2519
     * @param boolean $single
2520
     * @param mixed   $default if we don't find anything, what should we return?
2521
     * @return mixed single value if $single; array if ! $single
2522
     * @throws \EE_Error
2523
     */
2524
    public function get_extra_meta($meta_key, $single = false, $default = null)
2525
    {
2526
        if ($single) {
2527
            $result = $this->get_first_related('Extra_Meta', array(array('EXM_key' => $meta_key)));
2528
            if ($result instanceof EE_Extra_Meta) {
2529
                return $result->value();
2530
            }
2531
        } else {
2532
            $results = $this->get_many_related('Extra_Meta', array(array('EXM_key' => $meta_key)));
2533
            if ($results) {
0 ignored issues
show
Bug Best Practice introduced by
The expression $results of type EE_Base_Class[] is implicitly converted to a boolean; are you sure this is intended? If so, consider using ! empty($expr) instead to make it clear that you intend to check for an array without elements.

This check marks implicit conversions of arrays to boolean values in a comparison. While in PHP an empty array is considered to be equal (but not identical) to false, this is not always apparent.

Consider making the comparison explicit by using empty(..) or ! empty(...) instead.

Loading history...
2534
                $values = array();
2535
                foreach ($results as $result) {
2536
                    if ($result instanceof EE_Extra_Meta) {
2537
                        $values[$result->ID()] = $result->value();
2538
                    }
2539
                }
2540
                return $values;
2541
            }
2542
        }
2543
        //if nothing discovered yet return default.
2544
        return apply_filters(
2545
            'FHEE__EE_Base_Class__get_extra_meta__default_value',
2546
            $default,
2547
            $meta_key,
2548
            $single,
2549
            $this
2550
            );
2551
    }
2552
2553
2554
2555
    /**
2556
     * Returns a simple array of all the extra meta associated with this model object.
2557
     * If $one_of_each_key is true (Default), it will be an array of simple key-value pairs, keys being the
2558
     * extra meta's key, and teh value being its value. However, if there are duplicate extra meta rows with
2559
     * the same key, only one will be used. (eg array('foo'=>'bar','monkey'=>123))
2560
     * If $one_of_each_key is false, it will return an array with the top-level keys being
2561
     * the extra meta keys, but their values are also arrays, which have the extra-meta's ID as their sub-key, and
2562
     * finally the extra meta's value as each sub-value. (eg
2563
     * array('foo'=>array(1=>'bar',2=>'bill'),'monkey'=>array(3=>123)))
2564
     *
2565
     * @param boolean $one_of_each_key
2566
     * @return array
2567
     * @throws \EE_Error
2568
     */
2569
    public function all_extra_meta_array($one_of_each_key = true)
2570
    {
2571
        $return_array = array();
2572
        if ($one_of_each_key) {
2573
            $extra_meta_objs = $this->get_many_related('Extra_Meta', array('group_by' => 'EXM_key'));
2574
            foreach ($extra_meta_objs as $extra_meta_obj) {
2575
                if ($extra_meta_obj instanceof EE_Extra_Meta) {
2576
                    $return_array[$extra_meta_obj->key()] = $extra_meta_obj->value();
2577
                }
2578
            }
2579
        } else {
2580
            $extra_meta_objs = $this->get_many_related('Extra_Meta');
2581
            foreach ($extra_meta_objs as $extra_meta_obj) {
2582
                if ($extra_meta_obj instanceof EE_Extra_Meta) {
2583
                    if ( ! isset($return_array[$extra_meta_obj->key()])) {
2584
                        $return_array[$extra_meta_obj->key()] = array();
2585
                    }
2586
                    $return_array[$extra_meta_obj->key()][$extra_meta_obj->ID()] = $extra_meta_obj->value();
2587
                }
2588
            }
2589
        }
2590
        return $return_array;
2591
    }
2592
2593
2594
2595
    /**
2596
     * Gets a pretty nice displayable nice for this model object. Often overridden
2597
     *
2598
     * @return string
2599
     * @throws \EE_Error
2600
     */
2601
    public function name()
2602
    {
2603
        //find a field that's not a text field
2604
        $field_we_can_use = $this->get_model()->get_a_field_of_type('EE_Text_Field_Base');
2605
        if ($field_we_can_use) {
2606
            return $this->get($field_we_can_use->get_name());
2607
        } else {
2608
            $first_few_properties = $this->model_field_array();
2609
            $first_few_properties = array_slice($first_few_properties, 0, 3);
2610
            $name_parts = array();
2611
            foreach ($first_few_properties as $name => $value) {
2612
                $name_parts[] = "$name:$value";
2613
            }
2614
            return implode(",", $name_parts);
2615
        }
2616
    }
2617
2618
2619
2620
    /**
2621
     * in_entity_map
2622
     * Checks if this model object has been proven to already be in the entity map
2623
     *
2624
     * @return boolean
2625
     * @throws \EE_Error
2626
     */
2627
    public function in_entity_map()
2628
    {
2629
        if ($this->ID() && $this->get_model()->get_from_entity_map($this->ID()) === $this) {
2630
            //well, if we looked, did we find it in the entity map?
2631
            return true;
2632
        } else {
2633
            return false;
2634
        }
2635
    }
2636
2637
2638
2639
    /**
2640
     * refresh_from_db
2641
     * Makes sure the fields and values on this model object are in-sync with what's in the database.
2642
     *
2643
     * @throws EE_Error if this model object isn't in the entity mapper (because then you should
2644
     * just use what's in the entity mapper and refresh it) and WP_DEBUG is TRUE
2645
     */
2646
    public function refresh_from_db()
2647
    {
2648
        if ($this->ID() && $this->in_entity_map()) {
2649
            $this->get_model()->refresh_entity_map_from_db($this->ID());
2650
        } else {
2651
            //if it doesn't have ID, you shouldn't be asking to refresh it from teh database (because its not in the database)
2652
            //if it has an ID but it's not in the map, and you're asking me to refresh it
2653
            //that's kinda dangerous. You should just use what's in the entity map, or add this to the entity map if there's
2654
            //absolutely nothing in it for this ID
2655
            if (WP_DEBUG) {
2656
                throw new EE_Error(
2657
                    sprintf(
2658
                        __('Trying to refresh a model object with ID "%1$s" that\'s not in the entity map? First off: you should put it in the entity map by calling %2$s. Second off, if you want what\'s in the database right now, you should just call %3$s yourself and discard this model object.',
2659
                            'event_espresso'),
2660
                        $this->ID(),
2661
                        get_class($this->get_model()) . '::instance()->add_to_entity_map()',
2662
                        get_class($this->get_model()) . '::instance()->refresh_entity_map()'
2663
                    )
2664
                );
2665
            }
2666
        }
2667
    }
2668
2669
2670
2671
    /**
2672
     * Because some other plugins, like Advanced Cron Manager, expect all objects to have this method
2673
     * (probably a bad assumption they have made, oh well)
2674
     *
2675
     * @return string
2676
     */
2677
    public function __toString()
2678
    {
2679
        try {
2680
            return sprintf('%s (%s)', $this->name(), $this->ID());
2681
        } catch (Exception $e) {
2682
            EE_Error::add_error($e->getMessage(), __FILE__, __FUNCTION__, __LINE__);
2683
            return '';
2684
        }
2685
    }
2686
2687
2688
2689
    /**
2690
     * Clear related model objects if they're already in the DB, because otherwise when we
2691
     * UN-serialize this model object we'll need to be careful to add them to the entity map.
2692
     * This means if we have made changes to those related model objects, and want to unserialize
2693
     * the this model object on a subsequent request, changes to those related model objects will be lost.
2694
     * Instead, those related model objects should be directly serialized and stored.
2695
     * Eg, the following won't work:
2696
     * $reg = EEM_Registration::instance()->get_one_by_ID( 123 );
2697
     * $att = $reg->attendee();
2698
     * $att->set( 'ATT_fname', 'Dirk' );
2699
     * update_option( 'my_option', serialize( $reg ) );
2700
     * //END REQUEST
2701
     * //START NEXT REQUEST
2702
     * $reg = get_option( 'my_option' );
2703
     * $reg->attendee()->save();
2704
     * And would need to be replace with:
2705
     * $reg = EEM_Registration::instance()->get_one_by_ID( 123 );
2706
     * $att = $reg->attendee();
2707
     * $att->set( 'ATT_fname', 'Dirk' );
2708
     * update_option( 'my_option', serialize( $reg ) );
2709
     * //END REQUEST
2710
     * //START NEXT REQUEST
2711
     * $att = get_option( 'my_option' );
2712
     * $att->save();
2713
     *
2714
     * @return array
2715
     * @throws \EE_Error
2716
     */
2717
    public function __sleep()
2718
    {
2719
        $model = $this->get_model();
2720
        foreach ($model->relation_settings() as $relation_name => $relation_obj) {
2721
            if ($relation_obj instanceof EE_Belongs_To_Relation) {
2722
                $classname = 'EE_' . $model->get_this_model_name();
2723
                if (
2724
                    $this->get_one_from_cache($relation_name) instanceof $classname
2725
                    && $this->get_one_from_cache($relation_name)->ID()
2726
                ) {
2727
                    $this->clear_cache($relation_name, $this->get_one_from_cache($relation_name)->ID());
2728
                }
2729
            }
2730
        }
2731
        $this->_props_n_values_provided_in_constructor = array();
2732
        $properties_to_serialize = get_object_vars($this);
2733
        //don't serialize the model. It's big and that risks recursion
2734
        unset($properties_to_serialize['_model']);
2735
        return array_keys($properties_to_serialize);
2736
    }
2737
2738
2739
2740
    /**
2741
     * restore _props_n_values_provided_in_constructor
2742
     * PLZ NOTE: this will reset the array to whatever fields values were present prior to serialization,
2743
     * and therefore should NOT be used to determine if state change has occurred since initial construction.
2744
     * At best, you would only be able to detect if state change has occurred during THIS request.
2745
     */
2746
    public function __wakeup()
2747
    {
2748
        $this->_props_n_values_provided_in_constructor = $this->_fields;
2749
    }
2750
2751
2752
2753
}
2754
2755
2756