Completed
Branch FET/11011/recurring-events-man... (755794)
by
unknown
08:33 queued 17s
created
caffeinated/payment_methods/Aim/EEG_Aim.gateway.php 3 patches
Doc Comments   +1 added lines, -1 removed lines patch added patch discarded remove patch
@@ -362,7 +362,7 @@
 block discarded – undo
362 362
     /**
363 363
      * Posts the request to AuthorizeNet & returns response.
364 364
      *
365
-     * @param $payment
365
+     * @param EEI_Payment $payment
366 366
      * @return \EE_AuthorizeNetAIM_Response
367 367
      */
368 368
     private function _sendRequest($payment)
Please login to merge, or discard this patch.
Indentation   +602 added lines, -602 removed lines patch added patch discarded remove patch
@@ -23,424 +23,424 @@  discard block
 block discarded – undo
23 23
 class EEG_Aim extends EE_Onsite_Gateway
24 24
 {
25 25
 
26
-    const LIVE_URL    = 'https://secure2.authorize.net/gateway/transact.dll'; // Authnet URL
27
-
28
-    const SANDBOX_URL = 'https://test.authorize.net/gateway/transact.dll';
29
-
30
-    protected $_login_id;
31
-
32
-    protected $_transaction_key;
33
-
34
-    protected $_server;
35
-
36
-    protected $_currencies_supported = array(
37
-        'AUD',
38
-        'USD',
39
-        'CAD',
40
-        'EUR',
41
-        'GBP',
42
-        'NZD',
43
-    );
44
-
45
-    /**
46
-     * Whether to send test transactions (even to live site)
47
-     *
48
-     * @var boolean
49
-     */
50
-    protected $_test_transactions;
51
-
52
-    private $VERIFY_PEER = false;
53
-
54
-    private $_x_post_fields = array(
55
-        "version"        => "3.1",
56
-        "delim_char"     => ",",
57
-        "delim_data"     => "TRUE",
58
-        "relay_response" => "FALSE",
59
-        "encap_char"     => "|",
60
-    );
61
-
62
-    private $_additional_line_items = array();
63
-
64
-    /**
65
-     * A list of all fields in the AIM API.
66
-     * Used to warn user if they try to set a field not offered in the API.
67
-     */
68
-    private $_all_aim_fields = array(
69
-        "address",
70
-        "allow_partial_auth",
71
-        "amount",
72
-        "auth_code",
73
-        "authentication_indicator",
74
-        "bank_aba_code",
75
-        "bank_acct_name",
76
-        "bank_acct_num",
77
-        "bank_acct_type",
78
-        "bank_check_number",
79
-        "bank_name",
80
-        "card_code",
81
-        "card_num",
82
-        "cardholder_authentication_value",
83
-        "city",
84
-        "company",
85
-        "country",
86
-        "cust_id",
87
-        "customer_ip",
88
-        "delim_char",
89
-        "delim_data",
90
-        "description",
91
-        "duplicate_window",
92
-        "duty",
93
-        "echeck_type",
94
-        "email",
95
-        "email_customer",
96
-        "encap_char",
97
-        "exp_date",
98
-        "fax",
99
-        "first_name",
100
-        "footer_email_receipt",
101
-        "freight",
102
-        "header_email_receipt",
103
-        "invoice_num",
104
-        "last_name",
105
-        "line_item",
106
-        "login",
107
-        "method",
108
-        "phone",
109
-        "po_num",
110
-        "recurring_billing",
111
-        "relay_response",
112
-        "ship_to_address",
113
-        "ship_to_city",
114
-        "ship_to_company",
115
-        "ship_to_country",
116
-        "ship_to_first_name",
117
-        "ship_to_last_name",
118
-        "ship_to_state",
119
-        "ship_to_zip",
120
-        "split_tender_id",
121
-        "state",
122
-        "tax",
123
-        "tax_exempt",
124
-        "test_request",
125
-        "tran_key",
126
-        "trans_id",
127
-        "type",
128
-        "version",
129
-        "zip",
130
-        "solution_id"
131
-    );
132
-
133
-
134
-    /**
135
-     * Gets the URL where the request should go. This is filterable
136
-     *
137
-     * @return string
138
-     */
139
-    protected function _get_server_url()
140
-    {
141
-        return apply_filters(
142
-            'FHEE__EEG_Aim___get_server_url',
143
-            $this->_debug_mode ? self::SANDBOX_URL : self::LIVE_URL,
144
-            $this
145
-        );
146
-    }
147
-
148
-
149
-    /**
150
-     * TEMPORARY CALLBACK! Do not use
151
-     * Callback which filters the server url. This is added so site admins can revert to using
152
-     * the old AIM server in case Akamai service breaks their integration.
153
-     * Using Akamai will, however, be mandatory on June 30th 2016 Authorize.net
154
-     * (see http://www.authorize.net/support/akamaifaqs/#firewall?utm_campaign=April%202016%20Technical%20Updates%20for%20Merchants.html&utm_medium=email&utm_source=Eloqua&elqTrackId=46103bdc375c411a979c2f658fc99074&elq=7026706360154fee9b6d588b27d8eb6a&elqaid=506&elqat=1&elqCampaignId=343)
155
-     * Once that happens, this will be obsolete and WILL BE REMOVED.
156
-     *
157
-     * @param string $url
158
-     * @param EEG_Aim $gateway_object
159
-     * @return string
160
-     */
161
-    public function possibly_use_deprecated_aim_server($url, EEG_Aim $gateway_object)
162
-    {
163
-        if ($gateway_object->_server === 'authorize.net' && ! $gateway_object->_debug_mode) {
164
-            return 'https://secure.authorize.net/gateway/transact.dll';
165
-        } else {
166
-            return $url;
167
-        }
168
-    }
169
-
170
-
171
-    /**
172
-     * Asks the gateway to do whatever it does to process the payment. Onsite gateways will
173
-     * usually send a request directly to the payment provider and update the payment's status based on that;
174
-     * whereas offsite gateways will usually just update the payment with the URL and query parameters to use
175
-     * for sending the request via http_remote_request()
176
-     *
177
-     * @param EEI_Payment $payment
178
-     * @param array $billing_info {
179
-     *  @type $credit_card string
180
-     *  @type $cvv string
181
-     *  @type $exp_month string
182
-     *  @type $exp_year string
183
-     *  @see parent::do_direct_payment
184
-     * }
185
-     * @return EEI_Payment updated
186
-     */
187
-    public function do_direct_payment($payment, $billing_info = null)
188
-    {
189
-        add_filter('FHEE__EEG_Aim___get_server_url', array($this, 'possibly_use_deprecated_aim_server'), 10, 2);
190
-        // Enable test mode if needed
191
-        // 4007000000027  <-- test successful visa
192
-        // 4222222222222  <-- test failure card number
193
-
194
-        $item_num = 1;
195
-        $transaction = $payment->transaction();
196
-        $gateway_formatter = $this->_get_gateway_formatter();
197
-        $order_description = $gateway_formatter->formatOrderDescription($payment);
198
-        $primary_registrant = $transaction->primary_registration();
199
-        // if we're are charging for the full amount, show the normal line items
200
-        // and the itemized total adds up properly
201
-        if ($this->_can_easily_itemize_transaction_for($payment)) {
202
-            $total_line_item = $transaction->total_line_item();
203
-            foreach ($total_line_item->get_items() as $line_item) {
204
-                if ($line_item->quantity() == 0) {
205
-                    continue;
206
-                }
207
-                $this->addLineItem(
208
-                    $item_num++,
209
-                    $gateway_formatter->formatLineItemName($line_item, $payment),
210
-                    $gateway_formatter->formatLineItemDesc($line_item, $payment),
211
-                    $line_item->quantity(),
212
-                    $line_item->unit_price(),
213
-                    'N'
214
-                );
215
-                $order_description .= $line_item->desc().', ';
216
-            }
217
-            foreach ($total_line_item->tax_descendants() as $tax_line_item) {
218
-                $this->addLineItem(
219
-                    $item_num++,
220
-                    $tax_line_item->name(),
221
-                    $tax_line_item->desc(),
222
-                    1,
223
-                    $tax_line_item->total(),
224
-                    'N'
225
-                );
226
-            }
227
-        }
228
-
229
-        // start transaction
230
-        // if in debug mode, use authorize.net's sandbox id; otherwise use the Event Espresso partner id
231
-        $partner_id = $this->_debug_mode ? 'AAA100302' : 'AAA105363';
232
-        $this->setField('solution_id', $partner_id);
233
-        $this->setField('amount', $gateway_formatter->formatCurrency($payment->amount()));
234
-        $this->setField('description', substr(rtrim($order_description, ', '), 0, 255));
235
-        $this->_set_sensitive_billing_data($billing_info);
236
-        $this->setField('first_name', $billing_info['first_name']);
237
-        $this->setField('last_name', $billing_info['last_name']);
238
-        $this->setField('email', $billing_info['email']);
239
-        $this->setField('company', $billing_info['company']);
240
-        $this->setField('address', $billing_info['address'].' '.$billing_info['address2']);
241
-        $this->setField('city', $billing_info['city']);
242
-        $this->setField('state', $billing_info['state']);
243
-        $this->setField('country', $billing_info['country']);
244
-        $this->setField('zip', $billing_info['zip']);
245
-        $this->setField('fax', $billing_info['fax']);
246
-        $this->setField('cust_id', $primary_registrant->ID());
247
-        $this->setField('phone', $billing_info['phone']);
248
-        // invoice_num would be nice to have it be unique per SPCO page-load, that way if users
249
-        // press back, they don't submit a duplicate. However, we may be keeping the user on teh same spco page
250
-        // in which case, we need to generate teh invoice num per request right here...
251
-        $this->setField('invoice_num', wp_generate_password(12, false));// $billing_info['_reg-page-billing-invoice-'.$this->_gateway_name]['value']);
252
-        // tell AIM that any duplicates sent in the next 5 minutes are to be ignored
253
-        $this->setField('duplicate_window', 5 * MINUTE_IN_SECONDS);
254
-
255
-        if ($this->_test_transactions) {
256
-            $this->test_request = "true";
257
-        }
258
-
259
-        // Capture response
260
-        $this->type = "AUTH_CAPTURE";
261
-        $response = $this->_sendRequest($payment);
262
-        if (! empty($response)) {
263
-            if ($response->error_message) {
264
-                $payment->set_status($this->_pay_model->failed_status());
265
-                $payment->set_gateway_response($response->error_message);
266
-            } else {
267
-                $payment_status = $response->approved
268
-                    ? $this->_pay_model->approved_status()
269
-                    : $this->_pay_model->declined_status();
270
-                $payment->set_status($payment_status);
271
-                // make sure we interpret the AMT as a float, not an international string (where periods are thousand separators)
272
-                $payment->set_amount((float) $response->amount);
273
-                $payment->set_gateway_response(
274
-                    sprintf(
275
-                        esc_html__('%1$s (Reason Code: %2$s)', 'event_espresso'),
276
-                        $response->response_reason_text,
277
-                        $response->response_reason_code
278
-                    )
279
-                );
280
-                if ($this->_debug_mode) {
281
-                    $txn_id = $response->invoice_number;
282
-                } else {
283
-                    $txn_id = $response->transaction_id;
284
-                }
285
-                $payment->set_txn_id_chq_nmbr($txn_id);
286
-            }
287
-            $payment->set_extra_accntng($primary_registrant->reg_code());
288
-            $payment->set_details(print_r($response, true));
289
-        } else {
290
-            $payment->set_status($this->_pay_model->failed_status());
291
-            $payment->set_gateway_response(__("There was no response from Authorize.net", 'event_espresso'));
292
-            $payment->set_details(print_r($response, true));
293
-        }
294
-        return $payment;
295
-    }
296
-
297
-
298
-    /**
299
-     * Sets billing data for the upcoming request to AIM that is considered sensitive;
300
-     * also this method can be overridden by children classes to easily change
301
-     * what billing data gets sent
302
-     *
303
-     * @param array $billing_info
304
-     */
305
-    protected function _set_sensitive_billing_data($billing_info)
306
-    {
307
-        $this->setField('card_num', $billing_info['credit_card']);
308
-        $this->setField('exp_date', $billing_info['exp_month'] . $billing_info['exp_year']);
309
-        $this->setField('card_code', $billing_info['cvv']);
310
-    }
311
-
312
-
313
-    /**
314
-     * Add a line item.
315
-     *
316
-     * @param string $item_id
317
-     * @param string $item_name
318
-     * @param string $item_description
319
-     * @param string $item_quantity
320
-     * @param string $item_unit_price
321
-     * @param string $item_taxable
322
-     */
323
-    public function addLineItem($item_id, $item_name, $item_description, $item_quantity, $item_unit_price, $item_taxable)
324
-    {
325
-        $args = array(
326
-            substr($item_id, 0, 31),
327
-            substr($item_name, 0, 31),
328
-            substr($item_description, 0, 255),
329
-            number_format(abs($item_quantity), 2, '.', ''),
330
-            number_format(abs($item_unit_price), 2, '.', ''),
331
-            $item_taxable === 'N' ? 'N' : 'Y'
332
-        );
333
-        $this->_additional_line_items[] = implode('<|>', $args);
334
-    }
335
-
336
-
337
-    /**
338
-     * Set an individual name/value pair. This will append x_ to the name
339
-     * before posting.
340
-     *
341
-     * @param string $name
342
-     * @param string $value
343
-     * @throws AuthorizeNetException
344
-     */
345
-    protected function setField($name, $value)
346
-    {
347
-        if (in_array($name, $this->_all_aim_fields)) {
348
-            $this->_x_post_fields[ $name ] = $value;
349
-        } else {
350
-            throw new AuthorizeNetException("Error: no field $name exists in the AIM API.
26
+	const LIVE_URL    = 'https://secure2.authorize.net/gateway/transact.dll'; // Authnet URL
27
+
28
+	const SANDBOX_URL = 'https://test.authorize.net/gateway/transact.dll';
29
+
30
+	protected $_login_id;
31
+
32
+	protected $_transaction_key;
33
+
34
+	protected $_server;
35
+
36
+	protected $_currencies_supported = array(
37
+		'AUD',
38
+		'USD',
39
+		'CAD',
40
+		'EUR',
41
+		'GBP',
42
+		'NZD',
43
+	);
44
+
45
+	/**
46
+	 * Whether to send test transactions (even to live site)
47
+	 *
48
+	 * @var boolean
49
+	 */
50
+	protected $_test_transactions;
51
+
52
+	private $VERIFY_PEER = false;
53
+
54
+	private $_x_post_fields = array(
55
+		"version"        => "3.1",
56
+		"delim_char"     => ",",
57
+		"delim_data"     => "TRUE",
58
+		"relay_response" => "FALSE",
59
+		"encap_char"     => "|",
60
+	);
61
+
62
+	private $_additional_line_items = array();
63
+
64
+	/**
65
+	 * A list of all fields in the AIM API.
66
+	 * Used to warn user if they try to set a field not offered in the API.
67
+	 */
68
+	private $_all_aim_fields = array(
69
+		"address",
70
+		"allow_partial_auth",
71
+		"amount",
72
+		"auth_code",
73
+		"authentication_indicator",
74
+		"bank_aba_code",
75
+		"bank_acct_name",
76
+		"bank_acct_num",
77
+		"bank_acct_type",
78
+		"bank_check_number",
79
+		"bank_name",
80
+		"card_code",
81
+		"card_num",
82
+		"cardholder_authentication_value",
83
+		"city",
84
+		"company",
85
+		"country",
86
+		"cust_id",
87
+		"customer_ip",
88
+		"delim_char",
89
+		"delim_data",
90
+		"description",
91
+		"duplicate_window",
92
+		"duty",
93
+		"echeck_type",
94
+		"email",
95
+		"email_customer",
96
+		"encap_char",
97
+		"exp_date",
98
+		"fax",
99
+		"first_name",
100
+		"footer_email_receipt",
101
+		"freight",
102
+		"header_email_receipt",
103
+		"invoice_num",
104
+		"last_name",
105
+		"line_item",
106
+		"login",
107
+		"method",
108
+		"phone",
109
+		"po_num",
110
+		"recurring_billing",
111
+		"relay_response",
112
+		"ship_to_address",
113
+		"ship_to_city",
114
+		"ship_to_company",
115
+		"ship_to_country",
116
+		"ship_to_first_name",
117
+		"ship_to_last_name",
118
+		"ship_to_state",
119
+		"ship_to_zip",
120
+		"split_tender_id",
121
+		"state",
122
+		"tax",
123
+		"tax_exempt",
124
+		"test_request",
125
+		"tran_key",
126
+		"trans_id",
127
+		"type",
128
+		"version",
129
+		"zip",
130
+		"solution_id"
131
+	);
132
+
133
+
134
+	/**
135
+	 * Gets the URL where the request should go. This is filterable
136
+	 *
137
+	 * @return string
138
+	 */
139
+	protected function _get_server_url()
140
+	{
141
+		return apply_filters(
142
+			'FHEE__EEG_Aim___get_server_url',
143
+			$this->_debug_mode ? self::SANDBOX_URL : self::LIVE_URL,
144
+			$this
145
+		);
146
+	}
147
+
148
+
149
+	/**
150
+	 * TEMPORARY CALLBACK! Do not use
151
+	 * Callback which filters the server url. This is added so site admins can revert to using
152
+	 * the old AIM server in case Akamai service breaks their integration.
153
+	 * Using Akamai will, however, be mandatory on June 30th 2016 Authorize.net
154
+	 * (see http://www.authorize.net/support/akamaifaqs/#firewall?utm_campaign=April%202016%20Technical%20Updates%20for%20Merchants.html&utm_medium=email&utm_source=Eloqua&elqTrackId=46103bdc375c411a979c2f658fc99074&elq=7026706360154fee9b6d588b27d8eb6a&elqaid=506&elqat=1&elqCampaignId=343)
155
+	 * Once that happens, this will be obsolete and WILL BE REMOVED.
156
+	 *
157
+	 * @param string $url
158
+	 * @param EEG_Aim $gateway_object
159
+	 * @return string
160
+	 */
161
+	public function possibly_use_deprecated_aim_server($url, EEG_Aim $gateway_object)
162
+	{
163
+		if ($gateway_object->_server === 'authorize.net' && ! $gateway_object->_debug_mode) {
164
+			return 'https://secure.authorize.net/gateway/transact.dll';
165
+		} else {
166
+			return $url;
167
+		}
168
+	}
169
+
170
+
171
+	/**
172
+	 * Asks the gateway to do whatever it does to process the payment. Onsite gateways will
173
+	 * usually send a request directly to the payment provider and update the payment's status based on that;
174
+	 * whereas offsite gateways will usually just update the payment with the URL and query parameters to use
175
+	 * for sending the request via http_remote_request()
176
+	 *
177
+	 * @param EEI_Payment $payment
178
+	 * @param array $billing_info {
179
+	 *  @type $credit_card string
180
+	 *  @type $cvv string
181
+	 *  @type $exp_month string
182
+	 *  @type $exp_year string
183
+	 *  @see parent::do_direct_payment
184
+	 * }
185
+	 * @return EEI_Payment updated
186
+	 */
187
+	public function do_direct_payment($payment, $billing_info = null)
188
+	{
189
+		add_filter('FHEE__EEG_Aim___get_server_url', array($this, 'possibly_use_deprecated_aim_server'), 10, 2);
190
+		// Enable test mode if needed
191
+		// 4007000000027  <-- test successful visa
192
+		// 4222222222222  <-- test failure card number
193
+
194
+		$item_num = 1;
195
+		$transaction = $payment->transaction();
196
+		$gateway_formatter = $this->_get_gateway_formatter();
197
+		$order_description = $gateway_formatter->formatOrderDescription($payment);
198
+		$primary_registrant = $transaction->primary_registration();
199
+		// if we're are charging for the full amount, show the normal line items
200
+		// and the itemized total adds up properly
201
+		if ($this->_can_easily_itemize_transaction_for($payment)) {
202
+			$total_line_item = $transaction->total_line_item();
203
+			foreach ($total_line_item->get_items() as $line_item) {
204
+				if ($line_item->quantity() == 0) {
205
+					continue;
206
+				}
207
+				$this->addLineItem(
208
+					$item_num++,
209
+					$gateway_formatter->formatLineItemName($line_item, $payment),
210
+					$gateway_formatter->formatLineItemDesc($line_item, $payment),
211
+					$line_item->quantity(),
212
+					$line_item->unit_price(),
213
+					'N'
214
+				);
215
+				$order_description .= $line_item->desc().', ';
216
+			}
217
+			foreach ($total_line_item->tax_descendants() as $tax_line_item) {
218
+				$this->addLineItem(
219
+					$item_num++,
220
+					$tax_line_item->name(),
221
+					$tax_line_item->desc(),
222
+					1,
223
+					$tax_line_item->total(),
224
+					'N'
225
+				);
226
+			}
227
+		}
228
+
229
+		// start transaction
230
+		// if in debug mode, use authorize.net's sandbox id; otherwise use the Event Espresso partner id
231
+		$partner_id = $this->_debug_mode ? 'AAA100302' : 'AAA105363';
232
+		$this->setField('solution_id', $partner_id);
233
+		$this->setField('amount', $gateway_formatter->formatCurrency($payment->amount()));
234
+		$this->setField('description', substr(rtrim($order_description, ', '), 0, 255));
235
+		$this->_set_sensitive_billing_data($billing_info);
236
+		$this->setField('first_name', $billing_info['first_name']);
237
+		$this->setField('last_name', $billing_info['last_name']);
238
+		$this->setField('email', $billing_info['email']);
239
+		$this->setField('company', $billing_info['company']);
240
+		$this->setField('address', $billing_info['address'].' '.$billing_info['address2']);
241
+		$this->setField('city', $billing_info['city']);
242
+		$this->setField('state', $billing_info['state']);
243
+		$this->setField('country', $billing_info['country']);
244
+		$this->setField('zip', $billing_info['zip']);
245
+		$this->setField('fax', $billing_info['fax']);
246
+		$this->setField('cust_id', $primary_registrant->ID());
247
+		$this->setField('phone', $billing_info['phone']);
248
+		// invoice_num would be nice to have it be unique per SPCO page-load, that way if users
249
+		// press back, they don't submit a duplicate. However, we may be keeping the user on teh same spco page
250
+		// in which case, we need to generate teh invoice num per request right here...
251
+		$this->setField('invoice_num', wp_generate_password(12, false));// $billing_info['_reg-page-billing-invoice-'.$this->_gateway_name]['value']);
252
+		// tell AIM that any duplicates sent in the next 5 minutes are to be ignored
253
+		$this->setField('duplicate_window', 5 * MINUTE_IN_SECONDS);
254
+
255
+		if ($this->_test_transactions) {
256
+			$this->test_request = "true";
257
+		}
258
+
259
+		// Capture response
260
+		$this->type = "AUTH_CAPTURE";
261
+		$response = $this->_sendRequest($payment);
262
+		if (! empty($response)) {
263
+			if ($response->error_message) {
264
+				$payment->set_status($this->_pay_model->failed_status());
265
+				$payment->set_gateway_response($response->error_message);
266
+			} else {
267
+				$payment_status = $response->approved
268
+					? $this->_pay_model->approved_status()
269
+					: $this->_pay_model->declined_status();
270
+				$payment->set_status($payment_status);
271
+				// make sure we interpret the AMT as a float, not an international string (where periods are thousand separators)
272
+				$payment->set_amount((float) $response->amount);
273
+				$payment->set_gateway_response(
274
+					sprintf(
275
+						esc_html__('%1$s (Reason Code: %2$s)', 'event_espresso'),
276
+						$response->response_reason_text,
277
+						$response->response_reason_code
278
+					)
279
+				);
280
+				if ($this->_debug_mode) {
281
+					$txn_id = $response->invoice_number;
282
+				} else {
283
+					$txn_id = $response->transaction_id;
284
+				}
285
+				$payment->set_txn_id_chq_nmbr($txn_id);
286
+			}
287
+			$payment->set_extra_accntng($primary_registrant->reg_code());
288
+			$payment->set_details(print_r($response, true));
289
+		} else {
290
+			$payment->set_status($this->_pay_model->failed_status());
291
+			$payment->set_gateway_response(__("There was no response from Authorize.net", 'event_espresso'));
292
+			$payment->set_details(print_r($response, true));
293
+		}
294
+		return $payment;
295
+	}
296
+
297
+
298
+	/**
299
+	 * Sets billing data for the upcoming request to AIM that is considered sensitive;
300
+	 * also this method can be overridden by children classes to easily change
301
+	 * what billing data gets sent
302
+	 *
303
+	 * @param array $billing_info
304
+	 */
305
+	protected function _set_sensitive_billing_data($billing_info)
306
+	{
307
+		$this->setField('card_num', $billing_info['credit_card']);
308
+		$this->setField('exp_date', $billing_info['exp_month'] . $billing_info['exp_year']);
309
+		$this->setField('card_code', $billing_info['cvv']);
310
+	}
311
+
312
+
313
+	/**
314
+	 * Add a line item.
315
+	 *
316
+	 * @param string $item_id
317
+	 * @param string $item_name
318
+	 * @param string $item_description
319
+	 * @param string $item_quantity
320
+	 * @param string $item_unit_price
321
+	 * @param string $item_taxable
322
+	 */
323
+	public function addLineItem($item_id, $item_name, $item_description, $item_quantity, $item_unit_price, $item_taxable)
324
+	{
325
+		$args = array(
326
+			substr($item_id, 0, 31),
327
+			substr($item_name, 0, 31),
328
+			substr($item_description, 0, 255),
329
+			number_format(abs($item_quantity), 2, '.', ''),
330
+			number_format(abs($item_unit_price), 2, '.', ''),
331
+			$item_taxable === 'N' ? 'N' : 'Y'
332
+		);
333
+		$this->_additional_line_items[] = implode('<|>', $args);
334
+	}
335
+
336
+
337
+	/**
338
+	 * Set an individual name/value pair. This will append x_ to the name
339
+	 * before posting.
340
+	 *
341
+	 * @param string $name
342
+	 * @param string $value
343
+	 * @throws AuthorizeNetException
344
+	 */
345
+	protected function setField($name, $value)
346
+	{
347
+		if (in_array($name, $this->_all_aim_fields)) {
348
+			$this->_x_post_fields[ $name ] = $value;
349
+		} else {
350
+			throw new AuthorizeNetException("Error: no field $name exists in the AIM API.
351 351
             To set a custom field use setCustomField('field','value') instead.");
352
-        }
353
-    }
354
-
355
-
356
-    /**
357
-     * Posts the request to AuthorizeNet & returns response.
358
-     *
359
-     * @param $payment
360
-     * @return \EE_AuthorizeNetAIM_Response
361
-     */
362
-    private function _sendRequest($payment)
363
-    {
364
-        $this->_x_post_fields['login'] = $this->_login_id;
365
-        $this->_x_post_fields['tran_key'] = $this->_transaction_key;
366
-        $x_keys = array();
367
-        foreach ($this->_x_post_fields as $key => $value) {
368
-            $x_keys[] = "x_$key=" . urlencode($this->_get_unsupported_character_remover()->format($value));
369
-        }
370
-        // Add line items
371
-        foreach ($this->_additional_line_items as $key => $value) {
372
-            $x_keys[] =  "x_line_item=" . urlencode($this->_get_unsupported_character_remover()->format($value));
373
-        }
374
-        $this->_log_clean_request($x_keys, $payment);
375
-        $post_url = $this->_get_server_url();
376
-        $curl_request = curl_init($post_url);
377
-        $post_body = implode("&", $x_keys);
378
-        curl_setopt($curl_request, CURLOPT_POSTFIELDS, $post_body);
379
-        curl_setopt($curl_request, CURLOPT_HEADER, 0);
380
-        curl_setopt($curl_request, CURLOPT_TIMEOUT, 45);
381
-        curl_setopt($curl_request, CURLOPT_RETURNTRANSFER, 1);
382
-        curl_setopt($curl_request, CURLOPT_SSL_VERIFYHOST, 2);
383
-        if ($this->VERIFY_PEER) {
384
-            curl_setopt($curl_request, CURLOPT_CAINFO, dirname(__DIR__) . '/ssl/cert.pem');
385
-        } else {
386
-            curl_setopt($curl_request, CURLOPT_SSL_VERIFYPEER, false);
387
-        }
388
-
389
-        if (preg_match('/xml/', $post_url)) {
390
-            curl_setopt($curl_request, CURLOPT_HTTPHEADER, array("Content-Type: text/xml"));
391
-        }
392
-
393
-        $response = curl_exec($curl_request);
394
-
395
-        curl_close($curl_request);
396
-        $response_obj =  new EE_AuthorizeNetAIM_Response($response);
397
-
398
-        return $this->_log_and_clean_response($response_obj, $payment);
399
-    }
400
-
401
-
402
-    /**
403
-     * Logs the clean data only
404
-     *
405
-     * @param array $request_array
406
-     * @param EEI_Payment $payment
407
-     */
408
-    protected function _log_clean_request($request_array, $payment)
409
-    {
410
-        $keys_to_filter_out = array('x_card_num', 'x_card_code', 'x_exp_date');
411
-        foreach ($request_array as $index => $keyvaltogether) {
412
-            foreach ($keys_to_filter_out as $key) {
413
-                if (strpos($keyvaltogether, $key) === 0) {
414
-                    // found it at the first character
415
-                    // so its one of them
416
-                    unset($request_array[ $index ]);
417
-                }
418
-            }
419
-        }
420
-        $this->log(
421
-            array(
422
-                'AIM Request sent:' => $request_array,
423
-                'Server URL'        => $this->_get_server_url()
424
-            ),
425
-            $payment
426
-        );
427
-    }
428
-
429
-
430
-
431
-    /**
432
-     * Logs the response and cleans it
433
-     *
434
-     * @param EE_AuthorizeNetAIM_Response $response_obj
435
-     * @param EE_Payment                  $payment
436
-     * @return \EE_AuthorizeNetAIM_Response
437
-     */
438
-    private function _log_and_clean_response($response_obj, $payment)
439
-    {
440
-        $response_obj->account_number = '';
441
-        $this->log(array('AIM Response received:' => (array) $response_obj), $payment);
442
-        return $response_obj;
443
-    }
352
+		}
353
+	}
354
+
355
+
356
+	/**
357
+	 * Posts the request to AuthorizeNet & returns response.
358
+	 *
359
+	 * @param $payment
360
+	 * @return \EE_AuthorizeNetAIM_Response
361
+	 */
362
+	private function _sendRequest($payment)
363
+	{
364
+		$this->_x_post_fields['login'] = $this->_login_id;
365
+		$this->_x_post_fields['tran_key'] = $this->_transaction_key;
366
+		$x_keys = array();
367
+		foreach ($this->_x_post_fields as $key => $value) {
368
+			$x_keys[] = "x_$key=" . urlencode($this->_get_unsupported_character_remover()->format($value));
369
+		}
370
+		// Add line items
371
+		foreach ($this->_additional_line_items as $key => $value) {
372
+			$x_keys[] =  "x_line_item=" . urlencode($this->_get_unsupported_character_remover()->format($value));
373
+		}
374
+		$this->_log_clean_request($x_keys, $payment);
375
+		$post_url = $this->_get_server_url();
376
+		$curl_request = curl_init($post_url);
377
+		$post_body = implode("&", $x_keys);
378
+		curl_setopt($curl_request, CURLOPT_POSTFIELDS, $post_body);
379
+		curl_setopt($curl_request, CURLOPT_HEADER, 0);
380
+		curl_setopt($curl_request, CURLOPT_TIMEOUT, 45);
381
+		curl_setopt($curl_request, CURLOPT_RETURNTRANSFER, 1);
382
+		curl_setopt($curl_request, CURLOPT_SSL_VERIFYHOST, 2);
383
+		if ($this->VERIFY_PEER) {
384
+			curl_setopt($curl_request, CURLOPT_CAINFO, dirname(__DIR__) . '/ssl/cert.pem');
385
+		} else {
386
+			curl_setopt($curl_request, CURLOPT_SSL_VERIFYPEER, false);
387
+		}
388
+
389
+		if (preg_match('/xml/', $post_url)) {
390
+			curl_setopt($curl_request, CURLOPT_HTTPHEADER, array("Content-Type: text/xml"));
391
+		}
392
+
393
+		$response = curl_exec($curl_request);
394
+
395
+		curl_close($curl_request);
396
+		$response_obj =  new EE_AuthorizeNetAIM_Response($response);
397
+
398
+		return $this->_log_and_clean_response($response_obj, $payment);
399
+	}
400
+
401
+
402
+	/**
403
+	 * Logs the clean data only
404
+	 *
405
+	 * @param array $request_array
406
+	 * @param EEI_Payment $payment
407
+	 */
408
+	protected function _log_clean_request($request_array, $payment)
409
+	{
410
+		$keys_to_filter_out = array('x_card_num', 'x_card_code', 'x_exp_date');
411
+		foreach ($request_array as $index => $keyvaltogether) {
412
+			foreach ($keys_to_filter_out as $key) {
413
+				if (strpos($keyvaltogether, $key) === 0) {
414
+					// found it at the first character
415
+					// so its one of them
416
+					unset($request_array[ $index ]);
417
+				}
418
+			}
419
+		}
420
+		$this->log(
421
+			array(
422
+				'AIM Request sent:' => $request_array,
423
+				'Server URL'        => $this->_get_server_url()
424
+			),
425
+			$payment
426
+		);
427
+	}
428
+
429
+
430
+
431
+	/**
432
+	 * Logs the response and cleans it
433
+	 *
434
+	 * @param EE_AuthorizeNetAIM_Response $response_obj
435
+	 * @param EE_Payment                  $payment
436
+	 * @return \EE_AuthorizeNetAIM_Response
437
+	 */
438
+	private function _log_and_clean_response($response_obj, $payment)
439
+	{
440
+		$response_obj->account_number = '';
441
+		$this->log(array('AIM Response received:' => (array) $response_obj), $payment);
442
+		return $response_obj;
443
+	}
444 444
 }
445 445
 
446 446
 
@@ -456,192 +456,192 @@  discard block
 block discarded – undo
456 456
 class EE_AuthorizeNetAIM_Response
457 457
 {
458 458
 
459
-    const APPROVED = '1';
460
-    const DECLINED = '2';
461
-    const ERROR = '3';
462
-    const HELD = '4';
463
-
464
-    protected $_x_post_fields = array(
465
-        "version"        => "3.1",
466
-        "delim_char"     => ",",
467
-        "delim_data"     => "TRUE",
468
-        "relay_response" => "FALSE",
469
-        "encap_char"     => "|",
470
-    );
471
-    public $approved;
472
-    public $declined;
473
-    public $error;
474
-    public $held;
475
-    public $response_code;
476
-    public $response_subcode;
477
-    public $response_reason_code;
478
-    public $response_reason_text;
479
-    public $authorization_code;
480
-    public $avs_response;
481
-    public $transaction_id;
482
-    public $invoice_number;
483
-    public $description;
484
-    public $amount;
485
-    public $method;
486
-    public $transaction_type;
487
-    public $customer_id;
488
-    public $first_name;
489
-    public $last_name;
490
-    public $company;
491
-    public $address;
492
-    public $city;
493
-    public $state;
494
-    public $zip_code;
495
-    public $country;
496
-    public $phone;
497
-    public $fax;
498
-    public $email_address;
499
-    public $ship_to_first_name;
500
-    public $ship_to_last_name;
501
-    public $ship_to_company;
502
-    public $ship_to_address;
503
-    public $ship_to_city;
504
-    public $ship_to_state;
505
-    public $ship_to_zip_code;
506
-    public $ship_to_country;
507
-    public $tax;
508
-    public $duty;
509
-    public $freight;
510
-    public $tax_exempt;
511
-    public $purchase_order_number;
512
-    public $md5_hash;
513
-    public $card_code_response;
514
-    public $cavv_response; // cardholder_authentication_verification_response
515
-    public $account_number;
516
-    public $card_type;
517
-    public $split_tender_id;
518
-    public $requested_amount;
519
-    public $balance_on_card;
520
-    public $response; // The response string from AuthorizeNet.
521
-    public $error_message;
522
-    private $_response_array = array(); // An array with the split response.
523
-
524
-
525
-    /**
526
-     * Constructor. Parses the AuthorizeNet response string
527
-     *
528
-     * @param string $response The response from the AuthNet server.
529
-     * @var string   $delimiter The delimiter used (default is ",")
530
-     * @var string   $encap_char The encap_char used (default is "|")
531
-     * @var array    $custom_fields Any custom fields set in the request.
532
-     */
533
-
534
-    public function __construct($response)
535
-    {
536
-        $encap_char = $this->_x_post_fields['encap_char'];
537
-        $delimiter = $this->_x_post_fields['delim_char'];
538
-        if ($response) {
539
-            // Split Array
540
-            $this->response = $response;
541
-            if ($encap_char) {
542
-                $this->_response_array = explode($encap_char . $delimiter . $encap_char, substr($response, 1, -1));
543
-            } else {
544
-                $this->_response_array = explode($delimiter, $response);
545
-            }
546
-
547
-            /**
548
-             * If AuthorizeNet doesn't return a delimited response.
549
-             */
550
-            if (count($this->_response_array) < 10) {
551
-                $this->approved = false;
552
-                $this->error = true;
553
-                $this->error_message = sprintf(
554
-                    esc_html__('Unrecognized response from Authorize.net: %1$s', 'event_espresso'),
555
-                    esc_html($response)
556
-                );
557
-                return;
558
-            }
559
-
560
-
561
-
562
-            // Set all fields
563
-            $this->response_code = $this->_response_array[0];
564
-            $this->response_subcode = $this->_response_array[1];
565
-            $this->response_reason_code = $this->_response_array[2];
566
-            $this->response_reason_text = $this->_response_array[3];
567
-            $this->authorization_code = $this->_response_array[4];
568
-            $this->avs_response = $this->_response_array[5];
569
-            $this->transaction_id = $this->_response_array[6];
570
-            $this->invoice_number = $this->_response_array[7];
571
-            $this->description = $this->_response_array[8];
572
-            $this->amount = $this->_response_array[9];
573
-            $this->method = $this->_response_array[10];
574
-            $this->transaction_type = $this->_response_array[11];
575
-            $this->customer_id = $this->_response_array[12];
576
-            $this->first_name = $this->_response_array[13];
577
-            $this->last_name = $this->_response_array[14];
578
-            $this->company = $this->_response_array[15];
579
-            $this->address = $this->_response_array[16];
580
-            $this->city = $this->_response_array[17];
581
-            $this->state = $this->_response_array[18];
582
-            $this->zip_code = $this->_response_array[19];
583
-            $this->country = $this->_response_array[20];
584
-            $this->phone = $this->_response_array[21];
585
-            $this->fax = $this->_response_array[22];
586
-            $this->email_address = $this->_response_array[23];
587
-            $this->ship_to_first_name = $this->_response_array[24];
588
-            $this->ship_to_last_name = $this->_response_array[25];
589
-            $this->ship_to_company = $this->_response_array[26];
590
-            $this->ship_to_address = $this->_response_array[27];
591
-            $this->ship_to_city = $this->_response_array[28];
592
-            $this->ship_to_state = $this->_response_array[29];
593
-            $this->ship_to_zip_code = $this->_response_array[30];
594
-            $this->ship_to_country = $this->_response_array[31];
595
-            $this->tax = $this->_response_array[32];
596
-            $this->duty = $this->_response_array[33];
597
-            $this->freight = $this->_response_array[34];
598
-            $this->tax_exempt = $this->_response_array[35];
599
-            $this->purchase_order_number = $this->_response_array[36];
600
-            $this->md5_hash = $this->_response_array[37];
601
-            $this->card_code_response = $this->_response_array[38];
602
-            $this->cavv_response = $this->_response_array[39];
603
-            $this->account_number = $this->_response_array[50];
604
-            $this->card_type = $this->_response_array[51];
605
-            $this->split_tender_id = $this->_response_array[52];
606
-            $this->requested_amount = $this->_response_array[53];
607
-            $this->balance_on_card = $this->_response_array[54];
608
-
609
-            $this->approved = ($this->response_code === self::APPROVED);
610
-            $this->declined = ($this->response_code === self::DECLINED);
611
-            $this->error = ($this->response_code === self::ERROR);
612
-            $this->held = ($this->response_code === self::HELD);
613
-        } else {
614
-            $this->approved = false;
615
-            $this->error = true;
616
-            $this->error_message = esc_html__(
617
-                'Error connecting to Authorize.net',
618
-                'event_espresso'
619
-            );
620
-        }
621
-    }
459
+	const APPROVED = '1';
460
+	const DECLINED = '2';
461
+	const ERROR = '3';
462
+	const HELD = '4';
463
+
464
+	protected $_x_post_fields = array(
465
+		"version"        => "3.1",
466
+		"delim_char"     => ",",
467
+		"delim_data"     => "TRUE",
468
+		"relay_response" => "FALSE",
469
+		"encap_char"     => "|",
470
+	);
471
+	public $approved;
472
+	public $declined;
473
+	public $error;
474
+	public $held;
475
+	public $response_code;
476
+	public $response_subcode;
477
+	public $response_reason_code;
478
+	public $response_reason_text;
479
+	public $authorization_code;
480
+	public $avs_response;
481
+	public $transaction_id;
482
+	public $invoice_number;
483
+	public $description;
484
+	public $amount;
485
+	public $method;
486
+	public $transaction_type;
487
+	public $customer_id;
488
+	public $first_name;
489
+	public $last_name;
490
+	public $company;
491
+	public $address;
492
+	public $city;
493
+	public $state;
494
+	public $zip_code;
495
+	public $country;
496
+	public $phone;
497
+	public $fax;
498
+	public $email_address;
499
+	public $ship_to_first_name;
500
+	public $ship_to_last_name;
501
+	public $ship_to_company;
502
+	public $ship_to_address;
503
+	public $ship_to_city;
504
+	public $ship_to_state;
505
+	public $ship_to_zip_code;
506
+	public $ship_to_country;
507
+	public $tax;
508
+	public $duty;
509
+	public $freight;
510
+	public $tax_exempt;
511
+	public $purchase_order_number;
512
+	public $md5_hash;
513
+	public $card_code_response;
514
+	public $cavv_response; // cardholder_authentication_verification_response
515
+	public $account_number;
516
+	public $card_type;
517
+	public $split_tender_id;
518
+	public $requested_amount;
519
+	public $balance_on_card;
520
+	public $response; // The response string from AuthorizeNet.
521
+	public $error_message;
522
+	private $_response_array = array(); // An array with the split response.
523
+
524
+
525
+	/**
526
+	 * Constructor. Parses the AuthorizeNet response string
527
+	 *
528
+	 * @param string $response The response from the AuthNet server.
529
+	 * @var string   $delimiter The delimiter used (default is ",")
530
+	 * @var string   $encap_char The encap_char used (default is "|")
531
+	 * @var array    $custom_fields Any custom fields set in the request.
532
+	 */
533
+
534
+	public function __construct($response)
535
+	{
536
+		$encap_char = $this->_x_post_fields['encap_char'];
537
+		$delimiter = $this->_x_post_fields['delim_char'];
538
+		if ($response) {
539
+			// Split Array
540
+			$this->response = $response;
541
+			if ($encap_char) {
542
+				$this->_response_array = explode($encap_char . $delimiter . $encap_char, substr($response, 1, -1));
543
+			} else {
544
+				$this->_response_array = explode($delimiter, $response);
545
+			}
546
+
547
+			/**
548
+			 * If AuthorizeNet doesn't return a delimited response.
549
+			 */
550
+			if (count($this->_response_array) < 10) {
551
+				$this->approved = false;
552
+				$this->error = true;
553
+				$this->error_message = sprintf(
554
+					esc_html__('Unrecognized response from Authorize.net: %1$s', 'event_espresso'),
555
+					esc_html($response)
556
+				);
557
+				return;
558
+			}
559
+
560
+
561
+
562
+			// Set all fields
563
+			$this->response_code = $this->_response_array[0];
564
+			$this->response_subcode = $this->_response_array[1];
565
+			$this->response_reason_code = $this->_response_array[2];
566
+			$this->response_reason_text = $this->_response_array[3];
567
+			$this->authorization_code = $this->_response_array[4];
568
+			$this->avs_response = $this->_response_array[5];
569
+			$this->transaction_id = $this->_response_array[6];
570
+			$this->invoice_number = $this->_response_array[7];
571
+			$this->description = $this->_response_array[8];
572
+			$this->amount = $this->_response_array[9];
573
+			$this->method = $this->_response_array[10];
574
+			$this->transaction_type = $this->_response_array[11];
575
+			$this->customer_id = $this->_response_array[12];
576
+			$this->first_name = $this->_response_array[13];
577
+			$this->last_name = $this->_response_array[14];
578
+			$this->company = $this->_response_array[15];
579
+			$this->address = $this->_response_array[16];
580
+			$this->city = $this->_response_array[17];
581
+			$this->state = $this->_response_array[18];
582
+			$this->zip_code = $this->_response_array[19];
583
+			$this->country = $this->_response_array[20];
584
+			$this->phone = $this->_response_array[21];
585
+			$this->fax = $this->_response_array[22];
586
+			$this->email_address = $this->_response_array[23];
587
+			$this->ship_to_first_name = $this->_response_array[24];
588
+			$this->ship_to_last_name = $this->_response_array[25];
589
+			$this->ship_to_company = $this->_response_array[26];
590
+			$this->ship_to_address = $this->_response_array[27];
591
+			$this->ship_to_city = $this->_response_array[28];
592
+			$this->ship_to_state = $this->_response_array[29];
593
+			$this->ship_to_zip_code = $this->_response_array[30];
594
+			$this->ship_to_country = $this->_response_array[31];
595
+			$this->tax = $this->_response_array[32];
596
+			$this->duty = $this->_response_array[33];
597
+			$this->freight = $this->_response_array[34];
598
+			$this->tax_exempt = $this->_response_array[35];
599
+			$this->purchase_order_number = $this->_response_array[36];
600
+			$this->md5_hash = $this->_response_array[37];
601
+			$this->card_code_response = $this->_response_array[38];
602
+			$this->cavv_response = $this->_response_array[39];
603
+			$this->account_number = $this->_response_array[50];
604
+			$this->card_type = $this->_response_array[51];
605
+			$this->split_tender_id = $this->_response_array[52];
606
+			$this->requested_amount = $this->_response_array[53];
607
+			$this->balance_on_card = $this->_response_array[54];
608
+
609
+			$this->approved = ($this->response_code === self::APPROVED);
610
+			$this->declined = ($this->response_code === self::DECLINED);
611
+			$this->error = ($this->response_code === self::ERROR);
612
+			$this->held = ($this->response_code === self::HELD);
613
+		} else {
614
+			$this->approved = false;
615
+			$this->error = true;
616
+			$this->error_message = esc_html__(
617
+				'Error connecting to Authorize.net',
618
+				'event_espresso'
619
+			);
620
+		}
621
+	}
622 622
 }
623 623
 
624 624
 if (! class_exists('AuthorizeNetException')) {
625
-    /**
626
-     * Class AuthorizeNetException
627
-     *
628
-     * @package    AuthorizeNet
629
-     */
630
-    class AuthorizeNetException extends Exception
631
-    {
632
-
633
-        /**
634
-         * Construct the exception. Note: The message is NOT binary safe.
635
-         *
636
-         * @link http://php.net/manual/en/exception.construct.php
637
-         * @param string $message [optional] The Exception message to throw.
638
-         * @param int $code [optional] The Exception code.
639
-         * @param Exception $previous [optional] The previous exception used for the exception chaining. Since 5.3.0
640
-         * @since 5.1.0
641
-         */
642
-        public function __construct($message = "", $code = 0, Exception $previous = null)
643
-        {
644
-            parent::__construct($message, $code, $previous);
645
-        }
646
-    }
625
+	/**
626
+	 * Class AuthorizeNetException
627
+	 *
628
+	 * @package    AuthorizeNet
629
+	 */
630
+	class AuthorizeNetException extends Exception
631
+	{
632
+
633
+		/**
634
+		 * Construct the exception. Note: The message is NOT binary safe.
635
+		 *
636
+		 * @link http://php.net/manual/en/exception.construct.php
637
+		 * @param string $message [optional] The Exception message to throw.
638
+		 * @param int $code [optional] The Exception code.
639
+		 * @param Exception $previous [optional] The previous exception used for the exception chaining. Since 5.3.0
640
+		 * @since 5.1.0
641
+		 */
642
+		public function __construct($message = "", $code = 0, Exception $previous = null)
643
+		{
644
+			parent::__construct($message, $code, $previous);
645
+		}
646
+	}
647 647
 }
Please login to merge, or discard this patch.
Spacing   +11 added lines, -11 removed lines patch added patch discarded remove patch
@@ -248,7 +248,7 @@  discard block
 block discarded – undo
248 248
         // invoice_num would be nice to have it be unique per SPCO page-load, that way if users
249 249
         // press back, they don't submit a duplicate. However, we may be keeping the user on teh same spco page
250 250
         // in which case, we need to generate teh invoice num per request right here...
251
-        $this->setField('invoice_num', wp_generate_password(12, false));// $billing_info['_reg-page-billing-invoice-'.$this->_gateway_name]['value']);
251
+        $this->setField('invoice_num', wp_generate_password(12, false)); // $billing_info['_reg-page-billing-invoice-'.$this->_gateway_name]['value']);
252 252
         // tell AIM that any duplicates sent in the next 5 minutes are to be ignored
253 253
         $this->setField('duplicate_window', 5 * MINUTE_IN_SECONDS);
254 254
 
@@ -259,7 +259,7 @@  discard block
 block discarded – undo
259 259
         // Capture response
260 260
         $this->type = "AUTH_CAPTURE";
261 261
         $response = $this->_sendRequest($payment);
262
-        if (! empty($response)) {
262
+        if ( ! empty($response)) {
263 263
             if ($response->error_message) {
264 264
                 $payment->set_status($this->_pay_model->failed_status());
265 265
                 $payment->set_gateway_response($response->error_message);
@@ -305,7 +305,7 @@  discard block
 block discarded – undo
305 305
     protected function _set_sensitive_billing_data($billing_info)
306 306
     {
307 307
         $this->setField('card_num', $billing_info['credit_card']);
308
-        $this->setField('exp_date', $billing_info['exp_month'] . $billing_info['exp_year']);
308
+        $this->setField('exp_date', $billing_info['exp_month'].$billing_info['exp_year']);
309 309
         $this->setField('card_code', $billing_info['cvv']);
310 310
     }
311 311
 
@@ -345,7 +345,7 @@  discard block
 block discarded – undo
345 345
     protected function setField($name, $value)
346 346
     {
347 347
         if (in_array($name, $this->_all_aim_fields)) {
348
-            $this->_x_post_fields[ $name ] = $value;
348
+            $this->_x_post_fields[$name] = $value;
349 349
         } else {
350 350
             throw new AuthorizeNetException("Error: no field $name exists in the AIM API.
351 351
             To set a custom field use setCustomField('field','value') instead.");
@@ -365,11 +365,11 @@  discard block
 block discarded – undo
365 365
         $this->_x_post_fields['tran_key'] = $this->_transaction_key;
366 366
         $x_keys = array();
367 367
         foreach ($this->_x_post_fields as $key => $value) {
368
-            $x_keys[] = "x_$key=" . urlencode($this->_get_unsupported_character_remover()->format($value));
368
+            $x_keys[] = "x_$key=".urlencode($this->_get_unsupported_character_remover()->format($value));
369 369
         }
370 370
         // Add line items
371 371
         foreach ($this->_additional_line_items as $key => $value) {
372
-            $x_keys[] =  "x_line_item=" . urlencode($this->_get_unsupported_character_remover()->format($value));
372
+            $x_keys[] = "x_line_item=".urlencode($this->_get_unsupported_character_remover()->format($value));
373 373
         }
374 374
         $this->_log_clean_request($x_keys, $payment);
375 375
         $post_url = $this->_get_server_url();
@@ -381,7 +381,7 @@  discard block
 block discarded – undo
381 381
         curl_setopt($curl_request, CURLOPT_RETURNTRANSFER, 1);
382 382
         curl_setopt($curl_request, CURLOPT_SSL_VERIFYHOST, 2);
383 383
         if ($this->VERIFY_PEER) {
384
-            curl_setopt($curl_request, CURLOPT_CAINFO, dirname(__DIR__) . '/ssl/cert.pem');
384
+            curl_setopt($curl_request, CURLOPT_CAINFO, dirname(__DIR__).'/ssl/cert.pem');
385 385
         } else {
386 386
             curl_setopt($curl_request, CURLOPT_SSL_VERIFYPEER, false);
387 387
         }
@@ -393,7 +393,7 @@  discard block
 block discarded – undo
393 393
         $response = curl_exec($curl_request);
394 394
 
395 395
         curl_close($curl_request);
396
-        $response_obj =  new EE_AuthorizeNetAIM_Response($response);
396
+        $response_obj = new EE_AuthorizeNetAIM_Response($response);
397 397
 
398 398
         return $this->_log_and_clean_response($response_obj, $payment);
399 399
     }
@@ -413,7 +413,7 @@  discard block
 block discarded – undo
413 413
                 if (strpos($keyvaltogether, $key) === 0) {
414 414
                     // found it at the first character
415 415
                     // so its one of them
416
-                    unset($request_array[ $index ]);
416
+                    unset($request_array[$index]);
417 417
                 }
418 418
             }
419 419
         }
@@ -539,7 +539,7 @@  discard block
 block discarded – undo
539 539
             // Split Array
540 540
             $this->response = $response;
541 541
             if ($encap_char) {
542
-                $this->_response_array = explode($encap_char . $delimiter . $encap_char, substr($response, 1, -1));
542
+                $this->_response_array = explode($encap_char.$delimiter.$encap_char, substr($response, 1, -1));
543 543
             } else {
544 544
                 $this->_response_array = explode($delimiter, $response);
545 545
             }
@@ -621,7 +621,7 @@  discard block
 block discarded – undo
621 621
     }
622 622
 }
623 623
 
624
-if (! class_exists('AuthorizeNetException')) {
624
+if ( ! class_exists('AuthorizeNetException')) {
625 625
     /**
626 626
      * Class AuthorizeNetException
627 627
      *
Please login to merge, or discard this patch.
core/EE_Load_Espresso_Core.core.php 2 patches
Unused Use Statements   -5 removed lines patch added patch discarded remove patch
@@ -1,12 +1,7 @@
 block discarded – undo
1 1
 <?php
2 2
 
3
-use EventEspresso\core\domain\DomainFactory;
4
-use EventEspresso\core\domain\values\FilePath;
5
-use EventEspresso\core\domain\values\FullyQualifiedName;
6
-use EventEspresso\core\domain\values\Version;
7 3
 use EventEspresso\core\exceptions\InvalidDataTypeException;
8 4
 use EventEspresso\core\exceptions\InvalidInterfaceException;
9
-use EventEspresso\core\services\loaders\LoaderFactory;
10 5
 use EventEspresso\core\services\request\RequestDecoratorInterface;
11 6
 use EventEspresso\core\services\request\RequestInterface;
12 7
 use EventEspresso\core\services\request\RequestStackCoreAppInterface;
Please login to merge, or discard this patch.
Indentation   +110 added lines, -110 removed lines patch added patch discarded remove patch
@@ -27,114 +27,114 @@
 block discarded – undo
27 27
 class EE_Load_Espresso_Core implements RequestDecoratorInterface, RequestStackCoreAppInterface
28 28
 {
29 29
 
30
-    /**
31
-     * @var RequestInterface $request
32
-     */
33
-    protected $request;
34
-
35
-    /**
36
-     * @var ResponseInterface $response
37
-     */
38
-    protected $response;
39
-
40
-    /**
41
-     * @var EE_Dependency_Map $dependency_map
42
-     */
43
-    protected $dependency_map;
44
-
45
-    /**
46
-     * @var EE_Registry $registry
47
-     */
48
-    protected $registry;
49
-
50
-
51
-    /**
52
-     * EE_Load_Espresso_Core constructor
53
-     *
54
-     * @param EE_Registry       $registry
55
-     * @param EE_Dependency_Map $dependency_map
56
-     * @throws EE_Error
57
-     */
58
-    public function __construct(EE_Registry $registry, EE_Dependency_Map $dependency_map)
59
-    {
60
-        EE_Error::doing_it_wrong(
61
-            __METHOD__,
62
-            sprintf(
63
-                esc_html__(
64
-                    'This class is deprecated. Please use %1$s instead. All Event Espresso request stack classes have been moved to %2$s and are now under the %3$s namespace',
65
-                    'event_espresso'
66
-                ),
67
-                'EventEspresso\core\services\request\RequestStackCoreApp',
68
-                '\core\services\request',
69
-                'EventEspresso\core\services\request'
70
-            ),
71
-            '4.9.53'
72
-        );
73
-    }
74
-
75
-
76
-    /**
77
-     * handle
78
-     * sets hooks for running rest of system
79
-     * provides "AHEE__EE_System__construct__complete" hook for EE Addons to use as their starting point
80
-     * starting EE Addons from any other point may lead to problems
81
-     *
82
-     * @param RequestInterface  $request
83
-     * @param ResponseInterface $response
84
-     * @return ResponseInterface
85
-     * @throws EE_Error
86
-     * @throws InvalidDataTypeException
87
-     * @throws InvalidInterfaceException
88
-     * @throws InvalidArgumentException
89
-     * @throws DomainException
90
-     */
91
-    public function handleRequest(RequestInterface $request, ResponseInterface $response)
92
-    {
93
-    }
94
-
95
-
96
-    /**
97
-     * @return RequestInterface
98
-     */
99
-    public function request()
100
-    {
101
-    }
102
-
103
-
104
-    /**
105
-     * @return ResponseInterface
106
-     */
107
-    public function response()
108
-    {
109
-    }
110
-
111
-
112
-    /**
113
-     * @return EE_Dependency_Map
114
-     * @throws EE_Error
115
-     */
116
-    public function dependency_map()
117
-    {
118
-    }
119
-
120
-
121
-    /**
122
-     * @return EE_Registry
123
-     * @throws EE_Error
124
-     */
125
-    public function registry()
126
-    {
127
-    }
128
-
129
-
130
-    /**
131
-     * called after the request stack has been fully processed
132
-     * if any of the middleware apps has requested the plugin be deactivated, then we do that now
133
-     *
134
-     * @param RequestInterface  $request
135
-     * @param ResponseInterface $response
136
-     */
137
-    public function handleResponse(RequestInterface $request, ResponseInterface $response)
138
-    {
139
-    }
30
+	/**
31
+	 * @var RequestInterface $request
32
+	 */
33
+	protected $request;
34
+
35
+	/**
36
+	 * @var ResponseInterface $response
37
+	 */
38
+	protected $response;
39
+
40
+	/**
41
+	 * @var EE_Dependency_Map $dependency_map
42
+	 */
43
+	protected $dependency_map;
44
+
45
+	/**
46
+	 * @var EE_Registry $registry
47
+	 */
48
+	protected $registry;
49
+
50
+
51
+	/**
52
+	 * EE_Load_Espresso_Core constructor
53
+	 *
54
+	 * @param EE_Registry       $registry
55
+	 * @param EE_Dependency_Map $dependency_map
56
+	 * @throws EE_Error
57
+	 */
58
+	public function __construct(EE_Registry $registry, EE_Dependency_Map $dependency_map)
59
+	{
60
+		EE_Error::doing_it_wrong(
61
+			__METHOD__,
62
+			sprintf(
63
+				esc_html__(
64
+					'This class is deprecated. Please use %1$s instead. All Event Espresso request stack classes have been moved to %2$s and are now under the %3$s namespace',
65
+					'event_espresso'
66
+				),
67
+				'EventEspresso\core\services\request\RequestStackCoreApp',
68
+				'\core\services\request',
69
+				'EventEspresso\core\services\request'
70
+			),
71
+			'4.9.53'
72
+		);
73
+	}
74
+
75
+
76
+	/**
77
+	 * handle
78
+	 * sets hooks for running rest of system
79
+	 * provides "AHEE__EE_System__construct__complete" hook for EE Addons to use as their starting point
80
+	 * starting EE Addons from any other point may lead to problems
81
+	 *
82
+	 * @param RequestInterface  $request
83
+	 * @param ResponseInterface $response
84
+	 * @return ResponseInterface
85
+	 * @throws EE_Error
86
+	 * @throws InvalidDataTypeException
87
+	 * @throws InvalidInterfaceException
88
+	 * @throws InvalidArgumentException
89
+	 * @throws DomainException
90
+	 */
91
+	public function handleRequest(RequestInterface $request, ResponseInterface $response)
92
+	{
93
+	}
94
+
95
+
96
+	/**
97
+	 * @return RequestInterface
98
+	 */
99
+	public function request()
100
+	{
101
+	}
102
+
103
+
104
+	/**
105
+	 * @return ResponseInterface
106
+	 */
107
+	public function response()
108
+	{
109
+	}
110
+
111
+
112
+	/**
113
+	 * @return EE_Dependency_Map
114
+	 * @throws EE_Error
115
+	 */
116
+	public function dependency_map()
117
+	{
118
+	}
119
+
120
+
121
+	/**
122
+	 * @return EE_Registry
123
+	 * @throws EE_Error
124
+	 */
125
+	public function registry()
126
+	{
127
+	}
128
+
129
+
130
+	/**
131
+	 * called after the request stack has been fully processed
132
+	 * if any of the middleware apps has requested the plugin be deactivated, then we do that now
133
+	 *
134
+	 * @param RequestInterface  $request
135
+	 * @param ResponseInterface $response
136
+	 */
137
+	public function handleResponse(RequestInterface $request, ResponseInterface $response)
138
+	{
139
+	}
140 140
 }
Please login to merge, or discard this patch.
admin_pages/messages/Messages_Admin_Page.core.php 3 patches
Doc Comments   +2 added lines, -2 removed lines patch added patch discarded remove patch
@@ -2597,7 +2597,7 @@  discard block
 block discarded – undo
2597 2597
      *
2598 2598
      * @param int $index This helps us know which template field to select from the request array.
2599 2599
      *
2600
-     * @return array
2600
+     * @return string
2601 2601
      */
2602 2602
     protected function _set_message_template_column_values($index)
2603 2603
     {
@@ -3088,7 +3088,7 @@  discard block
 block discarded – undo
3088 3088
      *
3089 3089
      * @param  int  $GRP_ID        The group being deleted
3090 3090
      * @param  bool $include_group whether to delete the Message Template Group as well.
3091
-     * @return bool boolean to indicate the success of the deletes or not.
3091
+     * @return integer boolean to indicate the success of the deletes or not.
3092 3092
      * @throws EE_Error
3093 3093
      * @throws InvalidArgumentException
3094 3094
      * @throws InvalidDataTypeException
Please login to merge, or discard this patch.
Indentation   +4484 added lines, -4484 removed lines patch added patch discarded remove patch
@@ -19,2588 +19,2588 @@  discard block
 block discarded – undo
19 19
 class Messages_Admin_Page extends EE_Admin_Page
20 20
 {
21 21
 
22
-    /**
23
-     * @type EE_Message_Resource_Manager $_message_resource_manager
24
-     */
25
-    protected $_message_resource_manager;
26
-
27
-    /**
28
-     * @type string $_active_message_type_name
29
-     */
30
-    protected $_active_message_type_name = '';
31
-
32
-    /**
33
-     * @type EE_messenger $_active_messenger
34
-     */
35
-    protected $_active_messenger;
36
-    protected $_activate_state;
37
-    protected $_activate_meta_box_type;
38
-    protected $_current_message_meta_box;
39
-    protected $_current_message_meta_box_object;
40
-    protected $_context_switcher;
41
-    protected $_shortcodes = array();
42
-    protected $_active_messengers = array();
43
-    protected $_active_message_types = array();
44
-
45
-    /**
46
-     * @var EE_Message_Template_Group $_message_template_group
47
-     */
48
-    protected $_message_template_group;
49
-    protected $_m_mt_settings = array();
50
-
51
-
52
-    /**
53
-     * This is set via the _set_message_template_group method and holds whatever the template pack for the group is.
54
-     * IF there is no group then it gets automatically set to the Default template pack.
55
-     *
56
-     * @since 4.5.0
57
-     *
58
-     * @var EE_Messages_Template_Pack
59
-     */
60
-    protected $_template_pack;
61
-
62
-
63
-    /**
64
-     * This is set via the _set_message_template_group method and holds whatever the template pack variation for the
65
-     * group is.  If there is no group then it automatically gets set to default.
66
-     *
67
-     * @since 4.5.0
68
-     *
69
-     * @var string
70
-     */
71
-    protected $_variation;
72
-
73
-
74
-    /**
75
-     * @param bool $routing
76
-     * @throws EE_Error
77
-     */
78
-    public function __construct($routing = true)
79
-    {
80
-        // make sure messages autoloader is running
81
-        EED_Messages::set_autoloaders();
82
-        parent::__construct($routing);
83
-    }
84
-
85
-
86
-    protected function _init_page_props()
87
-    {
88
-        $this->page_slug = EE_MSG_PG_SLUG;
89
-        $this->page_label = esc_html__('Messages Settings', 'event_espresso');
90
-        $this->_admin_base_url = EE_MSG_ADMIN_URL;
91
-        $this->_admin_base_path = EE_MSG_ADMIN;
92
-
93
-        $this->_activate_state = isset($this->_req_data['activate_state']) ? (array) $this->_req_data['activate_state']
94
-            : array();
95
-
96
-        $this->_active_messenger = isset($this->_req_data['messenger']) ? $this->_req_data['messenger'] : null;
97
-        $this->_load_message_resource_manager();
98
-    }
99
-
100
-
101
-    /**
102
-     * loads messenger objects into the $_active_messengers property (so we can access the needed methods)
103
-     *
104
-     * @throws EE_Error
105
-     * @throws InvalidDataTypeException
106
-     * @throws InvalidInterfaceException
107
-     * @throws InvalidArgumentException
108
-     * @throws ReflectionException
109
-     */
110
-    protected function _load_message_resource_manager()
111
-    {
112
-        $this->_message_resource_manager = EE_Registry::instance()->load_lib('Message_Resource_Manager');
113
-    }
114
-
115
-
116
-    /**
117
-     * @deprecated 4.9.9.rc.014
118
-     * @return array
119
-     * @throws EE_Error
120
-     * @throws InvalidArgumentException
121
-     * @throws InvalidDataTypeException
122
-     * @throws InvalidInterfaceException
123
-     */
124
-    public function get_messengers_for_list_table()
125
-    {
126
-        EE_Error::doing_it_wrong(
127
-            __METHOD__,
128
-            sprintf(
129
-                esc_html__(
130
-                    'This method is no longer in use.  There is no replacement for it. The method was used to generate a set of values for use in creating a messenger filter dropdown which is now generated differently via %s',
131
-                    'event_espresso'
132
-                ),
133
-                'Messages_Admin_Page::get_messengers_select_input()'
134
-            ),
135
-            '4.9.9.rc.014'
136
-        );
137
-
138
-        $m_values = array();
139
-        $active_messengers = EEM_Message::instance()->get_all(array('group_by' => 'MSG_messenger'));
140
-        // setup messengers for selects
141
-        $i = 1;
142
-        foreach ($active_messengers as $active_messenger) {
143
-            if ($active_messenger instanceof EE_Message) {
144
-                $m_values[ $i ]['id'] = $active_messenger->messenger();
145
-                $m_values[ $i ]['text'] = ucwords($active_messenger->messenger_label());
146
-                $i++;
147
-            }
148
-        }
149
-
150
-        return $m_values;
151
-    }
152
-
153
-
154
-    /**
155
-     * @deprecated 4.9.9.rc.014
156
-     * @return array
157
-     * @throws EE_Error
158
-     * @throws InvalidArgumentException
159
-     * @throws InvalidDataTypeException
160
-     * @throws InvalidInterfaceException
161
-     */
162
-    public function get_message_types_for_list_table()
163
-    {
164
-        EE_Error::doing_it_wrong(
165
-            __METHOD__,
166
-            sprintf(
167
-                esc_html__(
168
-                    'This method is no longer in use.  There is no replacement for it. The method was used to generate a set of values for use in creating a message type filter dropdown which is now generated differently via %s',
169
-                    'event_espresso'
170
-                ),
171
-                'Messages_Admin_Page::get_message_types_select_input()'
172
-            ),
173
-            '4.9.9.rc.014'
174
-        );
175
-
176
-        $mt_values = array();
177
-        $active_messages = EEM_Message::instance()->get_all(array('group_by' => 'MSG_message_type'));
178
-        $i = 1;
179
-        foreach ($active_messages as $active_message) {
180
-            if ($active_message instanceof EE_Message) {
181
-                $mt_values[ $i ]['id'] = $active_message->message_type();
182
-                $mt_values[ $i ]['text'] = ucwords($active_message->message_type_label());
183
-                $i++;
184
-            }
185
-        }
186
-
187
-        return $mt_values;
188
-    }
189
-
190
-
191
-    /**
192
-     * @deprecated 4.9.9.rc.014
193
-     * @return array
194
-     * @throws EE_Error
195
-     * @throws InvalidArgumentException
196
-     * @throws InvalidDataTypeException
197
-     * @throws InvalidInterfaceException
198
-     */
199
-    public function get_contexts_for_message_types_for_list_table()
200
-    {
201
-        EE_Error::doing_it_wrong(
202
-            __METHOD__,
203
-            sprintf(
204
-                esc_html__(
205
-                    'This method is no longer in use.  There is no replacement for it. The method was used to generate a set of values for use in creating a message type context filter dropdown which is now generated differently via %s',
206
-                    'event_espresso'
207
-                ),
208
-                'Messages_Admin_Page::get_contexts_for_message_types_select_input()'
209
-            ),
210
-            '4.9.9.rc.014'
211
-        );
212
-
213
-        $contexts = array();
214
-        $active_message_contexts = EEM_Message::instance()->get_all(array('group_by' => 'MSG_context'));
215
-        foreach ($active_message_contexts as $active_message) {
216
-            if ($active_message instanceof EE_Message) {
217
-                $message_type = $active_message->message_type_object();
218
-                if ($message_type instanceof EE_message_type) {
219
-                    $message_type_contexts = $message_type->get_contexts();
220
-                    foreach ($message_type_contexts as $context => $context_details) {
221
-                        $contexts[ $context ] = $context_details['label'];
222
-                    }
223
-                }
224
-            }
225
-        }
226
-
227
-        return $contexts;
228
-    }
229
-
230
-
231
-    /**
232
-     * Generate select input with provided messenger options array.
233
-     *
234
-     * @param array $messenger_options Array of messengers indexed by messenger slug and values are the messenger
235
-     *                                 labels.
236
-     * @return string
237
-     * @throws EE_Error
238
-     */
239
-    public function get_messengers_select_input($messenger_options)
240
-    {
241
-        // if empty or just one value then just return an empty string
242
-        if (empty($messenger_options)
243
-            || ! is_array($messenger_options)
244
-            || count($messenger_options) === 1
245
-        ) {
246
-            return '';
247
-        }
248
-        // merge in default
249
-        $messenger_options = array_merge(
250
-            array('none_selected' => esc_html__('Show All Messengers', 'event_espresso')),
251
-            $messenger_options
252
-        );
253
-        $input = new EE_Select_Input(
254
-            $messenger_options,
255
-            array(
256
-                'html_name'  => 'ee_messenger_filter_by',
257
-                'html_id'    => 'ee_messenger_filter_by',
258
-                'html_class' => 'wide',
259
-                'default'    => isset($this->_req_data['ee_messenger_filter_by'])
260
-                    ? sanitize_title($this->_req_data['ee_messenger_filter_by'])
261
-                    : 'none_selected',
262
-            )
263
-        );
264
-
265
-        return $input->get_html_for_input();
266
-    }
267
-
268
-
269
-    /**
270
-     * Generate select input with provided message type options array.
271
-     *
272
-     * @param array $message_type_options Array of message types indexed by message type slug, and values are the
273
-     *                                    message type labels
274
-     * @return string
275
-     * @throws EE_Error
276
-     */
277
-    public function get_message_types_select_input($message_type_options)
278
-    {
279
-        // if empty or count of options is 1 then just return an empty string
280
-        if (empty($message_type_options)
281
-            || ! is_array($message_type_options)
282
-            || count($message_type_options) === 1
283
-        ) {
284
-            return '';
285
-        }
286
-        // merge in default
287
-        $message_type_options = array_merge(
288
-            array('none_selected' => esc_html__('Show All Message Types', 'event_espresso')),
289
-            $message_type_options
290
-        );
291
-        $input = new EE_Select_Input(
292
-            $message_type_options,
293
-            array(
294
-                'html_name'  => 'ee_message_type_filter_by',
295
-                'html_id'    => 'ee_message_type_filter_by',
296
-                'html_class' => 'wide',
297
-                'default'    => isset($this->_req_data['ee_message_type_filter_by'])
298
-                    ? sanitize_title($this->_req_data['ee_message_type_filter_by'])
299
-                    : 'none_selected',
300
-            )
301
-        );
302
-
303
-        return $input->get_html_for_input();
304
-    }
305
-
306
-
307
-    /**
308
-     * Generate select input with provide message type contexts array.
309
-     *
310
-     * @param array $context_options Array of message type contexts indexed by context slug, and values are the
311
-     *                               context label.
312
-     * @return string
313
-     * @throws EE_Error
314
-     */
315
-    public function get_contexts_for_message_types_select_input($context_options)
316
-    {
317
-        // if empty or count of options is one then just return empty string
318
-        if (empty($context_options)
319
-            || ! is_array($context_options)
320
-            || count($context_options) === 1
321
-        ) {
322
-            return '';
323
-        }
324
-        // merge in default
325
-        $context_options = array_merge(
326
-            array('none_selected' => esc_html__('Show all Contexts', 'event_espresso')),
327
-            $context_options
328
-        );
329
-        $input = new EE_Select_Input(
330
-            $context_options,
331
-            array(
332
-                'html_name'  => 'ee_context_filter_by',
333
-                'html_id'    => 'ee_context_filter_by',
334
-                'html_class' => 'wide',
335
-                'default'    => isset($this->_req_data['ee_context_filter_by'])
336
-                    ? sanitize_title($this->_req_data['ee_context_filter_by'])
337
-                    : 'none_selected',
338
-            )
339
-        );
340
-
341
-        return $input->get_html_for_input();
342
-    }
343
-
344
-
345
-    protected function _ajax_hooks()
346
-    {
347
-        add_action('wp_ajax_activate_messenger', array($this, 'activate_messenger_toggle'));
348
-        add_action('wp_ajax_activate_mt', array($this, 'activate_mt_toggle'));
349
-        add_action('wp_ajax_ee_msgs_save_settings', array($this, 'save_settings'));
350
-        add_action('wp_ajax_ee_msgs_update_mt_form', array($this, 'update_mt_form'));
351
-        add_action('wp_ajax_switch_template_pack', array($this, 'switch_template_pack'));
352
-        add_action('wp_ajax_toggle_context_template', array($this, 'toggle_context_template'));
353
-    }
354
-
355
-
356
-    protected function _define_page_props()
357
-    {
358
-        $this->_admin_page_title = $this->page_label;
359
-        $this->_labels = array(
360
-            'buttons'    => array(
361
-                'add'    => esc_html__('Add New Message Template', 'event_espresso'),
362
-                'edit'   => esc_html__('Edit Message Template', 'event_espresso'),
363
-                'delete' => esc_html__('Delete Message Template', 'event_espresso'),
364
-            ),
365
-            'publishbox' => esc_html__('Update Actions', 'event_espresso'),
366
-        );
367
-    }
368
-
369
-
370
-    /**
371
-     *        an array for storing key => value pairs of request actions and their corresponding methods
372
-     *
373
-     * @access protected
374
-     * @return void
375
-     */
376
-    protected function _set_page_routes()
377
-    {
378
-        $grp_id = ! empty($this->_req_data['GRP_ID']) && ! is_array($this->_req_data['GRP_ID'])
379
-            ? $this->_req_data['GRP_ID']
380
-            : 0;
381
-        $grp_id = empty($grp_id) && ! empty($this->_req_data['id'])
382
-            ? $this->_req_data['id']
383
-            : $grp_id;
384
-        $msg_id = ! empty($this->_req_data['MSG_ID']) && ! is_array($this->_req_data['MSG_ID'])
385
-            ? $this->_req_data['MSG_ID']
386
-            : 0;
387
-
388
-        $this->_page_routes = array(
389
-            'default'                          => array(
390
-                'func'       => '_message_queue_list_table',
391
-                'capability' => 'ee_read_global_messages',
392
-            ),
393
-            'global_mtps'                      => array(
394
-                'func'       => '_ee_default_messages_overview_list_table',
395
-                'capability' => 'ee_read_global_messages',
396
-            ),
397
-            'custom_mtps'                      => array(
398
-                'func'       => '_custom_mtps_preview',
399
-                'capability' => 'ee_read_messages',
400
-            ),
401
-            'add_new_message_template'         => array(
402
-                'func'       => '_add_message_template',
403
-                'capability' => 'ee_edit_messages',
404
-                'noheader'   => true,
405
-            ),
406
-            'edit_message_template'            => array(
407
-                'func'       => '_edit_message_template',
408
-                'capability' => 'ee_edit_message',
409
-                'obj_id'     => $grp_id,
410
-            ),
411
-            'preview_message'                  => array(
412
-                'func'               => '_preview_message',
413
-                'capability'         => 'ee_read_message',
414
-                'obj_id'             => $grp_id,
415
-                'noheader'           => true,
416
-                'headers_sent_route' => 'display_preview_message',
417
-            ),
418
-            'display_preview_message'          => array(
419
-                'func'       => '_display_preview_message',
420
-                'capability' => 'ee_read_message',
421
-                'obj_id'     => $grp_id,
422
-            ),
423
-            'insert_message_template'          => array(
424
-                'func'       => '_insert_or_update_message_template',
425
-                'capability' => 'ee_edit_messages',
426
-                'args'       => array('new_template' => true),
427
-                'noheader'   => true,
428
-            ),
429
-            'update_message_template'          => array(
430
-                'func'       => '_insert_or_update_message_template',
431
-                'capability' => 'ee_edit_message',
432
-                'obj_id'     => $grp_id,
433
-                'args'       => array('new_template' => false),
434
-                'noheader'   => true,
435
-            ),
436
-            'trash_message_template'           => array(
437
-                'func'       => '_trash_or_restore_message_template',
438
-                'capability' => 'ee_delete_message',
439
-                'obj_id'     => $grp_id,
440
-                'args'       => array('trash' => true, 'all' => true),
441
-                'noheader'   => true,
442
-            ),
443
-            'trash_message_template_context'   => array(
444
-                'func'       => '_trash_or_restore_message_template',
445
-                'capability' => 'ee_delete_message',
446
-                'obj_id'     => $grp_id,
447
-                'args'       => array('trash' => true),
448
-                'noheader'   => true,
449
-            ),
450
-            'restore_message_template'         => array(
451
-                'func'       => '_trash_or_restore_message_template',
452
-                'capability' => 'ee_delete_message',
453
-                'obj_id'     => $grp_id,
454
-                'args'       => array('trash' => false, 'all' => true),
455
-                'noheader'   => true,
456
-            ),
457
-            'restore_message_template_context' => array(
458
-                'func'       => '_trash_or_restore_message_template',
459
-                'capability' => 'ee_delete_message',
460
-                'obj_id'     => $grp_id,
461
-                'args'       => array('trash' => false),
462
-                'noheader'   => true,
463
-            ),
464
-            'delete_message_template'          => array(
465
-                'func'       => '_delete_message_template',
466
-                'capability' => 'ee_delete_message',
467
-                'obj_id'     => $grp_id,
468
-                'noheader'   => true,
469
-            ),
470
-            'reset_to_default'                 => array(
471
-                'func'       => '_reset_to_default_template',
472
-                'capability' => 'ee_edit_message',
473
-                'obj_id'     => $grp_id,
474
-                'noheader'   => true,
475
-            ),
476
-            'settings'                         => array(
477
-                'func'       => '_settings',
478
-                'capability' => 'manage_options',
479
-            ),
480
-            'update_global_settings'           => array(
481
-                'func'       => '_update_global_settings',
482
-                'capability' => 'manage_options',
483
-                'noheader'   => true,
484
-            ),
485
-            'generate_now'                     => array(
486
-                'func'       => '_generate_now',
487
-                'capability' => 'ee_send_message',
488
-                'noheader'   => true,
489
-            ),
490
-            'generate_and_send_now'            => array(
491
-                'func'       => '_generate_and_send_now',
492
-                'capability' => 'ee_send_message',
493
-                'noheader'   => true,
494
-            ),
495
-            'queue_for_resending'              => array(
496
-                'func'       => '_queue_for_resending',
497
-                'capability' => 'ee_send_message',
498
-                'noheader'   => true,
499
-            ),
500
-            'send_now'                         => array(
501
-                'func'       => '_send_now',
502
-                'capability' => 'ee_send_message',
503
-                'noheader'   => true,
504
-            ),
505
-            'delete_ee_message'                => array(
506
-                'func'       => '_delete_ee_messages',
507
-                'capability' => 'ee_delete_messages',
508
-                'noheader'   => true,
509
-            ),
510
-            'delete_ee_messages'               => array(
511
-                'func'       => '_delete_ee_messages',
512
-                'capability' => 'ee_delete_messages',
513
-                'noheader'   => true,
514
-                'obj_id'     => $msg_id,
515
-            ),
516
-        );
517
-    }
518
-
519
-
520
-    protected function _set_page_config()
521
-    {
522
-        $this->_page_config = array(
523
-            'default'                  => array(
524
-                'nav'           => array(
525
-                    'label' => esc_html__('Message Activity', 'event_espresso'),
526
-                    'order' => 10,
527
-                ),
528
-                'list_table'    => 'EE_Message_List_Table',
529
-                // 'qtips' => array( 'EE_Message_List_Table_Tips' ),
530
-                'require_nonce' => false,
531
-            ),
532
-            'global_mtps'              => array(
533
-                'nav'           => array(
534
-                    'label' => esc_html__('Default Message Templates', 'event_espresso'),
535
-                    'order' => 20,
536
-                ),
537
-                'list_table'    => 'Messages_Template_List_Table',
538
-                'help_tabs'     => array(
539
-                    'messages_overview_help_tab'                                => array(
540
-                        'title'    => esc_html__('Messages Overview', 'event_espresso'),
541
-                        'filename' => 'messages_overview',
542
-                    ),
543
-                    'messages_overview_messages_table_column_headings_help_tab' => array(
544
-                        'title'    => esc_html__('Messages Table Column Headings', 'event_espresso'),
545
-                        'filename' => 'messages_overview_table_column_headings',
546
-                    ),
547
-                    'messages_overview_messages_filters_help_tab'               => array(
548
-                        'title'    => esc_html__('Message Filters', 'event_espresso'),
549
-                        'filename' => 'messages_overview_filters',
550
-                    ),
551
-                    'messages_overview_messages_views_help_tab'                 => array(
552
-                        'title'    => esc_html__('Message Views', 'event_espresso'),
553
-                        'filename' => 'messages_overview_views',
554
-                    ),
555
-                    'message_overview_message_types_help_tab'                   => array(
556
-                        'title'    => esc_html__('Message Types', 'event_espresso'),
557
-                        'filename' => 'messages_overview_types',
558
-                    ),
559
-                    'messages_overview_messengers_help_tab'                     => array(
560
-                        'title'    => esc_html__('Messengers', 'event_espresso'),
561
-                        'filename' => 'messages_overview_messengers',
562
-                    ),
563
-                ),
564
-                'help_tour'     => array('Messages_Overview_Help_Tour'),
565
-                'require_nonce' => false,
566
-            ),
567
-            'custom_mtps'              => array(
568
-                'nav'           => array(
569
-                    'label' => esc_html__('Custom Message Templates', 'event_espresso'),
570
-                    'order' => 30,
571
-                ),
572
-                'help_tabs'     => array(),
573
-                'help_tour'     => array(),
574
-                'require_nonce' => false,
575
-            ),
576
-            'add_new_message_template' => array(
577
-                'nav'           => array(
578
-                    'label'      => esc_html__('Add New Message Templates', 'event_espresso'),
579
-                    'order'      => 5,
580
-                    'persistent' => false,
581
-                ),
582
-                'require_nonce' => false,
583
-            ),
584
-            'edit_message_template'    => array(
585
-                'labels'        => array(
586
-                    'buttons'    => array(
587
-                        'reset' => esc_html__('Reset Templates', 'event_espresso'),
588
-                    ),
589
-                    'publishbox' => esc_html__('Update Actions', 'event_espresso'),
590
-                ),
591
-                'nav'           => array(
592
-                    'label'      => esc_html__('Edit Message Templates', 'event_espresso'),
593
-                    'order'      => 5,
594
-                    'persistent' => false,
595
-                    'url'        => '',
596
-                ),
597
-                'metaboxes'     => array('_publish_post_box', '_register_edit_meta_boxes'),
598
-                'has_metaboxes' => true,
599
-                'help_tour'     => array('Message_Templates_Edit_Help_Tour'),
600
-                'help_tabs'     => array(
601
-                    'edit_message_template'            => array(
602
-                        'title'    => esc_html__('Message Template Editor', 'event_espresso'),
603
-                        'callback' => 'edit_message_template_help_tab',
604
-                    ),
605
-                    'message_templates_help_tab'       => array(
606
-                        'title'    => esc_html__('Message Templates', 'event_espresso'),
607
-                        'filename' => 'messages_templates',
608
-                    ),
609
-                    'message_template_shortcodes'      => array(
610
-                        'title'    => esc_html__('Message Shortcodes', 'event_espresso'),
611
-                        'callback' => 'message_template_shortcodes_help_tab',
612
-                    ),
613
-                    'message_preview_help_tab'         => array(
614
-                        'title'    => esc_html__('Message Preview', 'event_espresso'),
615
-                        'filename' => 'messages_preview',
616
-                    ),
617
-                    'messages_overview_other_help_tab' => array(
618
-                        'title'    => esc_html__('Messages Other', 'event_espresso'),
619
-                        'filename' => 'messages_overview_other',
620
-                    ),
621
-                ),
622
-                'require_nonce' => false,
623
-            ),
624
-            'display_preview_message'  => array(
625
-                'nav'           => array(
626
-                    'label'      => esc_html__('Message Preview', 'event_espresso'),
627
-                    'order'      => 5,
628
-                    'url'        => '',
629
-                    'persistent' => false,
630
-                ),
631
-                'help_tabs'     => array(
632
-                    'preview_message' => array(
633
-                        'title'    => esc_html__('About Previews', 'event_espresso'),
634
-                        'callback' => 'preview_message_help_tab',
635
-                    ),
636
-                ),
637
-                'require_nonce' => false,
638
-            ),
639
-            'settings'                 => array(
640
-                'nav'           => array(
641
-                    'label' => esc_html__('Settings', 'event_espresso'),
642
-                    'order' => 40,
643
-                ),
644
-                'metaboxes'     => array('_messages_settings_metaboxes'),
645
-                'help_tabs'     => array(
646
-                    'messages_settings_help_tab'               => array(
647
-                        'title'    => esc_html__('Messages Settings', 'event_espresso'),
648
-                        'filename' => 'messages_settings',
649
-                    ),
650
-                    'messages_settings_message_types_help_tab' => array(
651
-                        'title'    => esc_html__('Activating / Deactivating Message Types', 'event_espresso'),
652
-                        'filename' => 'messages_settings_message_types',
653
-                    ),
654
-                    'messages_settings_messengers_help_tab'    => array(
655
-                        'title'    => esc_html__('Activating / Deactivating Messengers', 'event_espresso'),
656
-                        'filename' => 'messages_settings_messengers',
657
-                    ),
658
-                ),
659
-                'help_tour'     => array('Messages_Settings_Help_Tour'),
660
-                'require_nonce' => false,
661
-            ),
662
-        );
663
-    }
664
-
665
-
666
-    protected function _add_screen_options()
667
-    {
668
-        // todo
669
-    }
670
-
671
-
672
-    protected function _add_screen_options_global_mtps()
673
-    {
674
-        /**
675
-         * Note: the reason for the value swap here on $this->_admin_page_title is because $this->_per_page_screen_options
676
-         * uses the $_admin_page_title property and we want different outputs in the different spots.
677
-         */
678
-        $page_title = $this->_admin_page_title;
679
-        $this->_admin_page_title = esc_html__('Global Message Templates', 'event_espresso');
680
-        $this->_per_page_screen_option();
681
-        $this->_admin_page_title = $page_title;
682
-    }
683
-
684
-
685
-    protected function _add_screen_options_default()
686
-    {
687
-        $this->_admin_page_title = esc_html__('Message Activity', 'event_espresso');
688
-        $this->_per_page_screen_option();
689
-    }
690
-
691
-
692
-    // none of the below group are currently used for Messages
693
-    protected function _add_feature_pointers()
694
-    {
695
-    }
696
-
697
-    public function admin_init()
698
-    {
699
-    }
700
-
701
-    public function admin_notices()
702
-    {
703
-    }
704
-
705
-    public function admin_footer_scripts()
706
-    {
707
-    }
708
-
709
-
710
-    public function messages_help_tab()
711
-    {
712
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_help_tab.template.php');
713
-    }
714
-
715
-
716
-    public function messengers_help_tab()
717
-    {
718
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messenger_help_tab.template.php');
719
-    }
720
-
721
-
722
-    public function message_types_help_tab()
723
-    {
724
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_message_type_help_tab.template.php');
725
-    }
726
-
727
-
728
-    public function messages_overview_help_tab()
729
-    {
730
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_overview_help_tab.template.php');
731
-    }
732
-
733
-
734
-    public function message_templates_help_tab()
735
-    {
736
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_message_templates_help_tab.template.php');
737
-    }
738
-
739
-
740
-    public function edit_message_template_help_tab()
741
-    {
742
-        $args['img1'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/editor.png' . '" alt="'
743
-                        . esc_attr__('Editor Title', 'event_espresso')
744
-                        . '" />';
745
-        $args['img2'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/switch-context.png' . '" alt="'
746
-                        . esc_attr__('Context Switcher and Preview', 'event_espresso')
747
-                        . '" />';
748
-        $args['img3'] = '<img class="left" src="' . EE_MSG_ASSETS_URL . 'images/form-fields.png' . '" alt="'
749
-                        . esc_attr__('Message Template Form Fields', 'event_espresso')
750
-                        . '" />';
751
-        $args['img4'] = '<img class="right" src="' . EE_MSG_ASSETS_URL . 'images/shortcodes-metabox.png' . '" alt="'
752
-                        . esc_attr__('Shortcodes Metabox', 'event_espresso')
753
-                        . '" />';
754
-        $args['img5'] = '<img class="right" src="' . EE_MSG_ASSETS_URL . 'images/publish-meta-box.png' . '" alt="'
755
-                        . esc_attr__('Publish Metabox', 'event_espresso')
756
-                        . '" />';
757
-        EEH_Template::display_template(
758
-            EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_templates_editor_help_tab.template.php',
759
-            $args
760
-        );
761
-    }
762
-
763
-
764
-    public function message_template_shortcodes_help_tab()
765
-    {
766
-        $this->_set_shortcodes();
767
-        $args['shortcodes'] = $this->_shortcodes;
768
-        EEH_Template::display_template(
769
-            EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_shortcodes_help_tab.template.php',
770
-            $args
771
-        );
772
-    }
773
-
774
-
775
-    public function preview_message_help_tab()
776
-    {
777
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_preview_help_tab.template.php');
778
-    }
779
-
780
-
781
-    public function settings_help_tab()
782
-    {
783
-        $args['img1'] = '<img class="inline-text" src="' . EE_MSG_ASSETS_URL . 'images/email-tab-active.png'
784
-                        . '" alt="' . esc_attr__('Active Email Tab', 'event_espresso') . '" />';
785
-        $args['img2'] = '<img class="inline-text" src="' . EE_MSG_ASSETS_URL . 'images/email-tab-inactive.png'
786
-                        . '" alt="' . esc_attr__('Inactive Email Tab', 'event_espresso') . '" />';
787
-        $args['img3'] = '<div class="switch">'
788
-                        . '<input class="ee-on-off-toggle ee-toggle-round-flat"'
789
-                        . ' type="checkbox" checked="checked">'
790
-                        . '<label for="ee-on-off-toggle-on"></label>'
791
-                        . '</div>';
792
-        $args['img4'] = '<div class="switch">'
793
-                        . '<input class="ee-on-off-toggle ee-toggle-round-flat"'
794
-                        . ' type="checkbox">'
795
-                        . '<label for="ee-on-off-toggle-on"></label>'
796
-                        . '</div>';
797
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_settings_help_tab.template.php', $args);
798
-    }
799
-
800
-
801
-    public function load_scripts_styles()
802
-    {
803
-        wp_register_style('espresso_ee_msg', EE_MSG_ASSETS_URL . 'ee_message_admin.css', EVENT_ESPRESSO_VERSION);
804
-        wp_enqueue_style('espresso_ee_msg');
805
-
806
-        wp_register_script(
807
-            'ee-messages-settings',
808
-            EE_MSG_ASSETS_URL . 'ee-messages-settings.js',
809
-            array('jquery-ui-droppable', 'ee-serialize-full-array'),
810
-            EVENT_ESPRESSO_VERSION,
811
-            true
812
-        );
813
-        wp_register_script(
814
-            'ee-msg-list-table-js',
815
-            EE_MSG_ASSETS_URL . 'ee_message_admin_list_table.js',
816
-            array('ee-dialog'),
817
-            EVENT_ESPRESSO_VERSION
818
-        );
819
-    }
820
-
821
-
822
-    public function load_scripts_styles_default()
823
-    {
824
-        wp_enqueue_script('ee-msg-list-table-js');
825
-    }
826
-
827
-
828
-    public function wp_editor_css($mce_css)
829
-    {
830
-        // if we're on the edit_message_template route
831
-        if ($this->_req_action === 'edit_message_template' && $this->_active_messenger instanceof EE_messenger) {
832
-            $message_type_name = $this->_active_message_type_name;
833
-
834
-            // we're going to REPLACE the existing mce css
835
-            // we need to get the css file location from the active messenger
836
-            $mce_css = $this->_active_messenger->get_variation(
837
-                $this->_template_pack,
838
-                $message_type_name,
839
-                true,
840
-                'wpeditor',
841
-                $this->_variation
842
-            );
843
-        }
844
-
845
-        return $mce_css;
846
-    }
847
-
848
-
849
-    public function load_scripts_styles_edit_message_template()
850
-    {
851
-
852
-        $this->_set_shortcodes();
853
-
854
-        EE_Registry::$i18n_js_strings['confirm_default_reset'] = sprintf(
855
-            esc_html__(
856
-                'Are you sure you want to reset the %s %s message templates?  Remember continuing will reset the templates for all contexts in this messenger and message type group.',
857
-                'event_espresso'
858
-            ),
859
-            $this->_message_template_group->messenger_obj()->label['singular'],
860
-            $this->_message_template_group->message_type_obj()->label['singular']
861
-        );
862
-        EE_Registry::$i18n_js_strings['confirm_switch_template_pack'] = esc_html__(
863
-            'Switching the template pack for a messages template will reset the content for the template so the new layout is loaded.  Any custom content in the existing template will be lost. Are you sure you wish to do this?',
864
-            'event_espresso'
865
-        );
866
-        EE_Registry::$i18n_js_strings['server_error'] = esc_html__(
867
-            'An unknown error occurred on the server while attempting to process your request. Please refresh the page and try again or contact support.',
868
-            'event_espresso'
869
-        );
870
-
871
-        wp_register_script(
872
-            'ee_msgs_edit_js',
873
-            EE_MSG_ASSETS_URL . 'ee_message_editor.js',
874
-            array('jquery'),
875
-            EVENT_ESPRESSO_VERSION
876
-        );
877
-
878
-        wp_enqueue_script('ee_admin_js');
879
-        wp_enqueue_script('ee_msgs_edit_js');
880
-
881
-        // add in special css for tiny_mce
882
-        add_filter('mce_css', array($this, 'wp_editor_css'));
883
-    }
884
-
885
-
886
-    public function load_scripts_styles_display_preview_message()
887
-    {
888
-
889
-        $this->_set_message_template_group();
890
-
891
-        if (isset($this->_req_data['messenger'])) {
892
-            $this->_active_messenger = $this->_message_resource_manager->get_active_messenger(
893
-                $this->_req_data['messenger']
894
-            );
895
-        }
896
-
897
-        $message_type_name = isset($this->_req_data['message_type']) ? $this->_req_data['message_type'] : '';
898
-
899
-
900
-        wp_enqueue_style(
901
-            'espresso_preview_css',
902
-            $this->_active_messenger->get_variation(
903
-                $this->_template_pack,
904
-                $message_type_name,
905
-                true,
906
-                'preview',
907
-                $this->_variation
908
-            )
909
-        );
910
-    }
911
-
912
-
913
-    public function load_scripts_styles_settings()
914
-    {
915
-        wp_register_style(
916
-            'ee-message-settings',
917
-            EE_MSG_ASSETS_URL . 'ee_message_settings.css',
918
-            array(),
919
-            EVENT_ESPRESSO_VERSION
920
-        );
921
-        wp_enqueue_style('ee-text-links');
922
-        wp_enqueue_style('ee-message-settings');
923
-        wp_enqueue_script('ee-messages-settings');
924
-    }
925
-
926
-
927
-    /**
928
-     * set views array for List Table
929
-     */
930
-    public function _set_list_table_views_global_mtps()
931
-    {
932
-        $this->_views = array(
933
-            'in_use' => array(
934
-                'slug'  => 'in_use',
935
-                'label' => esc_html__('In Use', 'event_espresso'),
936
-                'count' => 0,
937
-            ),
938
-        );
939
-    }
940
-
941
-
942
-    /**
943
-     * Set views array for the Custom Template List Table
944
-     */
945
-    public function _set_list_table_views_custom_mtps()
946
-    {
947
-        $this->_set_list_table_views_global_mtps();
948
-        $this->_views['in_use']['bulk_action'] = array(
949
-            'trash_message_template' => esc_html__('Move to Trash', 'event_espresso'),
950
-        );
951
-    }
952
-
953
-
954
-    /**
955
-     * set views array for message queue list table
956
-     *
957
-     * @throws InvalidDataTypeException
958
-     * @throws InvalidInterfaceException
959
-     * @throws InvalidArgumentException
960
-     * @throws EE_Error
961
-     * @throws ReflectionException
962
-     */
963
-    public function _set_list_table_views_default()
964
-    {
965
-        EE_Registry::instance()->load_helper('Template');
966
-
967
-        $common_bulk_actions = EE_Registry::instance()->CAP->current_user_can(
968
-            'ee_send_message',
969
-            'message_list_table_bulk_actions'
970
-        )
971
-            ? array(
972
-                'generate_now'          => esc_html__('Generate Now', 'event_espresso'),
973
-                'generate_and_send_now' => esc_html__('Generate and Send Now', 'event_espresso'),
974
-                'queue_for_resending'   => esc_html__('Queue for Resending', 'event_espresso'),
975
-                'send_now'              => esc_html__('Send Now', 'event_espresso'),
976
-            )
977
-            : array();
978
-
979
-        $delete_bulk_action = EE_Registry::instance()->CAP->current_user_can(
980
-            'ee_delete_messages',
981
-            'message_list_table_bulk_actions'
982
-        )
983
-            ? array('delete_ee_messages' => esc_html__('Delete Messages', 'event_espresso'))
984
-            : array();
985
-
986
-
987
-        $this->_views = array(
988
-            'all' => array(
989
-                'slug'        => 'all',
990
-                'label'       => esc_html__('All', 'event_espresso'),
991
-                'count'       => 0,
992
-                'bulk_action' => array_merge($common_bulk_actions, $delete_bulk_action),
993
-            ),
994
-        );
995
-
996
-
997
-        foreach (EEM_Message::instance()->all_statuses() as $status) {
998
-            if ($status === EEM_Message::status_debug_only && ! EEM_Message::debug()) {
999
-                continue;
1000
-            }
1001
-            $status_bulk_actions = $common_bulk_actions;
1002
-            // unset bulk actions not applying to status
1003
-            if (! empty($status_bulk_actions)) {
1004
-                switch ($status) {
1005
-                    case EEM_Message::status_idle:
1006
-                    case EEM_Message::status_resend:
1007
-                        $status_bulk_actions['send_now'] = $common_bulk_actions['send_now'];
1008
-                        break;
1009
-
1010
-                    case EEM_Message::status_failed:
1011
-                    case EEM_Message::status_debug_only:
1012
-                    case EEM_Message::status_messenger_executing:
1013
-                        $status_bulk_actions = array();
1014
-                        break;
1015
-
1016
-                    case EEM_Message::status_incomplete:
1017
-                        unset($status_bulk_actions['queue_for_resending'], $status_bulk_actions['send_now']);
1018
-                        break;
1019
-
1020
-                    case EEM_Message::status_retry:
1021
-                    case EEM_Message::status_sent:
1022
-                        unset($status_bulk_actions['generate_now'], $status_bulk_actions['generate_and_send_now']);
1023
-                        break;
1024
-                }
1025
-            }
1026
-
1027
-            // skip adding messenger executing status to views because it will be included with the Failed view.
1028
-            if ($status === EEM_Message::status_messenger_executing) {
1029
-                continue;
1030
-            }
1031
-
1032
-            $this->_views[ strtolower($status) ] = array(
1033
-                'slug'        => strtolower($status),
1034
-                'label'       => EEH_Template::pretty_status($status, false, 'sentence'),
1035
-                'count'       => 0,
1036
-                'bulk_action' => array_merge($status_bulk_actions, $delete_bulk_action),
1037
-            );
1038
-        }
1039
-    }
1040
-
1041
-
1042
-    protected function _ee_default_messages_overview_list_table()
1043
-    {
1044
-        $this->_admin_page_title = esc_html__('Default Message Templates', 'event_espresso');
1045
-        $this->display_admin_list_table_page_with_no_sidebar();
1046
-    }
1047
-
1048
-
1049
-    protected function _message_queue_list_table()
1050
-    {
1051
-        $this->_search_btn_label = esc_html__('Message Activity', 'event_espresso');
1052
-        $this->_template_args['per_column'] = 6;
1053
-        $this->_template_args['after_list_table'] = $this->_display_legend($this->_message_legend_items());
1054
-        $this->_template_args['before_list_table'] = '<h3>'
1055
-                                                     . EEM_Message::instance()->get_pretty_label_for_results()
1056
-                                                     . '</h3>';
1057
-        $this->display_admin_list_table_page_with_no_sidebar();
1058
-    }
1059
-
1060
-
1061
-    protected function _message_legend_items()
1062
-    {
1063
-
1064
-        $action_css_classes = EEH_MSG_Template::get_message_action_icons();
1065
-        $action_items = array();
1066
-
1067
-        foreach ($action_css_classes as $action_item => $action_details) {
1068
-            if ($action_item === 'see_notifications_for') {
1069
-                continue;
1070
-            }
1071
-            $action_items[ $action_item ] = array(
1072
-                'class' => $action_details['css_class'],
1073
-                'desc'  => $action_details['label'],
1074
-            );
1075
-        }
1076
-
1077
-        /** @type array $status_items status legend setup */
1078
-        $status_items = array(
1079
-            'sent_status'                => array(
1080
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_sent,
1081
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_sent, false, 'sentence'),
1082
-            ),
1083
-            'idle_status'                => array(
1084
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_idle,
1085
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_idle, false, 'sentence'),
1086
-            ),
1087
-            'failed_status'              => array(
1088
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_failed,
1089
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_failed, false, 'sentence'),
1090
-            ),
1091
-            'messenger_executing_status' => array(
1092
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_messenger_executing,
1093
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_messenger_executing, false, 'sentence'),
1094
-            ),
1095
-            'resend_status'              => array(
1096
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_resend,
1097
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_resend, false, 'sentence'),
1098
-            ),
1099
-            'incomplete_status'          => array(
1100
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_incomplete,
1101
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_incomplete, false, 'sentence'),
1102
-            ),
1103
-            'retry_status'               => array(
1104
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_retry,
1105
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_retry, false, 'sentence'),
1106
-            ),
1107
-        );
1108
-        if (EEM_Message::debug()) {
1109
-            $status_items['debug_only_status'] = array(
1110
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_debug_only,
1111
-                'desc'  => EEH_Template::pretty_status(EEM_Message::status_debug_only, false, 'sentence'),
1112
-            );
1113
-        }
1114
-
1115
-        return array_merge($action_items, $status_items);
1116
-    }
1117
-
1118
-
1119
-    protected function _custom_mtps_preview()
1120
-    {
1121
-        $this->_admin_page_title = esc_html__('Custom Message Templates (Preview)', 'event_espresso');
1122
-        $this->_template_args['preview_img'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/custom_mtps_preview.png"'
1123
-                                               . ' alt="' . esc_attr__(
1124
-                                                   'Preview Custom Message Templates screenshot',
1125
-                                                   'event_espresso'
1126
-                                               ) . '" />';
1127
-        $this->_template_args['preview_text'] = '<strong>'
1128
-                                                . esc_html__(
1129
-                                                    'Custom Message Templates is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. With the Custom Message Templates feature, you are able to create custom message templates and assign them on a per-event basis.',
1130
-                                                    'event_espresso'
1131
-                                                )
1132
-                                                . '</strong>';
1133
-
1134
-        $this->display_admin_caf_preview_page('custom_message_types', false);
1135
-    }
1136
-
1137
-
1138
-    /**
1139
-     * get_message_templates
1140
-     * This gets all the message templates for listing on the overview list.
1141
-     *
1142
-     * @access public
1143
-     * @param int    $perpage the amount of templates groups to show per page
1144
-     * @param string $type    the current _view we're getting templates for
1145
-     * @param bool   $count   return count?
1146
-     * @param bool   $all     disregard any paging info (get all data);
1147
-     * @param bool   $global  whether to return just global (true) or custom templates (false)
1148
-     * @return array
1149
-     * @throws EE_Error
1150
-     * @throws InvalidArgumentException
1151
-     * @throws InvalidDataTypeException
1152
-     * @throws InvalidInterfaceException
1153
-     */
1154
-    public function get_message_templates(
1155
-        $perpage = 10,
1156
-        $type = 'in_use',
1157
-        $count = false,
1158
-        $all = false,
1159
-        $global = true
1160
-    ) {
1161
-
1162
-        $MTP = EEM_Message_Template_Group::instance();
1163
-
1164
-        $this->_req_data['orderby'] = empty($this->_req_data['orderby']) ? 'GRP_ID' : $this->_req_data['orderby'];
1165
-        $orderby = $this->_req_data['orderby'];
1166
-
1167
-        $order = (isset($this->_req_data['order']) && ! empty($this->_req_data['order']))
1168
-            ? $this->_req_data['order']
1169
-            : 'ASC';
1170
-
1171
-        $current_page = isset($this->_req_data['paged']) && ! empty($this->_req_data['paged'])
1172
-            ? $this->_req_data['paged']
1173
-            : 1;
1174
-        $per_page = isset($this->_req_data['perpage']) && ! empty($this->_req_data['perpage'])
1175
-            ? $this->_req_data['perpage']
1176
-            : $perpage;
1177
-
1178
-        $offset = ($current_page - 1) * $per_page;
1179
-        $limit = $all ? null : array($offset, $per_page);
1180
-
1181
-
1182
-        // options will match what is in the _views array property
1183
-        switch ($type) {
1184
-            case 'in_use':
1185
-                $templates = $MTP->get_all_active_message_templates($orderby, $order, $limit, $count, $global, true);
1186
-                break;
1187
-            default:
1188
-                $templates = $MTP->get_all_trashed_grouped_message_templates($orderby, $order, $limit, $count, $global);
1189
-        }
1190
-
1191
-        return $templates;
1192
-    }
1193
-
1194
-
1195
-    /**
1196
-     * filters etc might need a list of installed message_types
1197
-     *
1198
-     * @return array an array of message type objects
1199
-     */
1200
-    public function get_installed_message_types()
1201
-    {
1202
-        $installed_message_types = $this->_message_resource_manager->installed_message_types();
1203
-        $installed = array();
1204
-
1205
-        foreach ($installed_message_types as $message_type) {
1206
-            $installed[ $message_type->name ] = $message_type;
1207
-        }
1208
-
1209
-        return $installed;
1210
-    }
1211
-
1212
-
1213
-    /**
1214
-     * _add_message_template
1215
-     *
1216
-     * This is used when creating a custom template. All Custom Templates start based off another template.
1217
-     *
1218
-     * @param string $message_type
1219
-     * @param string $messenger
1220
-     * @param string $GRP_ID
1221
-     *
1222
-     * @throws EE_error
1223
-     */
1224
-    protected function _add_message_template($message_type = '', $messenger = '', $GRP_ID = '')
1225
-    {
1226
-        // set values override any request data
1227
-        $message_type = ! empty($message_type) ? $message_type : '';
1228
-        $message_type = empty($message_type) && ! empty($this->_req_data['message_type'])
1229
-            ? $this->_req_data['message_type']
1230
-            : $message_type;
1231
-
1232
-        $messenger = ! empty($messenger) ? $messenger : '';
1233
-        $messenger = empty($messenger) && ! empty($this->_req_data['messenger'])
1234
-            ? $this->_req_data['messenger']
1235
-            : $messenger;
1236
-
1237
-        $GRP_ID = ! empty($GRP_ID) ? $GRP_ID : '';
1238
-        $GRP_ID = empty($GRP_ID) && ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : $GRP_ID;
1239
-
1240
-        // we need messenger and message type.  They should be coming from the event editor. If not here then return error
1241
-        if (empty($message_type) || empty($messenger)) {
1242
-            throw new EE_Error(
1243
-                esc_html__(
1244
-                    'Sorry, but we can\'t create new templates because we\'re missing the messenger or message type',
1245
-                    'event_espresso'
1246
-                )
1247
-            );
1248
-        }
1249
-
1250
-        // we need the GRP_ID for the template being used as the base for the new template
1251
-        if (empty($GRP_ID)) {
1252
-            throw new EE_Error(
1253
-                esc_html__(
1254
-                    'In order to create a custom message template the GRP_ID of the template being used as a base is needed',
1255
-                    'event_espresso'
1256
-                )
1257
-            );
1258
-        }
1259
-
1260
-        // let's just make sure the template gets generated!
1261
-
1262
-        // we need to reassign some variables for what the insert is expecting
1263
-        $this->_req_data['MTP_messenger'] = $messenger;
1264
-        $this->_req_data['MTP_message_type'] = $message_type;
1265
-        $this->_req_data['GRP_ID'] = $GRP_ID;
1266
-        $this->_insert_or_update_message_template(true);
1267
-    }
1268
-
1269
-
1270
-    /**
1271
-     * public wrapper for the _add_message_template method
1272
-     *
1273
-     * @param string $message_type     message type slug
1274
-     * @param string $messenger        messenger slug
1275
-     * @param int    $GRP_ID           GRP_ID for the related message template group this new template will be based
1276
-     *                                 off of.
1277
-     * @throws EE_error
1278
-     */
1279
-    public function add_message_template($message_type, $messenger, $GRP_ID)
1280
-    {
1281
-        $this->_add_message_template($message_type, $messenger, $GRP_ID);
1282
-    }
1283
-
1284
-
1285
-    /**
1286
-     * _edit_message_template
1287
-     *
1288
-     * @access protected
1289
-     * @return void
1290
-     * @throws InvalidIdentifierException
1291
-     * @throws DomainException
1292
-     * @throws EE_Error
1293
-     * @throws InvalidArgumentException
1294
-     * @throws ReflectionException
1295
-     * @throws InvalidDataTypeException
1296
-     * @throws InvalidInterfaceException
1297
-     */
1298
-    protected function _edit_message_template()
1299
-    {
1300
-        do_action('AHEE_log', __FILE__, __FUNCTION__, '');
1301
-        $template_fields = '';
1302
-        $sidebar_fields = '';
1303
-        // we filter the tinyMCE settings to remove the validation since message templates by their nature will not have
1304
-        // valid html in the templates.
1305
-        add_filter('tiny_mce_before_init', array($this, 'filter_tinymce_init'), 10, 2);
1306
-
1307
-        $GRP_ID = isset($this->_req_data['id']) && ! empty($this->_req_data['id'])
1308
-            ? absint($this->_req_data['id'])
1309
-            : false;
1310
-
1311
-        $this->_set_shortcodes(); // this also sets the _message_template property.
1312
-        $message_template_group = $this->_message_template_group;
1313
-        $c_label = $message_template_group->context_label();
1314
-        $c_config = $message_template_group->contexts_config();
1315
-
1316
-        reset($c_config);
1317
-        $context = isset($this->_req_data['context']) && ! empty($this->_req_data['context'])
1318
-            ? strtolower($this->_req_data['context'])
1319
-            : key($c_config);
1320
-
1321
-
1322
-        if (empty($GRP_ID)) {
1323
-            $action = 'insert_message_template';
1324
-            $edit_message_template_form_url = add_query_arg(
1325
-                array('action' => $action, 'noheader' => true),
1326
-                EE_MSG_ADMIN_URL
1327
-            );
1328
-        } else {
1329
-            $action = 'update_message_template';
1330
-            $edit_message_template_form_url = add_query_arg(
1331
-                array('action' => $action, 'noheader' => true),
1332
-                EE_MSG_ADMIN_URL
1333
-            );
1334
-        }
1335
-
1336
-        // set active messenger for this view
1337
-        $this->_active_messenger = $this->_message_resource_manager->get_active_messenger(
1338
-            $message_template_group->messenger()
1339
-        );
1340
-        $this->_active_message_type_name = $message_template_group->message_type();
1341
-
1342
-
1343
-        // Do we have any validation errors?
1344
-        $validators = $this->_get_transient();
1345
-        $v_fields = ! empty($validators) ? array_keys($validators) : array();
1346
-
1347
-
1348
-        // we need to assemble the title from Various details
1349
-        $context_label = sprintf(
1350
-            esc_html__('(%s %s)', 'event_espresso'),
1351
-            $c_config[ $context ]['label'],
1352
-            ucwords($c_label['label'])
1353
-        );
1354
-
1355
-        $title = sprintf(
1356
-            esc_html__(' %s %s Template %s', 'event_espresso'),
1357
-            ucwords($message_template_group->messenger_obj()->label['singular']),
1358
-            ucwords($message_template_group->message_type_obj()->label['singular']),
1359
-            $context_label
1360
-        );
1361
-
1362
-        $this->_template_args['GRP_ID'] = $GRP_ID;
1363
-        $this->_template_args['message_template'] = $message_template_group;
1364
-        $this->_template_args['is_extra_fields'] = false;
1365
-
1366
-
1367
-        // let's get EEH_MSG_Template so we can get template form fields
1368
-        $template_field_structure = EEH_MSG_Template::get_fields(
1369
-            $message_template_group->messenger(),
1370
-            $message_template_group->message_type()
1371
-        );
1372
-
1373
-        if (! $template_field_structure) {
1374
-            $template_field_structure = false;
1375
-            $template_fields = esc_html__(
1376
-                'There was an error in assembling the fields for this display (you should see an error message)',
1377
-                'event_espresso'
1378
-            );
1379
-        }
1380
-
1381
-
1382
-        $message_templates = $message_template_group->context_templates();
1383
-
1384
-
1385
-        // if we have the extra key.. then we need to remove the content index from the template_field_structure as it
1386
-        // will get handled in the "extra" array.
1387
-        if (is_array($template_field_structure[ $context ]) && isset($template_field_structure[ $context ]['extra'])) {
1388
-            foreach ($template_field_structure[ $context ]['extra'] as $reference_field => $new_fields) {
1389
-                unset($template_field_structure[ $context ][ $reference_field ]);
1390
-            }
1391
-        }
1392
-
1393
-        // let's loop through the template_field_structure and actually assemble the input fields!
1394
-        if (! empty($template_field_structure)) {
1395
-            foreach ($template_field_structure[ $context ] as $template_field => $field_setup_array) {
1396
-                // if this is an 'extra' template field then we need to remove any existing fields that are keyed up in
1397
-                // the extra array and reset them.
1398
-                if ($template_field === 'extra') {
1399
-                    $this->_template_args['is_extra_fields'] = true;
1400
-                    foreach ($field_setup_array as $reference_field => $new_fields_array) {
1401
-                        $message_template = $message_templates[ $context ][ $reference_field ];
1402
-                        $content = $message_template instanceof EE_Message_Template
1403
-                            ? $message_template->get('MTP_content')
1404
-                            : '';
1405
-                        foreach ($new_fields_array as $extra_field => $extra_array) {
1406
-                            // let's verify if we need this extra field via the shortcodes parameter.
1407
-                            $continue = false;
1408
-                            if (isset($extra_array['shortcodes_required'])) {
1409
-                                foreach ((array) $extra_array['shortcodes_required'] as $shortcode) {
1410
-                                    if (! array_key_exists($shortcode, $this->_shortcodes)) {
1411
-                                        $continue = true;
1412
-                                    }
1413
-                                }
1414
-                                if ($continue) {
1415
-                                    continue;
1416
-                                }
1417
-                            }
1418
-
1419
-                            $field_id = $reference_field
1420
-                                        . '-'
1421
-                                        . $extra_field
1422
-                                        . '-content';
1423
-                            $template_form_fields[ $field_id ] = $extra_array;
1424
-                            $template_form_fields[ $field_id ]['name'] = 'MTP_template_fields['
1425
-                                                                         . $reference_field
1426
-                                                                         . '][content]['
1427
-                                                                         . $extra_field . ']';
1428
-                            $css_class = isset($extra_array['css_class'])
1429
-                                ? $extra_array['css_class']
1430
-                                : '';
1431
-
1432
-                            $template_form_fields[ $field_id ]['css_class'] = ! empty($v_fields)
1433
-                                                                              && in_array($extra_field, $v_fields, true)
1434
-                                                                              &&
1435
-                                                                              (
1436
-                                                                                  is_array($validators[ $extra_field ])
1437
-                                                                                  && isset($validators[ $extra_field ]['msg'])
1438
-                                                                              )
1439
-                                ? 'validate-error ' . $css_class
1440
-                                : $css_class;
1441
-
1442
-                            $template_form_fields[ $field_id ]['value'] = ! empty($message_templates)
1443
-                                                                          && isset($content[ $extra_field ])
1444
-                                ? $content[ $extra_field ]
1445
-                                : '';
1446
-
1447
-                            // do we have a validation error?  if we do then let's use that value instead
1448
-                            $template_form_fields[ $field_id ]['value'] = isset($validators[ $extra_field ])
1449
-                                ? $validators[ $extra_field ]['value']
1450
-                                : $template_form_fields[ $field_id ]['value'];
1451
-
1452
-
1453
-                            $template_form_fields[ $field_id ]['db-col'] = 'MTP_content';
1454
-
1455
-                            // shortcode selector
1456
-                            $field_name_to_use = $extra_field === 'main'
1457
-                                ? 'content'
1458
-                                : $extra_field;
1459
-                            $template_form_fields[ $field_id ]['append_content'] = $this->_get_shortcode_selector(
1460
-                                $field_name_to_use,
1461
-                                $field_id
1462
-                            );
1463
-
1464
-                            if (isset($extra_array['input']) && $extra_array['input'] === 'wp_editor') {
1465
-                                // we want to decode the entities
1466
-                                $template_form_fields[ $field_id ]['value'] = $template_form_fields[ $field_id ]['value'];
1467
-                            }/**/
1468
-                        }
1469
-                        $templatefield_MTP_id = $reference_field . '-MTP_ID';
1470
-                        $templatefield_templatename_id = $reference_field . '-name';
1471
-
1472
-                        $template_form_fields[ $templatefield_MTP_id ] = array(
1473
-                            'name'       => 'MTP_template_fields[' . $reference_field . '][MTP_ID]',
1474
-                            'label'      => null,
1475
-                            'input'      => 'hidden',
1476
-                            'type'       => 'int',
1477
-                            'required'   => false,
1478
-                            'validation' => false,
1479
-                            'value'      => ! empty($message_templates) ? $message_template->ID() : '',
1480
-                            'css_class'  => '',
1481
-                            'format'     => '%d',
1482
-                            'db-col'     => 'MTP_ID',
1483
-                        );
1484
-
1485
-                        $template_form_fields[ $templatefield_templatename_id ] = array(
1486
-                            'name'       => 'MTP_template_fields[' . $reference_field . '][name]',
1487
-                            'label'      => null,
1488
-                            'input'      => 'hidden',
1489
-                            'type'       => 'string',
1490
-                            'required'   => false,
1491
-                            'validation' => true,
1492
-                            'value'      => $reference_field,
1493
-                            'css_class'  => '',
1494
-                            'format'     => '%s',
1495
-                            'db-col'     => 'MTP_template_field',
1496
-                        );
1497
-                    }
1498
-                    continue; // skip the next stuff, we got the necessary fields here for this dataset.
1499
-                } else {
1500
-                    $field_id = $template_field . '-content';
1501
-                    $template_form_fields[ $field_id ] = $field_setup_array;
1502
-                    $template_form_fields[ $field_id ]['name'] = 'MTP_template_fields[' . $template_field . '][content]';
1503
-                    $message_template = isset($message_templates[ $context ][ $template_field ])
1504
-                        ? $message_templates[ $context ][ $template_field ]
1505
-                        : null;
1506
-                    $template_form_fields[ $field_id ]['value'] = ! empty($message_templates)
1507
-                                                                  && is_array($message_templates[ $context ])
1508
-                                                                  && $message_template instanceof EE_Message_Template
1509
-                        ? $message_template->get('MTP_content')
1510
-                        : '';
1511
-
1512
-                    // do we have a validator error for this field?  if we do then we'll use that value instead
1513
-                    $template_form_fields[ $field_id ]['value'] = isset($validators[ $template_field ])
1514
-                        ? $validators[ $template_field ]['value']
1515
-                        : $template_form_fields[ $field_id ]['value'];
1516
-
1517
-
1518
-                    $template_form_fields[ $field_id ]['db-col'] = 'MTP_content';
1519
-                    $css_class = isset($field_setup_array['css_class'])
1520
-                        ? $field_setup_array['css_class']
1521
-                        : '';
1522
-                    $template_form_fields[ $field_id ]['css_class'] = ! empty($v_fields)
1523
-                                                                      && in_array($template_field, $v_fields, true)
1524
-                                                                      && isset($validators[ $template_field ]['msg'])
1525
-                        ? 'validate-error ' . $css_class
1526
-                        : $css_class;
1527
-
1528
-                    // shortcode selector
1529
-                    $template_form_fields[ $field_id ]['append_content'] = $this->_get_shortcode_selector(
1530
-                        $template_field,
1531
-                        $field_id
1532
-                    );
1533
-                }
1534
-
1535
-                // k took care of content field(s) now let's take care of others.
1536
-
1537
-                $templatefield_MTP_id = $template_field . '-MTP_ID';
1538
-                $templatefield_field_templatename_id = $template_field . '-name';
1539
-
1540
-                // foreach template field there are actually two form fields created
1541
-                $template_form_fields[ $templatefield_MTP_id ] = array(
1542
-                    'name'       => 'MTP_template_fields[' . $template_field . '][MTP_ID]',
1543
-                    'label'      => null,
1544
-                    'input'      => 'hidden',
1545
-                    'type'       => 'int',
1546
-                    'required'   => false,
1547
-                    'validation' => true,
1548
-                    'value'      => $message_template instanceof EE_Message_Template ? $message_template->ID() : '',
1549
-                    'css_class'  => '',
1550
-                    'format'     => '%d',
1551
-                    'db-col'     => 'MTP_ID',
1552
-                );
1553
-
1554
-                $template_form_fields[ $templatefield_field_templatename_id ] = array(
1555
-                    'name'       => 'MTP_template_fields[' . $template_field . '][name]',
1556
-                    'label'      => null,
1557
-                    'input'      => 'hidden',
1558
-                    'type'       => 'string',
1559
-                    'required'   => false,
1560
-                    'validation' => true,
1561
-                    'value'      => $template_field,
1562
-                    'css_class'  => '',
1563
-                    'format'     => '%s',
1564
-                    'db-col'     => 'MTP_template_field',
1565
-                );
1566
-            }
1567
-
1568
-            // add other fields
1569
-            $template_form_fields['ee-msg-current-context'] = array(
1570
-                'name'       => 'MTP_context',
1571
-                'label'      => null,
1572
-                'input'      => 'hidden',
1573
-                'type'       => 'string',
1574
-                'required'   => false,
1575
-                'validation' => true,
1576
-                'value'      => $context,
1577
-                'css_class'  => '',
1578
-                'format'     => '%s',
1579
-                'db-col'     => 'MTP_context',
1580
-            );
1581
-
1582
-            $template_form_fields['ee-msg-grp-id'] = array(
1583
-                'name'       => 'GRP_ID',
1584
-                'label'      => null,
1585
-                'input'      => 'hidden',
1586
-                'type'       => 'int',
1587
-                'required'   => false,
1588
-                'validation' => true,
1589
-                'value'      => $GRP_ID,
1590
-                'css_class'  => '',
1591
-                'format'     => '%d',
1592
-                'db-col'     => 'GRP_ID',
1593
-            );
1594
-
1595
-            $template_form_fields['ee-msg-messenger'] = array(
1596
-                'name'       => 'MTP_messenger',
1597
-                'label'      => null,
1598
-                'input'      => 'hidden',
1599
-                'type'       => 'string',
1600
-                'required'   => false,
1601
-                'validation' => true,
1602
-                'value'      => $message_template_group->messenger(),
1603
-                'css_class'  => '',
1604
-                'format'     => '%s',
1605
-                'db-col'     => 'MTP_messenger',
1606
-            );
1607
-
1608
-            $template_form_fields['ee-msg-message-type'] = array(
1609
-                'name'       => 'MTP_message_type',
1610
-                'label'      => null,
1611
-                'input'      => 'hidden',
1612
-                'type'       => 'string',
1613
-                'required'   => false,
1614
-                'validation' => true,
1615
-                'value'      => $message_template_group->message_type(),
1616
-                'css_class'  => '',
1617
-                'format'     => '%s',
1618
-                'db-col'     => 'MTP_message_type',
1619
-            );
1620
-
1621
-            $sidebar_form_fields['ee-msg-is-global'] = array(
1622
-                'name'       => 'MTP_is_global',
1623
-                'label'      => esc_html__('Global Template', 'event_espresso'),
1624
-                'input'      => 'hidden',
1625
-                'type'       => 'int',
1626
-                'required'   => false,
1627
-                'validation' => true,
1628
-                'value'      => $message_template_group->get('MTP_is_global'),
1629
-                'css_class'  => '',
1630
-                'format'     => '%d',
1631
-                'db-col'     => 'MTP_is_global',
1632
-            );
1633
-
1634
-            $sidebar_form_fields['ee-msg-is-override'] = array(
1635
-                'name'       => 'MTP_is_override',
1636
-                'label'      => esc_html__('Override all custom', 'event_espresso'),
1637
-                'input'      => $message_template_group->is_global() ? 'checkbox' : 'hidden',
1638
-                'type'       => 'int',
1639
-                'required'   => false,
1640
-                'validation' => true,
1641
-                'value'      => $message_template_group->get('MTP_is_override'),
1642
-                'css_class'  => '',
1643
-                'format'     => '%d',
1644
-                'db-col'     => 'MTP_is_override',
1645
-            );
1646
-
1647
-            $sidebar_form_fields['ee-msg-is-active'] = array(
1648
-                'name'       => 'MTP_is_active',
1649
-                'label'      => esc_html__('Active Template', 'event_espresso'),
1650
-                'input'      => 'hidden',
1651
-                'type'       => 'int',
1652
-                'required'   => false,
1653
-                'validation' => true,
1654
-                'value'      => $message_template_group->is_active(),
1655
-                'css_class'  => '',
1656
-                'format'     => '%d',
1657
-                'db-col'     => 'MTP_is_active',
1658
-            );
1659
-
1660
-            $sidebar_form_fields['ee-msg-deleted'] = array(
1661
-                'name'       => 'MTP_deleted',
1662
-                'label'      => null,
1663
-                'input'      => 'hidden',
1664
-                'type'       => 'int',
1665
-                'required'   => false,
1666
-                'validation' => true,
1667
-                'value'      => $message_template_group->get('MTP_deleted'),
1668
-                'css_class'  => '',
1669
-                'format'     => '%d',
1670
-                'db-col'     => 'MTP_deleted',
1671
-            );
1672
-            $sidebar_form_fields['ee-msg-author'] = array(
1673
-                'name'       => 'MTP_user_id',
1674
-                'label'      => esc_html__('Author', 'event_espresso'),
1675
-                'input'      => 'hidden',
1676
-                'type'       => 'int',
1677
-                'required'   => false,
1678
-                'validation' => false,
1679
-                'value'      => $message_template_group->user(),
1680
-                'format'     => '%d',
1681
-                'db-col'     => 'MTP_user_id',
1682
-            );
1683
-
1684
-            $sidebar_form_fields['ee-msg-route'] = array(
1685
-                'name'  => 'action',
1686
-                'input' => 'hidden',
1687
-                'type'  => 'string',
1688
-                'value' => $action,
1689
-            );
1690
-
1691
-            $sidebar_form_fields['ee-msg-id'] = array(
1692
-                'name'  => 'id',
1693
-                'input' => 'hidden',
1694
-                'type'  => 'int',
1695
-                'value' => $GRP_ID,
1696
-            );
1697
-            $sidebar_form_fields['ee-msg-evt-nonce'] = array(
1698
-                'name'  => $action . '_nonce',
1699
-                'input' => 'hidden',
1700
-                'type'  => 'string',
1701
-                'value' => wp_create_nonce($action . '_nonce'),
1702
-            );
1703
-
1704
-            if (isset($this->_req_data['template_switch']) && $this->_req_data['template_switch']) {
1705
-                $sidebar_form_fields['ee-msg-template-switch'] = array(
1706
-                    'name'  => 'template_switch',
1707
-                    'input' => 'hidden',
1708
-                    'type'  => 'int',
1709
-                    'value' => 1,
1710
-                );
1711
-            }
1712
-
1713
-
1714
-            $template_fields = $this->_generate_admin_form_fields($template_form_fields);
1715
-            $sidebar_fields = $this->_generate_admin_form_fields($sidebar_form_fields);
1716
-        } //end if ( !empty($template_field_structure) )
1717
-
1718
-        // set extra content for publish box
1719
-        $this->_template_args['publish_box_extra_content'] = $sidebar_fields;
1720
-        $this->_set_publish_post_box_vars(
1721
-            'id',
1722
-            $GRP_ID,
1723
-            false,
1724
-            add_query_arg(
1725
-                array('action' => 'global_mtps'),
1726
-                $this->_admin_base_url
1727
-            )
1728
-        );
1729
-
1730
-        // add preview button
1731
-        $preview_url = parent::add_query_args_and_nonce(
1732
-            array(
1733
-                'message_type' => $message_template_group->message_type(),
1734
-                'messenger'    => $message_template_group->messenger(),
1735
-                'context'      => $context,
1736
-                'GRP_ID'       => $GRP_ID,
1737
-                'action'       => 'preview_message',
1738
-            ),
1739
-            $this->_admin_base_url
1740
-        );
1741
-        $preview_button = '<a href="' . $preview_url . '" class="button-secondary messages-preview-button">'
1742
-                          . esc_html__('Preview', 'event_espresso')
1743
-                          . '</a>';
1744
-
1745
-
1746
-        // setup context switcher
1747
-        $context_switcher_args = array(
1748
-            'page'    => 'espresso_messages',
1749
-            'action'  => 'edit_message_template',
1750
-            'id'      => $GRP_ID,
1751
-            'context' => $context,
1752
-            'extra'   => $preview_button,
1753
-        );
1754
-        $this->_set_context_switcher($message_template_group, $context_switcher_args);
1755
-
1756
-
1757
-        // main box
1758
-        $this->_template_args['template_fields'] = $template_fields;
1759
-        $this->_template_args['sidebar_box_id'] = 'details';
1760
-        $this->_template_args['action'] = $action;
1761
-        $this->_template_args['context'] = $context;
1762
-        $this->_template_args['edit_message_template_form_url'] = $edit_message_template_form_url;
1763
-        $this->_template_args['learn_more_about_message_templates_link'] =
1764
-            $this->_learn_more_about_message_templates_link();
1765
-
1766
-
1767
-        $this->_template_args['before_admin_page_content'] = $this->add_context_switcher();
1768
-        $this->_template_args['before_admin_page_content'] .= $this->add_active_context_element(
1769
-            $message_template_group,
1770
-            $context,
1771
-            $context_label
1772
-        );
1773
-        $this->_template_args['before_admin_page_content'] .= $this->_add_form_element_before();
1774
-        $this->_template_args['after_admin_page_content'] = $this->_add_form_element_after();
1775
-
1776
-        $this->_template_path = $this->_template_args['GRP_ID']
1777
-            ? EE_MSG_TEMPLATE_PATH . 'ee_msg_details_main_edit_meta_box.template.php'
1778
-            : EE_MSG_TEMPLATE_PATH . 'ee_msg_details_main_add_meta_box.template.php';
1779
-
1780
-        // send along EE_Message_Template_Group object for further template use.
1781
-        $this->_template_args['MTP'] = $message_template_group;
1782
-
1783
-        $this->_template_args['admin_page_content'] = EEH_Template::display_template(
1784
-            $this->_template_path,
1785
-            $this->_template_args,
1786
-            true
1787
-        );
1788
-
1789
-
1790
-        // finally, let's set the admin_page title
1791
-        $this->_admin_page_title = sprintf(__('Editing %s', 'event_espresso'), $title);
1792
-
1793
-
1794
-        // we need to take care of setting the shortcodes property for use elsewhere.
1795
-        $this->_set_shortcodes();
1796
-
1797
-
1798
-        // final template wrapper
1799
-        $this->display_admin_page_with_sidebar();
1800
-    }
1801
-
1802
-
1803
-    public function filter_tinymce_init($mceInit, $editor_id)
1804
-    {
1805
-        return $mceInit;
1806
-    }
1807
-
1808
-
1809
-    public function add_context_switcher()
1810
-    {
1811
-        return $this->_context_switcher;
1812
-    }
1813
-
1814
-
1815
-    /**
1816
-     * Adds the activation/deactivation toggle for the message template context.
1817
-     *
1818
-     * @param EE_Message_Template_Group $message_template_group
1819
-     * @param string                    $context
1820
-     * @param string                    $context_label
1821
-     * @return string
1822
-     * @throws DomainException
1823
-     * @throws EE_Error
1824
-     * @throws InvalidIdentifierException
1825
-     */
1826
-    protected function add_active_context_element(
1827
-        EE_Message_Template_Group $message_template_group,
1828
-        $context,
1829
-        $context_label
1830
-    ) {
1831
-        $template_args = array(
1832
-            'context'                   => $context,
1833
-            'nonce'                     => wp_create_nonce('activate_' . $context . '_toggle_nonce'),
1834
-            'is_active'                 => $message_template_group->is_context_active($context),
1835
-            'on_off_action'             => $message_template_group->is_context_active($context)
1836
-                ? 'context-off'
1837
-                : 'context-on',
1838
-            'context_label'             => str_replace(array('(', ')'), '', $context_label),
1839
-            'message_template_group_id' => $message_template_group->ID(),
1840
-        );
1841
-        return EEH_Template::display_template(
1842
-            EE_MSG_TEMPLATE_PATH . 'ee_msg_editor_active_context_element.template.php',
1843
-            $template_args,
1844
-            true
1845
-        );
1846
-    }
1847
-
1848
-
1849
-    /**
1850
-     * Ajax callback for `toggle_context_template` ajax action.
1851
-     * Handles toggling the message context on or off.
1852
-     *
1853
-     * @throws EE_Error
1854
-     * @throws InvalidArgumentException
1855
-     * @throws InvalidDataTypeException
1856
-     * @throws InvalidIdentifierException
1857
-     * @throws InvalidInterfaceException
1858
-     */
1859
-    public function toggle_context_template()
1860
-    {
1861
-        $success = true;
1862
-        // check for required data
1863
-        if (! isset(
1864
-            $this->_req_data['message_template_group_id'],
1865
-            $this->_req_data['context'],
1866
-            $this->_req_data['status']
1867
-        )) {
1868
-            EE_Error::add_error(
1869
-                esc_html__('Required data for doing this action is not available.', 'event_espresso'),
1870
-                __FILE__,
1871
-                __FUNCTION__,
1872
-                __LINE__
1873
-            );
1874
-            $success = false;
1875
-        }
1876
-
1877
-        $nonce = isset($this->_req_data['toggle_context_nonce'])
1878
-            ? sanitize_text_field($this->_req_data['toggle_context_nonce'])
1879
-            : '';
1880
-        $nonce_ref = 'activate_' . $this->_req_data['context'] . '_toggle_nonce';
1881
-        $this->_verify_nonce($nonce, $nonce_ref);
1882
-        $status = $this->_req_data['status'];
1883
-        if ($status !== 'off' && $status !== 'on') {
1884
-            EE_Error::add_error(
1885
-                sprintf(
1886
-                    esc_html__('The given status (%s) is not valid. Must be "off" or "on"', 'event_espresso'),
1887
-                    $this->_req_data['status']
1888
-                ),
1889
-                __FILE__,
1890
-                __FUNCTION__,
1891
-                __LINE__
1892
-            );
1893
-            $success = false;
1894
-        }
1895
-        $message_template_group = EEM_Message_Template_Group::instance()->get_one_by_ID(
1896
-            $this->_req_data['message_template_group_id']
1897
-        );
1898
-        if (! $message_template_group instanceof EE_Message_Template_Group) {
1899
-            EE_Error::add_error(
1900
-                sprintf(
1901
-                    esc_html__(
1902
-                        'Unable to change the active state because the given id "%1$d" does not match a valid "%2$s"',
1903
-                        'event_espresso'
1904
-                    ),
1905
-                    $this->_req_data['message_template_group_id'],
1906
-                    'EE_Message_Template_Group'
1907
-                ),
1908
-                __FILE__,
1909
-                __FUNCTION__,
1910
-                __LINE__
1911
-            );
1912
-            $success = false;
1913
-        }
1914
-        if ($success) {
1915
-            $success = $status === 'off'
1916
-                ? $message_template_group->deactivate_context($this->_req_data['context'])
1917
-                : $message_template_group->activate_context($this->_req_data['context']);
1918
-        }
1919
-        $this->_template_args['success'] = $success;
1920
-        $this->_return_json();
1921
-    }
1922
-
1923
-
1924
-    public function _add_form_element_before()
1925
-    {
1926
-        return '<form method="post" action="'
1927
-               . $this->_template_args["edit_message_template_form_url"]
1928
-               . '" id="ee-msg-edit-frm">';
1929
-    }
1930
-
1931
-    public function _add_form_element_after()
1932
-    {
1933
-        return '</form>';
1934
-    }
1935
-
1936
-
1937
-    /**
1938
-     * This executes switching the template pack for a message template.
1939
-     *
1940
-     * @since 4.5.0
1941
-     * @throws EE_Error
1942
-     * @throws InvalidDataTypeException
1943
-     * @throws InvalidInterfaceException
1944
-     * @throws InvalidArgumentException
1945
-     * @throws ReflectionException
1946
-     */
1947
-    public function switch_template_pack()
1948
-    {
1949
-        $GRP_ID = ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : 0;
1950
-        $template_pack = ! empty($this->_req_data['template_pack']) ? $this->_req_data['template_pack'] : '';
1951
-
1952
-        // verify we have needed values.
1953
-        if (empty($GRP_ID) || empty($template_pack)) {
1954
-            $this->_template_args['error'] = true;
1955
-            EE_Error::add_error(
1956
-                esc_html__('The required date for switching templates is not available.', 'event_espresso'),
1957
-                __FILE__,
1958
-                __FUNCTION__,
1959
-                __LINE__
1960
-            );
1961
-        } else {
1962
-            // get template, set the new template_pack and then reset to default
1963
-            /** @type EE_Message_Template_Group $message_template_group */
1964
-            $message_template_group = EEM_Message_Template_Group::instance()->get_one_by_ID($GRP_ID);
1965
-
1966
-            $message_template_group->set_template_pack_name($template_pack);
1967
-            $this->_req_data['msgr'] = $message_template_group->messenger();
1968
-            $this->_req_data['mt'] = $message_template_group->message_type();
1969
-
1970
-            $query_args = $this->_reset_to_default_template();
1971
-
1972
-            if (empty($query_args['id'])) {
1973
-                EE_Error::add_error(
1974
-                    esc_html__(
1975
-                        'Something went wrong with switching the template pack. Please try again or contact EE support',
1976
-                        'event_espresso'
1977
-                    ),
1978
-                    __FILE__,
1979
-                    __FUNCTION__,
1980
-                    __LINE__
1981
-                );
1982
-                $this->_template_args['error'] = true;
1983
-            } else {
1984
-                $template_label = $message_template_group->get_template_pack()->label;
1985
-                $template_pack_labels = $message_template_group->messenger_obj()->get_supports_labels();
1986
-                EE_Error::add_success(
1987
-                    sprintf(
1988
-                        esc_html__(
1989
-                            'This message template has been successfully switched to use the %1$s %2$s.  Please wait while the page reloads with your new template.',
1990
-                            'event_espresso'
1991
-                        ),
1992
-                        $template_label,
1993
-                        $template_pack_labels->template_pack
1994
-                    )
1995
-                );
1996
-                // generate the redirect url for js.
1997
-                $url = self::add_query_args_and_nonce(
1998
-                    $query_args,
1999
-                    $this->_admin_base_url
2000
-                );
2001
-                $this->_template_args['data']['redirect_url'] = $url;
2002
-                $this->_template_args['success'] = true;
2003
-            }
2004
-
2005
-            $this->_return_json();
2006
-        }
2007
-    }
2008
-
2009
-
2010
-    /**
2011
-     * This handles resetting the template for the given messenger/message_type so that users can start from scratch if
2012
-     * they want.
2013
-     *
2014
-     * @access protected
2015
-     * @return array|null
2016
-     * @throws EE_Error
2017
-     * @throws InvalidArgumentException
2018
-     * @throws InvalidDataTypeException
2019
-     * @throws InvalidInterfaceException
2020
-     */
2021
-    protected function _reset_to_default_template()
2022
-    {
2023
-
2024
-        $templates = array();
2025
-        $GRP_ID = ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : 0;
2026
-        // we need to make sure we've got the info we need.
2027
-        if (! isset($this->_req_data['msgr'], $this->_req_data['mt'], $this->_req_data['GRP_ID'])) {
2028
-            EE_Error::add_error(
2029
-                esc_html__(
2030
-                    'In order to reset the template to its default we require the messenger, message type, and message template GRP_ID to know what is being reset.  At least one of these is missing.',
2031
-                    'event_espresso'
2032
-                ),
2033
-                __FILE__,
2034
-                __FUNCTION__,
2035
-                __LINE__
2036
-            );
2037
-        }
2038
-
2039
-        // all templates will be reset to whatever the defaults are
2040
-        // for the global template matching the messenger and message type.
2041
-        $success = ! empty($GRP_ID) ? true : false;
2042
-
2043
-        if ($success) {
2044
-            // let's first determine if the incoming template is a global template,
2045
-            // if it isn't then we need to get the global template matching messenger and message type.
2046
-            // $MTPG = EEM_Message_Template_Group::instance()->get_one_by_ID( $GRP_ID );
2047
-
2048
-
2049
-            // note this is ONLY deleting the template fields (Message Template rows) NOT the message template group.
2050
-            $success = $this->_delete_mtp_permanently($GRP_ID, false);
2051
-
2052
-            if ($success) {
2053
-                // if successfully deleted, lets generate the new ones.
2054
-                // Note. We set GLOBAL to true, because resets on ANY template
2055
-                // will use the related global template defaults for regeneration.
2056
-                // This means that if a custom template is reset it resets to whatever the related global template is.
2057
-                // HOWEVER, we DO keep the template pack and template variation set
2058
-                // for the current custom template when resetting.
2059
-                $templates = $this->_generate_new_templates(
2060
-                    $this->_req_data['msgr'],
2061
-                    $this->_req_data['mt'],
2062
-                    $GRP_ID,
2063
-                    true
2064
-                );
2065
-            }
2066
-        }
2067
-
2068
-        // any error messages?
2069
-        if (! $success) {
2070
-            EE_Error::add_error(
2071
-                esc_html__(
2072
-                    'Something went wrong with deleting existing templates. Unable to reset to default',
2073
-                    'event_espresso'
2074
-                ),
2075
-                __FILE__,
2076
-                __FUNCTION__,
2077
-                __LINE__
2078
-            );
2079
-        }
2080
-
2081
-        // all good, let's add a success message!
2082
-        if ($success && ! empty($templates)) {
2083
-            // the info for the template we generated is the first element in the returned array
2084
-            // $templates = $templates[0];
2085
-            EE_Error::overwrite_success();
2086
-            EE_Error::add_success(__('Templates have been reset to defaults.', 'event_espresso'));
2087
-        }
2088
-
2089
-
2090
-        $query_args = array(
2091
-            'id'      => isset($templates['GRP_ID']) ? $templates['GRP_ID'] : null,
2092
-            'context' => isset($templates['MTP_context']) ? $templates['MTP_context'] : null,
2093
-            'action'  => isset($templates['GRP_ID']) ? 'edit_message_template' : 'global_mtps',
2094
-        );
2095
-
2096
-        // if called via ajax then we return query args otherwise redirect
2097
-        if (defined('DOING_AJAX') && DOING_AJAX) {
2098
-            return $query_args;
2099
-        } else {
2100
-            $this->_redirect_after_action(false, '', '', $query_args, true);
2101
-
2102
-            return null;
2103
-        }
2104
-    }
2105
-
2106
-
2107
-    /**
2108
-     * Retrieve and set the message preview for display.
2109
-     *
2110
-     * @param bool $send if TRUE then we are doing an actual TEST send with the results of the preview.
2111
-     * @return string
2112
-     * @throws ReflectionException
2113
-     * @throws EE_Error
2114
-     * @throws InvalidArgumentException
2115
-     * @throws InvalidDataTypeException
2116
-     * @throws InvalidInterfaceException
2117
-     */
2118
-    public function _preview_message($send = false)
2119
-    {
2120
-        // first make sure we've got the necessary parameters
2121
-        if (! isset(
2122
-            $this->_req_data['message_type'],
2123
-            $this->_req_data['messenger'],
2124
-            $this->_req_data['messenger'],
2125
-            $this->_req_data['GRP_ID']
2126
-        )) {
2127
-            EE_Error::add_error(
2128
-                esc_html__('Missing necessary parameters for displaying preview', 'event_espresso'),
2129
-                __FILE__,
2130
-                __FUNCTION__,
2131
-                __LINE__
2132
-            );
2133
-        }
2134
-
2135
-        EE_Registry::instance()->REQ->set('GRP_ID', $this->_req_data['GRP_ID']);
2136
-
2137
-
2138
-        // get the preview!
2139
-        $preview = EED_Messages::preview_message(
2140
-            $this->_req_data['message_type'],
2141
-            $this->_req_data['context'],
2142
-            $this->_req_data['messenger'],
2143
-            $send
2144
-        );
2145
-
2146
-        if ($send) {
2147
-            return $preview;
2148
-        }
2149
-
2150
-        // let's add a button to go back to the edit view
2151
-        $query_args = array(
2152
-            'id'      => $this->_req_data['GRP_ID'],
2153
-            'context' => $this->_req_data['context'],
2154
-            'action'  => 'edit_message_template',
2155
-        );
2156
-        $go_back_url = parent::add_query_args_and_nonce($query_args, $this->_admin_base_url);
2157
-        $preview_button = '<a href="'
2158
-                          . $go_back_url
2159
-                          . '" class="button-secondary messages-preview-go-back-button">'
2160
-                          . esc_html__('Go Back to Edit', 'event_espresso')
2161
-                          . '</a>';
2162
-        $message_types = $this->get_installed_message_types();
2163
-        $active_messenger = $this->_message_resource_manager->get_active_messenger(
2164
-            $this->_req_data['messenger']
2165
-        );
2166
-        $active_messenger_label = $active_messenger instanceof EE_messenger
2167
-            ? ucwords($active_messenger->label['singular'])
2168
-            : esc_html__('Unknown Messenger', 'event_espresso');
2169
-        // let's provide a helpful title for context
2170
-        $preview_title = sprintf(
2171
-            esc_html__('Viewing Preview for %s %s Message Template', 'event_espresso'),
2172
-            $active_messenger_label,
2173
-            ucwords($message_types[ $this->_req_data['message_type'] ]->label['singular'])
2174
-        );
2175
-        // setup display of preview.
2176
-        $this->_admin_page_title = $preview_title;
2177
-        $this->_template_args['admin_page_content'] = $preview_button . '<br />' . $preview;
2178
-        $this->_template_args['data']['force_json'] = true;
2179
-
2180
-        return '';
2181
-    }
2182
-
2183
-
2184
-    /**
2185
-     * The initial _preview_message is on a no headers route.  It will optionally call this if necessary otherwise it
2186
-     * gets called automatically.
2187
-     *
2188
-     * @since 4.5.0
2189
-     *
2190
-     * @return string
2191
-     */
2192
-    protected function _display_preview_message()
2193
-    {
2194
-        $this->display_admin_page_with_no_sidebar();
2195
-    }
2196
-
2197
-
2198
-    /**
2199
-     * registers metaboxes that should show up on the "edit_message_template" page
2200
-     *
2201
-     * @access protected
2202
-     * @return void
2203
-     */
2204
-    protected function _register_edit_meta_boxes()
2205
-    {
2206
-        add_meta_box(
2207
-            'mtp_valid_shortcodes',
2208
-            esc_html__('Valid Shortcodes', 'event_espresso'),
2209
-            array($this, 'shortcode_meta_box'),
2210
-            $this->_current_screen->id,
2211
-            'side',
2212
-            'default'
2213
-        );
2214
-        add_meta_box(
2215
-            'mtp_extra_actions',
2216
-            esc_html__('Extra Actions', 'event_espresso'),
2217
-            array($this, 'extra_actions_meta_box'),
2218
-            $this->_current_screen->id,
2219
-            'side',
2220
-            'high'
2221
-        );
2222
-        add_meta_box(
2223
-            'mtp_templates',
2224
-            esc_html__('Template Styles', 'event_espresso'),
2225
-            array($this, 'template_pack_meta_box'),
2226
-            $this->_current_screen->id,
2227
-            'side',
2228
-            'high'
2229
-        );
2230
-    }
2231
-
2232
-
2233
-    /**
2234
-     * metabox content for all template pack and variation selection.
2235
-     *
2236
-     * @since 4.5.0
2237
-     * @return string
2238
-     * @throws DomainException
2239
-     * @throws EE_Error
2240
-     * @throws InvalidArgumentException
2241
-     * @throws ReflectionException
2242
-     * @throws InvalidDataTypeException
2243
-     * @throws InvalidInterfaceException
2244
-     */
2245
-    public function template_pack_meta_box()
2246
-    {
2247
-        $this->_set_message_template_group();
2248
-
2249
-        $tp_collection = EEH_MSG_Template::get_template_pack_collection();
2250
-
2251
-        $tp_select_values = array();
2252
-
2253
-        foreach ($tp_collection as $tp) {
2254
-            // only include template packs that support this messenger and message type!
2255
-            $supports = $tp->get_supports();
2256
-            if (! isset($supports[ $this->_message_template_group->messenger() ])
2257
-                || ! in_array(
2258
-                    $this->_message_template_group->message_type(),
2259
-                    $supports[ $this->_message_template_group->messenger() ],
2260
-                    true
2261
-                )
2262
-            ) {
2263
-                // not supported
2264
-                continue;
2265
-            }
2266
-
2267
-            $tp_select_values[] = array(
2268
-                'text' => $tp->label,
2269
-                'id'   => $tp->dbref,
2270
-            );
2271
-        }
2272
-
2273
-        // if empty $tp_select_values then we make sure default is set because EVERY message type should be supported by
2274
-        // the default template pack.  This still allows for the odd template pack to override.
2275
-        if (empty($tp_select_values)) {
2276
-            $tp_select_values[] = array(
2277
-                'text' => esc_html__('Default', 'event_espresso'),
2278
-                'id'   => 'default',
2279
-            );
2280
-        }
2281
-
2282
-        // setup variation select values for the currently selected template.
2283
-        $variations = $this->_message_template_group->get_template_pack()->get_variations(
2284
-            $this->_message_template_group->messenger(),
2285
-            $this->_message_template_group->message_type()
2286
-        );
2287
-        $variations_select_values = array();
2288
-        foreach ($variations as $variation => $label) {
2289
-            $variations_select_values[] = array(
2290
-                'text' => $label,
2291
-                'id'   => $variation,
2292
-            );
2293
-        }
2294
-
2295
-        $template_pack_labels = $this->_message_template_group->messenger_obj()->get_supports_labels();
2296
-
2297
-        $template_args['template_packs_selector'] = EEH_Form_Fields::select_input(
2298
-            'MTP_template_pack',
2299
-            $tp_select_values,
2300
-            $this->_message_template_group->get_template_pack_name()
2301
-        );
2302
-        $template_args['variations_selector'] = EEH_Form_Fields::select_input(
2303
-            'MTP_template_variation',
2304
-            $variations_select_values,
2305
-            $this->_message_template_group->get_template_pack_variation()
2306
-        );
2307
-        $template_args['template_pack_label'] = $template_pack_labels->template_pack;
2308
-        $template_args['template_variation_label'] = $template_pack_labels->template_variation;
2309
-        $template_args['template_pack_description'] = $template_pack_labels->template_pack_description;
2310
-        $template_args['template_variation_description'] = $template_pack_labels->template_variation_description;
2311
-
2312
-        $template = EE_MSG_TEMPLATE_PATH . 'template_pack_and_variations_metabox.template.php';
2313
-
2314
-        EEH_Template::display_template($template, $template_args);
2315
-    }
2316
-
2317
-
2318
-    /**
2319
-     * This meta box holds any extra actions related to Message Templates
2320
-     * For now, this includes Resetting templates to defaults and sending a test email.
2321
-     *
2322
-     * @access  public
2323
-     * @return void
2324
-     * @throws EE_Error
2325
-     */
2326
-    public function extra_actions_meta_box()
2327
-    {
2328
-        $template_form_fields = array();
2329
-
2330
-        $extra_args = array(
2331
-            'msgr'   => $this->_message_template_group->messenger(),
2332
-            'mt'     => $this->_message_template_group->message_type(),
2333
-            'GRP_ID' => $this->_message_template_group->GRP_ID(),
2334
-        );
2335
-        // first we need to see if there are any fields
2336
-        $fields = $this->_message_template_group->messenger_obj()->get_test_settings_fields();
2337
-
2338
-        if (! empty($fields)) {
2339
-            // yup there be fields
2340
-            foreach ($fields as $field => $config) {
2341
-                $field_id = $this->_message_template_group->messenger() . '_' . $field;
2342
-                $existing = $this->_message_template_group->messenger_obj()->get_existing_test_settings();
2343
-                $default = isset($config['default']) ? $config['default'] : '';
2344
-                $default = isset($config['value']) ? $config['value'] : $default;
2345
-
2346
-                // if type is hidden and the value is empty
2347
-                // something may have gone wrong so let's correct with the defaults
2348
-                $fix = $config['input'] === 'hidden'
2349
-                       && isset($existing[ $field ])
2350
-                       && empty($existing[ $field ])
2351
-                    ? $default
2352
-                    : '';
2353
-                $existing[ $field ] = isset($existing[ $field ]) && empty($fix)
2354
-                    ? $existing[ $field ]
2355
-                    : $fix;
2356
-
2357
-                $template_form_fields[ $field_id ] = array(
2358
-                    'name'       => 'test_settings_fld[' . $field . ']',
2359
-                    'label'      => $config['label'],
2360
-                    'input'      => $config['input'],
2361
-                    'type'       => $config['type'],
2362
-                    'required'   => $config['required'],
2363
-                    'validation' => $config['validation'],
2364
-                    'value'      => isset($existing[ $field ]) ? $existing[ $field ] : $default,
2365
-                    'css_class'  => $config['css_class'],
2366
-                    'options'    => isset($config['options']) ? $config['options'] : array(),
2367
-                    'default'    => $default,
2368
-                    'format'     => $config['format'],
2369
-                );
2370
-            }
2371
-        }
2372
-
2373
-        $test_settings_fields = ! empty($template_form_fields)
2374
-            ? $this->_generate_admin_form_fields($template_form_fields, 'string', 'ee_tst_settings_flds')
2375
-            : '';
2376
-
2377
-        $test_settings_html = '';
2378
-        // print out $test_settings_fields
2379
-        if (! empty($test_settings_fields)) {
2380
-            echo $test_settings_fields;
2381
-            $test_settings_html = '<input type="submit" class="button-primary mtp-test-button alignright" ';
2382
-            $test_settings_html .= 'name="test_button" value="';
2383
-            $test_settings_html .= esc_html__('Test Send', 'event_espresso');
2384
-            $test_settings_html .= '" /><div style="clear:both"></div>';
2385
-        }
2386
-
2387
-        // and button
2388
-        $test_settings_html .= '<p>'
2389
-                               . esc_html__('Need to reset this message type and start over?', 'event_espresso')
2390
-                               . '</p>';
2391
-        $test_settings_html .= '<div class="publishing-action alignright resetbutton">';
2392
-        $test_settings_html .= $this->get_action_link_or_button(
2393
-            'reset_to_default',
2394
-            'reset',
2395
-            $extra_args,
2396
-            'button-primary reset-default-button'
2397
-        );
2398
-        $test_settings_html .= '</div><div style="clear:both"></div>';
2399
-        echo $test_settings_html;
2400
-    }
2401
-
2402
-
2403
-    /**
2404
-     * This returns the shortcode selector skeleton for a given context and field.
2405
-     *
2406
-     * @since 4.9.rc.000
2407
-     * @param string $field           The name of the field retrieving shortcodes for.
2408
-     * @param string $linked_input_id The css id of the input that the shortcodes get added to.
2409
-     * @return string
2410
-     * @throws DomainException
2411
-     * @throws EE_Error
2412
-     * @throws InvalidArgumentException
2413
-     * @throws ReflectionException
2414
-     * @throws InvalidDataTypeException
2415
-     * @throws InvalidInterfaceException
2416
-     */
2417
-    protected function _get_shortcode_selector($field, $linked_input_id)
2418
-    {
2419
-        $template_args = array(
2420
-            'shortcodes'      => $this->_get_shortcodes(array($field), true),
2421
-            'fieldname'       => $field,
2422
-            'linked_input_id' => $linked_input_id,
2423
-        );
2424
-
2425
-        return EEH_Template::display_template(
2426
-            EE_MSG_TEMPLATE_PATH . 'shortcode_selector_skeleton.template.php',
2427
-            $template_args,
2428
-            true
2429
-        );
2430
-    }
2431
-
2432
-
2433
-    /**
2434
-     * This just takes care of returning the meta box content for shortcodes (only used on the edit message template
2435
-     * page)
2436
-     *
2437
-     * @access public
2438
-     * @return void
2439
-     * @throws EE_Error
2440
-     * @throws InvalidArgumentException
2441
-     * @throws ReflectionException
2442
-     * @throws InvalidDataTypeException
2443
-     * @throws InvalidInterfaceException
2444
-     */
2445
-    public function shortcode_meta_box()
2446
-    {
2447
-        $shortcodes = $this->_get_shortcodes(array(), false); // just make sure shortcodes property is set
2448
-        // $messenger = $this->_message_template_group->messenger_obj();
2449
-        // now let's set the content depending on the status of the shortcodes array
2450
-        if (empty($shortcodes)) {
2451
-            $content = '<p>' . esc_html__('There are no valid shortcodes available', 'event_espresso') . '</p>';
2452
-            echo $content;
2453
-        } else {
2454
-            // $alt = 0;
2455
-            ?>
22
+	/**
23
+	 * @type EE_Message_Resource_Manager $_message_resource_manager
24
+	 */
25
+	protected $_message_resource_manager;
26
+
27
+	/**
28
+	 * @type string $_active_message_type_name
29
+	 */
30
+	protected $_active_message_type_name = '';
31
+
32
+	/**
33
+	 * @type EE_messenger $_active_messenger
34
+	 */
35
+	protected $_active_messenger;
36
+	protected $_activate_state;
37
+	protected $_activate_meta_box_type;
38
+	protected $_current_message_meta_box;
39
+	protected $_current_message_meta_box_object;
40
+	protected $_context_switcher;
41
+	protected $_shortcodes = array();
42
+	protected $_active_messengers = array();
43
+	protected $_active_message_types = array();
44
+
45
+	/**
46
+	 * @var EE_Message_Template_Group $_message_template_group
47
+	 */
48
+	protected $_message_template_group;
49
+	protected $_m_mt_settings = array();
50
+
51
+
52
+	/**
53
+	 * This is set via the _set_message_template_group method and holds whatever the template pack for the group is.
54
+	 * IF there is no group then it gets automatically set to the Default template pack.
55
+	 *
56
+	 * @since 4.5.0
57
+	 *
58
+	 * @var EE_Messages_Template_Pack
59
+	 */
60
+	protected $_template_pack;
61
+
62
+
63
+	/**
64
+	 * This is set via the _set_message_template_group method and holds whatever the template pack variation for the
65
+	 * group is.  If there is no group then it automatically gets set to default.
66
+	 *
67
+	 * @since 4.5.0
68
+	 *
69
+	 * @var string
70
+	 */
71
+	protected $_variation;
72
+
73
+
74
+	/**
75
+	 * @param bool $routing
76
+	 * @throws EE_Error
77
+	 */
78
+	public function __construct($routing = true)
79
+	{
80
+		// make sure messages autoloader is running
81
+		EED_Messages::set_autoloaders();
82
+		parent::__construct($routing);
83
+	}
84
+
85
+
86
+	protected function _init_page_props()
87
+	{
88
+		$this->page_slug = EE_MSG_PG_SLUG;
89
+		$this->page_label = esc_html__('Messages Settings', 'event_espresso');
90
+		$this->_admin_base_url = EE_MSG_ADMIN_URL;
91
+		$this->_admin_base_path = EE_MSG_ADMIN;
92
+
93
+		$this->_activate_state = isset($this->_req_data['activate_state']) ? (array) $this->_req_data['activate_state']
94
+			: array();
95
+
96
+		$this->_active_messenger = isset($this->_req_data['messenger']) ? $this->_req_data['messenger'] : null;
97
+		$this->_load_message_resource_manager();
98
+	}
99
+
100
+
101
+	/**
102
+	 * loads messenger objects into the $_active_messengers property (so we can access the needed methods)
103
+	 *
104
+	 * @throws EE_Error
105
+	 * @throws InvalidDataTypeException
106
+	 * @throws InvalidInterfaceException
107
+	 * @throws InvalidArgumentException
108
+	 * @throws ReflectionException
109
+	 */
110
+	protected function _load_message_resource_manager()
111
+	{
112
+		$this->_message_resource_manager = EE_Registry::instance()->load_lib('Message_Resource_Manager');
113
+	}
114
+
115
+
116
+	/**
117
+	 * @deprecated 4.9.9.rc.014
118
+	 * @return array
119
+	 * @throws EE_Error
120
+	 * @throws InvalidArgumentException
121
+	 * @throws InvalidDataTypeException
122
+	 * @throws InvalidInterfaceException
123
+	 */
124
+	public function get_messengers_for_list_table()
125
+	{
126
+		EE_Error::doing_it_wrong(
127
+			__METHOD__,
128
+			sprintf(
129
+				esc_html__(
130
+					'This method is no longer in use.  There is no replacement for it. The method was used to generate a set of values for use in creating a messenger filter dropdown which is now generated differently via %s',
131
+					'event_espresso'
132
+				),
133
+				'Messages_Admin_Page::get_messengers_select_input()'
134
+			),
135
+			'4.9.9.rc.014'
136
+		);
137
+
138
+		$m_values = array();
139
+		$active_messengers = EEM_Message::instance()->get_all(array('group_by' => 'MSG_messenger'));
140
+		// setup messengers for selects
141
+		$i = 1;
142
+		foreach ($active_messengers as $active_messenger) {
143
+			if ($active_messenger instanceof EE_Message) {
144
+				$m_values[ $i ]['id'] = $active_messenger->messenger();
145
+				$m_values[ $i ]['text'] = ucwords($active_messenger->messenger_label());
146
+				$i++;
147
+			}
148
+		}
149
+
150
+		return $m_values;
151
+	}
152
+
153
+
154
+	/**
155
+	 * @deprecated 4.9.9.rc.014
156
+	 * @return array
157
+	 * @throws EE_Error
158
+	 * @throws InvalidArgumentException
159
+	 * @throws InvalidDataTypeException
160
+	 * @throws InvalidInterfaceException
161
+	 */
162
+	public function get_message_types_for_list_table()
163
+	{
164
+		EE_Error::doing_it_wrong(
165
+			__METHOD__,
166
+			sprintf(
167
+				esc_html__(
168
+					'This method is no longer in use.  There is no replacement for it. The method was used to generate a set of values for use in creating a message type filter dropdown which is now generated differently via %s',
169
+					'event_espresso'
170
+				),
171
+				'Messages_Admin_Page::get_message_types_select_input()'
172
+			),
173
+			'4.9.9.rc.014'
174
+		);
175
+
176
+		$mt_values = array();
177
+		$active_messages = EEM_Message::instance()->get_all(array('group_by' => 'MSG_message_type'));
178
+		$i = 1;
179
+		foreach ($active_messages as $active_message) {
180
+			if ($active_message instanceof EE_Message) {
181
+				$mt_values[ $i ]['id'] = $active_message->message_type();
182
+				$mt_values[ $i ]['text'] = ucwords($active_message->message_type_label());
183
+				$i++;
184
+			}
185
+		}
186
+
187
+		return $mt_values;
188
+	}
189
+
190
+
191
+	/**
192
+	 * @deprecated 4.9.9.rc.014
193
+	 * @return array
194
+	 * @throws EE_Error
195
+	 * @throws InvalidArgumentException
196
+	 * @throws InvalidDataTypeException
197
+	 * @throws InvalidInterfaceException
198
+	 */
199
+	public function get_contexts_for_message_types_for_list_table()
200
+	{
201
+		EE_Error::doing_it_wrong(
202
+			__METHOD__,
203
+			sprintf(
204
+				esc_html__(
205
+					'This method is no longer in use.  There is no replacement for it. The method was used to generate a set of values for use in creating a message type context filter dropdown which is now generated differently via %s',
206
+					'event_espresso'
207
+				),
208
+				'Messages_Admin_Page::get_contexts_for_message_types_select_input()'
209
+			),
210
+			'4.9.9.rc.014'
211
+		);
212
+
213
+		$contexts = array();
214
+		$active_message_contexts = EEM_Message::instance()->get_all(array('group_by' => 'MSG_context'));
215
+		foreach ($active_message_contexts as $active_message) {
216
+			if ($active_message instanceof EE_Message) {
217
+				$message_type = $active_message->message_type_object();
218
+				if ($message_type instanceof EE_message_type) {
219
+					$message_type_contexts = $message_type->get_contexts();
220
+					foreach ($message_type_contexts as $context => $context_details) {
221
+						$contexts[ $context ] = $context_details['label'];
222
+					}
223
+				}
224
+			}
225
+		}
226
+
227
+		return $contexts;
228
+	}
229
+
230
+
231
+	/**
232
+	 * Generate select input with provided messenger options array.
233
+	 *
234
+	 * @param array $messenger_options Array of messengers indexed by messenger slug and values are the messenger
235
+	 *                                 labels.
236
+	 * @return string
237
+	 * @throws EE_Error
238
+	 */
239
+	public function get_messengers_select_input($messenger_options)
240
+	{
241
+		// if empty or just one value then just return an empty string
242
+		if (empty($messenger_options)
243
+			|| ! is_array($messenger_options)
244
+			|| count($messenger_options) === 1
245
+		) {
246
+			return '';
247
+		}
248
+		// merge in default
249
+		$messenger_options = array_merge(
250
+			array('none_selected' => esc_html__('Show All Messengers', 'event_espresso')),
251
+			$messenger_options
252
+		);
253
+		$input = new EE_Select_Input(
254
+			$messenger_options,
255
+			array(
256
+				'html_name'  => 'ee_messenger_filter_by',
257
+				'html_id'    => 'ee_messenger_filter_by',
258
+				'html_class' => 'wide',
259
+				'default'    => isset($this->_req_data['ee_messenger_filter_by'])
260
+					? sanitize_title($this->_req_data['ee_messenger_filter_by'])
261
+					: 'none_selected',
262
+			)
263
+		);
264
+
265
+		return $input->get_html_for_input();
266
+	}
267
+
268
+
269
+	/**
270
+	 * Generate select input with provided message type options array.
271
+	 *
272
+	 * @param array $message_type_options Array of message types indexed by message type slug, and values are the
273
+	 *                                    message type labels
274
+	 * @return string
275
+	 * @throws EE_Error
276
+	 */
277
+	public function get_message_types_select_input($message_type_options)
278
+	{
279
+		// if empty or count of options is 1 then just return an empty string
280
+		if (empty($message_type_options)
281
+			|| ! is_array($message_type_options)
282
+			|| count($message_type_options) === 1
283
+		) {
284
+			return '';
285
+		}
286
+		// merge in default
287
+		$message_type_options = array_merge(
288
+			array('none_selected' => esc_html__('Show All Message Types', 'event_espresso')),
289
+			$message_type_options
290
+		);
291
+		$input = new EE_Select_Input(
292
+			$message_type_options,
293
+			array(
294
+				'html_name'  => 'ee_message_type_filter_by',
295
+				'html_id'    => 'ee_message_type_filter_by',
296
+				'html_class' => 'wide',
297
+				'default'    => isset($this->_req_data['ee_message_type_filter_by'])
298
+					? sanitize_title($this->_req_data['ee_message_type_filter_by'])
299
+					: 'none_selected',
300
+			)
301
+		);
302
+
303
+		return $input->get_html_for_input();
304
+	}
305
+
306
+
307
+	/**
308
+	 * Generate select input with provide message type contexts array.
309
+	 *
310
+	 * @param array $context_options Array of message type contexts indexed by context slug, and values are the
311
+	 *                               context label.
312
+	 * @return string
313
+	 * @throws EE_Error
314
+	 */
315
+	public function get_contexts_for_message_types_select_input($context_options)
316
+	{
317
+		// if empty or count of options is one then just return empty string
318
+		if (empty($context_options)
319
+			|| ! is_array($context_options)
320
+			|| count($context_options) === 1
321
+		) {
322
+			return '';
323
+		}
324
+		// merge in default
325
+		$context_options = array_merge(
326
+			array('none_selected' => esc_html__('Show all Contexts', 'event_espresso')),
327
+			$context_options
328
+		);
329
+		$input = new EE_Select_Input(
330
+			$context_options,
331
+			array(
332
+				'html_name'  => 'ee_context_filter_by',
333
+				'html_id'    => 'ee_context_filter_by',
334
+				'html_class' => 'wide',
335
+				'default'    => isset($this->_req_data['ee_context_filter_by'])
336
+					? sanitize_title($this->_req_data['ee_context_filter_by'])
337
+					: 'none_selected',
338
+			)
339
+		);
340
+
341
+		return $input->get_html_for_input();
342
+	}
343
+
344
+
345
+	protected function _ajax_hooks()
346
+	{
347
+		add_action('wp_ajax_activate_messenger', array($this, 'activate_messenger_toggle'));
348
+		add_action('wp_ajax_activate_mt', array($this, 'activate_mt_toggle'));
349
+		add_action('wp_ajax_ee_msgs_save_settings', array($this, 'save_settings'));
350
+		add_action('wp_ajax_ee_msgs_update_mt_form', array($this, 'update_mt_form'));
351
+		add_action('wp_ajax_switch_template_pack', array($this, 'switch_template_pack'));
352
+		add_action('wp_ajax_toggle_context_template', array($this, 'toggle_context_template'));
353
+	}
354
+
355
+
356
+	protected function _define_page_props()
357
+	{
358
+		$this->_admin_page_title = $this->page_label;
359
+		$this->_labels = array(
360
+			'buttons'    => array(
361
+				'add'    => esc_html__('Add New Message Template', 'event_espresso'),
362
+				'edit'   => esc_html__('Edit Message Template', 'event_espresso'),
363
+				'delete' => esc_html__('Delete Message Template', 'event_espresso'),
364
+			),
365
+			'publishbox' => esc_html__('Update Actions', 'event_espresso'),
366
+		);
367
+	}
368
+
369
+
370
+	/**
371
+	 *        an array for storing key => value pairs of request actions and their corresponding methods
372
+	 *
373
+	 * @access protected
374
+	 * @return void
375
+	 */
376
+	protected function _set_page_routes()
377
+	{
378
+		$grp_id = ! empty($this->_req_data['GRP_ID']) && ! is_array($this->_req_data['GRP_ID'])
379
+			? $this->_req_data['GRP_ID']
380
+			: 0;
381
+		$grp_id = empty($grp_id) && ! empty($this->_req_data['id'])
382
+			? $this->_req_data['id']
383
+			: $grp_id;
384
+		$msg_id = ! empty($this->_req_data['MSG_ID']) && ! is_array($this->_req_data['MSG_ID'])
385
+			? $this->_req_data['MSG_ID']
386
+			: 0;
387
+
388
+		$this->_page_routes = array(
389
+			'default'                          => array(
390
+				'func'       => '_message_queue_list_table',
391
+				'capability' => 'ee_read_global_messages',
392
+			),
393
+			'global_mtps'                      => array(
394
+				'func'       => '_ee_default_messages_overview_list_table',
395
+				'capability' => 'ee_read_global_messages',
396
+			),
397
+			'custom_mtps'                      => array(
398
+				'func'       => '_custom_mtps_preview',
399
+				'capability' => 'ee_read_messages',
400
+			),
401
+			'add_new_message_template'         => array(
402
+				'func'       => '_add_message_template',
403
+				'capability' => 'ee_edit_messages',
404
+				'noheader'   => true,
405
+			),
406
+			'edit_message_template'            => array(
407
+				'func'       => '_edit_message_template',
408
+				'capability' => 'ee_edit_message',
409
+				'obj_id'     => $grp_id,
410
+			),
411
+			'preview_message'                  => array(
412
+				'func'               => '_preview_message',
413
+				'capability'         => 'ee_read_message',
414
+				'obj_id'             => $grp_id,
415
+				'noheader'           => true,
416
+				'headers_sent_route' => 'display_preview_message',
417
+			),
418
+			'display_preview_message'          => array(
419
+				'func'       => '_display_preview_message',
420
+				'capability' => 'ee_read_message',
421
+				'obj_id'     => $grp_id,
422
+			),
423
+			'insert_message_template'          => array(
424
+				'func'       => '_insert_or_update_message_template',
425
+				'capability' => 'ee_edit_messages',
426
+				'args'       => array('new_template' => true),
427
+				'noheader'   => true,
428
+			),
429
+			'update_message_template'          => array(
430
+				'func'       => '_insert_or_update_message_template',
431
+				'capability' => 'ee_edit_message',
432
+				'obj_id'     => $grp_id,
433
+				'args'       => array('new_template' => false),
434
+				'noheader'   => true,
435
+			),
436
+			'trash_message_template'           => array(
437
+				'func'       => '_trash_or_restore_message_template',
438
+				'capability' => 'ee_delete_message',
439
+				'obj_id'     => $grp_id,
440
+				'args'       => array('trash' => true, 'all' => true),
441
+				'noheader'   => true,
442
+			),
443
+			'trash_message_template_context'   => array(
444
+				'func'       => '_trash_or_restore_message_template',
445
+				'capability' => 'ee_delete_message',
446
+				'obj_id'     => $grp_id,
447
+				'args'       => array('trash' => true),
448
+				'noheader'   => true,
449
+			),
450
+			'restore_message_template'         => array(
451
+				'func'       => '_trash_or_restore_message_template',
452
+				'capability' => 'ee_delete_message',
453
+				'obj_id'     => $grp_id,
454
+				'args'       => array('trash' => false, 'all' => true),
455
+				'noheader'   => true,
456
+			),
457
+			'restore_message_template_context' => array(
458
+				'func'       => '_trash_or_restore_message_template',
459
+				'capability' => 'ee_delete_message',
460
+				'obj_id'     => $grp_id,
461
+				'args'       => array('trash' => false),
462
+				'noheader'   => true,
463
+			),
464
+			'delete_message_template'          => array(
465
+				'func'       => '_delete_message_template',
466
+				'capability' => 'ee_delete_message',
467
+				'obj_id'     => $grp_id,
468
+				'noheader'   => true,
469
+			),
470
+			'reset_to_default'                 => array(
471
+				'func'       => '_reset_to_default_template',
472
+				'capability' => 'ee_edit_message',
473
+				'obj_id'     => $grp_id,
474
+				'noheader'   => true,
475
+			),
476
+			'settings'                         => array(
477
+				'func'       => '_settings',
478
+				'capability' => 'manage_options',
479
+			),
480
+			'update_global_settings'           => array(
481
+				'func'       => '_update_global_settings',
482
+				'capability' => 'manage_options',
483
+				'noheader'   => true,
484
+			),
485
+			'generate_now'                     => array(
486
+				'func'       => '_generate_now',
487
+				'capability' => 'ee_send_message',
488
+				'noheader'   => true,
489
+			),
490
+			'generate_and_send_now'            => array(
491
+				'func'       => '_generate_and_send_now',
492
+				'capability' => 'ee_send_message',
493
+				'noheader'   => true,
494
+			),
495
+			'queue_for_resending'              => array(
496
+				'func'       => '_queue_for_resending',
497
+				'capability' => 'ee_send_message',
498
+				'noheader'   => true,
499
+			),
500
+			'send_now'                         => array(
501
+				'func'       => '_send_now',
502
+				'capability' => 'ee_send_message',
503
+				'noheader'   => true,
504
+			),
505
+			'delete_ee_message'                => array(
506
+				'func'       => '_delete_ee_messages',
507
+				'capability' => 'ee_delete_messages',
508
+				'noheader'   => true,
509
+			),
510
+			'delete_ee_messages'               => array(
511
+				'func'       => '_delete_ee_messages',
512
+				'capability' => 'ee_delete_messages',
513
+				'noheader'   => true,
514
+				'obj_id'     => $msg_id,
515
+			),
516
+		);
517
+	}
518
+
519
+
520
+	protected function _set_page_config()
521
+	{
522
+		$this->_page_config = array(
523
+			'default'                  => array(
524
+				'nav'           => array(
525
+					'label' => esc_html__('Message Activity', 'event_espresso'),
526
+					'order' => 10,
527
+				),
528
+				'list_table'    => 'EE_Message_List_Table',
529
+				// 'qtips' => array( 'EE_Message_List_Table_Tips' ),
530
+				'require_nonce' => false,
531
+			),
532
+			'global_mtps'              => array(
533
+				'nav'           => array(
534
+					'label' => esc_html__('Default Message Templates', 'event_espresso'),
535
+					'order' => 20,
536
+				),
537
+				'list_table'    => 'Messages_Template_List_Table',
538
+				'help_tabs'     => array(
539
+					'messages_overview_help_tab'                                => array(
540
+						'title'    => esc_html__('Messages Overview', 'event_espresso'),
541
+						'filename' => 'messages_overview',
542
+					),
543
+					'messages_overview_messages_table_column_headings_help_tab' => array(
544
+						'title'    => esc_html__('Messages Table Column Headings', 'event_espresso'),
545
+						'filename' => 'messages_overview_table_column_headings',
546
+					),
547
+					'messages_overview_messages_filters_help_tab'               => array(
548
+						'title'    => esc_html__('Message Filters', 'event_espresso'),
549
+						'filename' => 'messages_overview_filters',
550
+					),
551
+					'messages_overview_messages_views_help_tab'                 => array(
552
+						'title'    => esc_html__('Message Views', 'event_espresso'),
553
+						'filename' => 'messages_overview_views',
554
+					),
555
+					'message_overview_message_types_help_tab'                   => array(
556
+						'title'    => esc_html__('Message Types', 'event_espresso'),
557
+						'filename' => 'messages_overview_types',
558
+					),
559
+					'messages_overview_messengers_help_tab'                     => array(
560
+						'title'    => esc_html__('Messengers', 'event_espresso'),
561
+						'filename' => 'messages_overview_messengers',
562
+					),
563
+				),
564
+				'help_tour'     => array('Messages_Overview_Help_Tour'),
565
+				'require_nonce' => false,
566
+			),
567
+			'custom_mtps'              => array(
568
+				'nav'           => array(
569
+					'label' => esc_html__('Custom Message Templates', 'event_espresso'),
570
+					'order' => 30,
571
+				),
572
+				'help_tabs'     => array(),
573
+				'help_tour'     => array(),
574
+				'require_nonce' => false,
575
+			),
576
+			'add_new_message_template' => array(
577
+				'nav'           => array(
578
+					'label'      => esc_html__('Add New Message Templates', 'event_espresso'),
579
+					'order'      => 5,
580
+					'persistent' => false,
581
+				),
582
+				'require_nonce' => false,
583
+			),
584
+			'edit_message_template'    => array(
585
+				'labels'        => array(
586
+					'buttons'    => array(
587
+						'reset' => esc_html__('Reset Templates', 'event_espresso'),
588
+					),
589
+					'publishbox' => esc_html__('Update Actions', 'event_espresso'),
590
+				),
591
+				'nav'           => array(
592
+					'label'      => esc_html__('Edit Message Templates', 'event_espresso'),
593
+					'order'      => 5,
594
+					'persistent' => false,
595
+					'url'        => '',
596
+				),
597
+				'metaboxes'     => array('_publish_post_box', '_register_edit_meta_boxes'),
598
+				'has_metaboxes' => true,
599
+				'help_tour'     => array('Message_Templates_Edit_Help_Tour'),
600
+				'help_tabs'     => array(
601
+					'edit_message_template'            => array(
602
+						'title'    => esc_html__('Message Template Editor', 'event_espresso'),
603
+						'callback' => 'edit_message_template_help_tab',
604
+					),
605
+					'message_templates_help_tab'       => array(
606
+						'title'    => esc_html__('Message Templates', 'event_espresso'),
607
+						'filename' => 'messages_templates',
608
+					),
609
+					'message_template_shortcodes'      => array(
610
+						'title'    => esc_html__('Message Shortcodes', 'event_espresso'),
611
+						'callback' => 'message_template_shortcodes_help_tab',
612
+					),
613
+					'message_preview_help_tab'         => array(
614
+						'title'    => esc_html__('Message Preview', 'event_espresso'),
615
+						'filename' => 'messages_preview',
616
+					),
617
+					'messages_overview_other_help_tab' => array(
618
+						'title'    => esc_html__('Messages Other', 'event_espresso'),
619
+						'filename' => 'messages_overview_other',
620
+					),
621
+				),
622
+				'require_nonce' => false,
623
+			),
624
+			'display_preview_message'  => array(
625
+				'nav'           => array(
626
+					'label'      => esc_html__('Message Preview', 'event_espresso'),
627
+					'order'      => 5,
628
+					'url'        => '',
629
+					'persistent' => false,
630
+				),
631
+				'help_tabs'     => array(
632
+					'preview_message' => array(
633
+						'title'    => esc_html__('About Previews', 'event_espresso'),
634
+						'callback' => 'preview_message_help_tab',
635
+					),
636
+				),
637
+				'require_nonce' => false,
638
+			),
639
+			'settings'                 => array(
640
+				'nav'           => array(
641
+					'label' => esc_html__('Settings', 'event_espresso'),
642
+					'order' => 40,
643
+				),
644
+				'metaboxes'     => array('_messages_settings_metaboxes'),
645
+				'help_tabs'     => array(
646
+					'messages_settings_help_tab'               => array(
647
+						'title'    => esc_html__('Messages Settings', 'event_espresso'),
648
+						'filename' => 'messages_settings',
649
+					),
650
+					'messages_settings_message_types_help_tab' => array(
651
+						'title'    => esc_html__('Activating / Deactivating Message Types', 'event_espresso'),
652
+						'filename' => 'messages_settings_message_types',
653
+					),
654
+					'messages_settings_messengers_help_tab'    => array(
655
+						'title'    => esc_html__('Activating / Deactivating Messengers', 'event_espresso'),
656
+						'filename' => 'messages_settings_messengers',
657
+					),
658
+				),
659
+				'help_tour'     => array('Messages_Settings_Help_Tour'),
660
+				'require_nonce' => false,
661
+			),
662
+		);
663
+	}
664
+
665
+
666
+	protected function _add_screen_options()
667
+	{
668
+		// todo
669
+	}
670
+
671
+
672
+	protected function _add_screen_options_global_mtps()
673
+	{
674
+		/**
675
+		 * Note: the reason for the value swap here on $this->_admin_page_title is because $this->_per_page_screen_options
676
+		 * uses the $_admin_page_title property and we want different outputs in the different spots.
677
+		 */
678
+		$page_title = $this->_admin_page_title;
679
+		$this->_admin_page_title = esc_html__('Global Message Templates', 'event_espresso');
680
+		$this->_per_page_screen_option();
681
+		$this->_admin_page_title = $page_title;
682
+	}
683
+
684
+
685
+	protected function _add_screen_options_default()
686
+	{
687
+		$this->_admin_page_title = esc_html__('Message Activity', 'event_espresso');
688
+		$this->_per_page_screen_option();
689
+	}
690
+
691
+
692
+	// none of the below group are currently used for Messages
693
+	protected function _add_feature_pointers()
694
+	{
695
+	}
696
+
697
+	public function admin_init()
698
+	{
699
+	}
700
+
701
+	public function admin_notices()
702
+	{
703
+	}
704
+
705
+	public function admin_footer_scripts()
706
+	{
707
+	}
708
+
709
+
710
+	public function messages_help_tab()
711
+	{
712
+		EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_help_tab.template.php');
713
+	}
714
+
715
+
716
+	public function messengers_help_tab()
717
+	{
718
+		EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messenger_help_tab.template.php');
719
+	}
720
+
721
+
722
+	public function message_types_help_tab()
723
+	{
724
+		EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_message_type_help_tab.template.php');
725
+	}
726
+
727
+
728
+	public function messages_overview_help_tab()
729
+	{
730
+		EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_overview_help_tab.template.php');
731
+	}
732
+
733
+
734
+	public function message_templates_help_tab()
735
+	{
736
+		EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_message_templates_help_tab.template.php');
737
+	}
738
+
739
+
740
+	public function edit_message_template_help_tab()
741
+	{
742
+		$args['img1'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/editor.png' . '" alt="'
743
+						. esc_attr__('Editor Title', 'event_espresso')
744
+						. '" />';
745
+		$args['img2'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/switch-context.png' . '" alt="'
746
+						. esc_attr__('Context Switcher and Preview', 'event_espresso')
747
+						. '" />';
748
+		$args['img3'] = '<img class="left" src="' . EE_MSG_ASSETS_URL . 'images/form-fields.png' . '" alt="'
749
+						. esc_attr__('Message Template Form Fields', 'event_espresso')
750
+						. '" />';
751
+		$args['img4'] = '<img class="right" src="' . EE_MSG_ASSETS_URL . 'images/shortcodes-metabox.png' . '" alt="'
752
+						. esc_attr__('Shortcodes Metabox', 'event_espresso')
753
+						. '" />';
754
+		$args['img5'] = '<img class="right" src="' . EE_MSG_ASSETS_URL . 'images/publish-meta-box.png' . '" alt="'
755
+						. esc_attr__('Publish Metabox', 'event_espresso')
756
+						. '" />';
757
+		EEH_Template::display_template(
758
+			EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_templates_editor_help_tab.template.php',
759
+			$args
760
+		);
761
+	}
762
+
763
+
764
+	public function message_template_shortcodes_help_tab()
765
+	{
766
+		$this->_set_shortcodes();
767
+		$args['shortcodes'] = $this->_shortcodes;
768
+		EEH_Template::display_template(
769
+			EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_shortcodes_help_tab.template.php',
770
+			$args
771
+		);
772
+	}
773
+
774
+
775
+	public function preview_message_help_tab()
776
+	{
777
+		EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_preview_help_tab.template.php');
778
+	}
779
+
780
+
781
+	public function settings_help_tab()
782
+	{
783
+		$args['img1'] = '<img class="inline-text" src="' . EE_MSG_ASSETS_URL . 'images/email-tab-active.png'
784
+						. '" alt="' . esc_attr__('Active Email Tab', 'event_espresso') . '" />';
785
+		$args['img2'] = '<img class="inline-text" src="' . EE_MSG_ASSETS_URL . 'images/email-tab-inactive.png'
786
+						. '" alt="' . esc_attr__('Inactive Email Tab', 'event_espresso') . '" />';
787
+		$args['img3'] = '<div class="switch">'
788
+						. '<input class="ee-on-off-toggle ee-toggle-round-flat"'
789
+						. ' type="checkbox" checked="checked">'
790
+						. '<label for="ee-on-off-toggle-on"></label>'
791
+						. '</div>';
792
+		$args['img4'] = '<div class="switch">'
793
+						. '<input class="ee-on-off-toggle ee-toggle-round-flat"'
794
+						. ' type="checkbox">'
795
+						. '<label for="ee-on-off-toggle-on"></label>'
796
+						. '</div>';
797
+		EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_settings_help_tab.template.php', $args);
798
+	}
799
+
800
+
801
+	public function load_scripts_styles()
802
+	{
803
+		wp_register_style('espresso_ee_msg', EE_MSG_ASSETS_URL . 'ee_message_admin.css', EVENT_ESPRESSO_VERSION);
804
+		wp_enqueue_style('espresso_ee_msg');
805
+
806
+		wp_register_script(
807
+			'ee-messages-settings',
808
+			EE_MSG_ASSETS_URL . 'ee-messages-settings.js',
809
+			array('jquery-ui-droppable', 'ee-serialize-full-array'),
810
+			EVENT_ESPRESSO_VERSION,
811
+			true
812
+		);
813
+		wp_register_script(
814
+			'ee-msg-list-table-js',
815
+			EE_MSG_ASSETS_URL . 'ee_message_admin_list_table.js',
816
+			array('ee-dialog'),
817
+			EVENT_ESPRESSO_VERSION
818
+		);
819
+	}
820
+
821
+
822
+	public function load_scripts_styles_default()
823
+	{
824
+		wp_enqueue_script('ee-msg-list-table-js');
825
+	}
826
+
827
+
828
+	public function wp_editor_css($mce_css)
829
+	{
830
+		// if we're on the edit_message_template route
831
+		if ($this->_req_action === 'edit_message_template' && $this->_active_messenger instanceof EE_messenger) {
832
+			$message_type_name = $this->_active_message_type_name;
833
+
834
+			// we're going to REPLACE the existing mce css
835
+			// we need to get the css file location from the active messenger
836
+			$mce_css = $this->_active_messenger->get_variation(
837
+				$this->_template_pack,
838
+				$message_type_name,
839
+				true,
840
+				'wpeditor',
841
+				$this->_variation
842
+			);
843
+		}
844
+
845
+		return $mce_css;
846
+	}
847
+
848
+
849
+	public function load_scripts_styles_edit_message_template()
850
+	{
851
+
852
+		$this->_set_shortcodes();
853
+
854
+		EE_Registry::$i18n_js_strings['confirm_default_reset'] = sprintf(
855
+			esc_html__(
856
+				'Are you sure you want to reset the %s %s message templates?  Remember continuing will reset the templates for all contexts in this messenger and message type group.',
857
+				'event_espresso'
858
+			),
859
+			$this->_message_template_group->messenger_obj()->label['singular'],
860
+			$this->_message_template_group->message_type_obj()->label['singular']
861
+		);
862
+		EE_Registry::$i18n_js_strings['confirm_switch_template_pack'] = esc_html__(
863
+			'Switching the template pack for a messages template will reset the content for the template so the new layout is loaded.  Any custom content in the existing template will be lost. Are you sure you wish to do this?',
864
+			'event_espresso'
865
+		);
866
+		EE_Registry::$i18n_js_strings['server_error'] = esc_html__(
867
+			'An unknown error occurred on the server while attempting to process your request. Please refresh the page and try again or contact support.',
868
+			'event_espresso'
869
+		);
870
+
871
+		wp_register_script(
872
+			'ee_msgs_edit_js',
873
+			EE_MSG_ASSETS_URL . 'ee_message_editor.js',
874
+			array('jquery'),
875
+			EVENT_ESPRESSO_VERSION
876
+		);
877
+
878
+		wp_enqueue_script('ee_admin_js');
879
+		wp_enqueue_script('ee_msgs_edit_js');
880
+
881
+		// add in special css for tiny_mce
882
+		add_filter('mce_css', array($this, 'wp_editor_css'));
883
+	}
884
+
885
+
886
+	public function load_scripts_styles_display_preview_message()
887
+	{
888
+
889
+		$this->_set_message_template_group();
890
+
891
+		if (isset($this->_req_data['messenger'])) {
892
+			$this->_active_messenger = $this->_message_resource_manager->get_active_messenger(
893
+				$this->_req_data['messenger']
894
+			);
895
+		}
896
+
897
+		$message_type_name = isset($this->_req_data['message_type']) ? $this->_req_data['message_type'] : '';
898
+
899
+
900
+		wp_enqueue_style(
901
+			'espresso_preview_css',
902
+			$this->_active_messenger->get_variation(
903
+				$this->_template_pack,
904
+				$message_type_name,
905
+				true,
906
+				'preview',
907
+				$this->_variation
908
+			)
909
+		);
910
+	}
911
+
912
+
913
+	public function load_scripts_styles_settings()
914
+	{
915
+		wp_register_style(
916
+			'ee-message-settings',
917
+			EE_MSG_ASSETS_URL . 'ee_message_settings.css',
918
+			array(),
919
+			EVENT_ESPRESSO_VERSION
920
+		);
921
+		wp_enqueue_style('ee-text-links');
922
+		wp_enqueue_style('ee-message-settings');
923
+		wp_enqueue_script('ee-messages-settings');
924
+	}
925
+
926
+
927
+	/**
928
+	 * set views array for List Table
929
+	 */
930
+	public function _set_list_table_views_global_mtps()
931
+	{
932
+		$this->_views = array(
933
+			'in_use' => array(
934
+				'slug'  => 'in_use',
935
+				'label' => esc_html__('In Use', 'event_espresso'),
936
+				'count' => 0,
937
+			),
938
+		);
939
+	}
940
+
941
+
942
+	/**
943
+	 * Set views array for the Custom Template List Table
944
+	 */
945
+	public function _set_list_table_views_custom_mtps()
946
+	{
947
+		$this->_set_list_table_views_global_mtps();
948
+		$this->_views['in_use']['bulk_action'] = array(
949
+			'trash_message_template' => esc_html__('Move to Trash', 'event_espresso'),
950
+		);
951
+	}
952
+
953
+
954
+	/**
955
+	 * set views array for message queue list table
956
+	 *
957
+	 * @throws InvalidDataTypeException
958
+	 * @throws InvalidInterfaceException
959
+	 * @throws InvalidArgumentException
960
+	 * @throws EE_Error
961
+	 * @throws ReflectionException
962
+	 */
963
+	public function _set_list_table_views_default()
964
+	{
965
+		EE_Registry::instance()->load_helper('Template');
966
+
967
+		$common_bulk_actions = EE_Registry::instance()->CAP->current_user_can(
968
+			'ee_send_message',
969
+			'message_list_table_bulk_actions'
970
+		)
971
+			? array(
972
+				'generate_now'          => esc_html__('Generate Now', 'event_espresso'),
973
+				'generate_and_send_now' => esc_html__('Generate and Send Now', 'event_espresso'),
974
+				'queue_for_resending'   => esc_html__('Queue for Resending', 'event_espresso'),
975
+				'send_now'              => esc_html__('Send Now', 'event_espresso'),
976
+			)
977
+			: array();
978
+
979
+		$delete_bulk_action = EE_Registry::instance()->CAP->current_user_can(
980
+			'ee_delete_messages',
981
+			'message_list_table_bulk_actions'
982
+		)
983
+			? array('delete_ee_messages' => esc_html__('Delete Messages', 'event_espresso'))
984
+			: array();
985
+
986
+
987
+		$this->_views = array(
988
+			'all' => array(
989
+				'slug'        => 'all',
990
+				'label'       => esc_html__('All', 'event_espresso'),
991
+				'count'       => 0,
992
+				'bulk_action' => array_merge($common_bulk_actions, $delete_bulk_action),
993
+			),
994
+		);
995
+
996
+
997
+		foreach (EEM_Message::instance()->all_statuses() as $status) {
998
+			if ($status === EEM_Message::status_debug_only && ! EEM_Message::debug()) {
999
+				continue;
1000
+			}
1001
+			$status_bulk_actions = $common_bulk_actions;
1002
+			// unset bulk actions not applying to status
1003
+			if (! empty($status_bulk_actions)) {
1004
+				switch ($status) {
1005
+					case EEM_Message::status_idle:
1006
+					case EEM_Message::status_resend:
1007
+						$status_bulk_actions['send_now'] = $common_bulk_actions['send_now'];
1008
+						break;
1009
+
1010
+					case EEM_Message::status_failed:
1011
+					case EEM_Message::status_debug_only:
1012
+					case EEM_Message::status_messenger_executing:
1013
+						$status_bulk_actions = array();
1014
+						break;
1015
+
1016
+					case EEM_Message::status_incomplete:
1017
+						unset($status_bulk_actions['queue_for_resending'], $status_bulk_actions['send_now']);
1018
+						break;
1019
+
1020
+					case EEM_Message::status_retry:
1021
+					case EEM_Message::status_sent:
1022
+						unset($status_bulk_actions['generate_now'], $status_bulk_actions['generate_and_send_now']);
1023
+						break;
1024
+				}
1025
+			}
1026
+
1027
+			// skip adding messenger executing status to views because it will be included with the Failed view.
1028
+			if ($status === EEM_Message::status_messenger_executing) {
1029
+				continue;
1030
+			}
1031
+
1032
+			$this->_views[ strtolower($status) ] = array(
1033
+				'slug'        => strtolower($status),
1034
+				'label'       => EEH_Template::pretty_status($status, false, 'sentence'),
1035
+				'count'       => 0,
1036
+				'bulk_action' => array_merge($status_bulk_actions, $delete_bulk_action),
1037
+			);
1038
+		}
1039
+	}
1040
+
1041
+
1042
+	protected function _ee_default_messages_overview_list_table()
1043
+	{
1044
+		$this->_admin_page_title = esc_html__('Default Message Templates', 'event_espresso');
1045
+		$this->display_admin_list_table_page_with_no_sidebar();
1046
+	}
1047
+
1048
+
1049
+	protected function _message_queue_list_table()
1050
+	{
1051
+		$this->_search_btn_label = esc_html__('Message Activity', 'event_espresso');
1052
+		$this->_template_args['per_column'] = 6;
1053
+		$this->_template_args['after_list_table'] = $this->_display_legend($this->_message_legend_items());
1054
+		$this->_template_args['before_list_table'] = '<h3>'
1055
+													 . EEM_Message::instance()->get_pretty_label_for_results()
1056
+													 . '</h3>';
1057
+		$this->display_admin_list_table_page_with_no_sidebar();
1058
+	}
1059
+
1060
+
1061
+	protected function _message_legend_items()
1062
+	{
1063
+
1064
+		$action_css_classes = EEH_MSG_Template::get_message_action_icons();
1065
+		$action_items = array();
1066
+
1067
+		foreach ($action_css_classes as $action_item => $action_details) {
1068
+			if ($action_item === 'see_notifications_for') {
1069
+				continue;
1070
+			}
1071
+			$action_items[ $action_item ] = array(
1072
+				'class' => $action_details['css_class'],
1073
+				'desc'  => $action_details['label'],
1074
+			);
1075
+		}
1076
+
1077
+		/** @type array $status_items status legend setup */
1078
+		$status_items = array(
1079
+			'sent_status'                => array(
1080
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_sent,
1081
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_sent, false, 'sentence'),
1082
+			),
1083
+			'idle_status'                => array(
1084
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_idle,
1085
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_idle, false, 'sentence'),
1086
+			),
1087
+			'failed_status'              => array(
1088
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_failed,
1089
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_failed, false, 'sentence'),
1090
+			),
1091
+			'messenger_executing_status' => array(
1092
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_messenger_executing,
1093
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_messenger_executing, false, 'sentence'),
1094
+			),
1095
+			'resend_status'              => array(
1096
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_resend,
1097
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_resend, false, 'sentence'),
1098
+			),
1099
+			'incomplete_status'          => array(
1100
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_incomplete,
1101
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_incomplete, false, 'sentence'),
1102
+			),
1103
+			'retry_status'               => array(
1104
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_retry,
1105
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_retry, false, 'sentence'),
1106
+			),
1107
+		);
1108
+		if (EEM_Message::debug()) {
1109
+			$status_items['debug_only_status'] = array(
1110
+				'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_debug_only,
1111
+				'desc'  => EEH_Template::pretty_status(EEM_Message::status_debug_only, false, 'sentence'),
1112
+			);
1113
+		}
1114
+
1115
+		return array_merge($action_items, $status_items);
1116
+	}
1117
+
1118
+
1119
+	protected function _custom_mtps_preview()
1120
+	{
1121
+		$this->_admin_page_title = esc_html__('Custom Message Templates (Preview)', 'event_espresso');
1122
+		$this->_template_args['preview_img'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/custom_mtps_preview.png"'
1123
+											   . ' alt="' . esc_attr__(
1124
+												   'Preview Custom Message Templates screenshot',
1125
+												   'event_espresso'
1126
+											   ) . '" />';
1127
+		$this->_template_args['preview_text'] = '<strong>'
1128
+												. esc_html__(
1129
+													'Custom Message Templates is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. With the Custom Message Templates feature, you are able to create custom message templates and assign them on a per-event basis.',
1130
+													'event_espresso'
1131
+												)
1132
+												. '</strong>';
1133
+
1134
+		$this->display_admin_caf_preview_page('custom_message_types', false);
1135
+	}
1136
+
1137
+
1138
+	/**
1139
+	 * get_message_templates
1140
+	 * This gets all the message templates for listing on the overview list.
1141
+	 *
1142
+	 * @access public
1143
+	 * @param int    $perpage the amount of templates groups to show per page
1144
+	 * @param string $type    the current _view we're getting templates for
1145
+	 * @param bool   $count   return count?
1146
+	 * @param bool   $all     disregard any paging info (get all data);
1147
+	 * @param bool   $global  whether to return just global (true) or custom templates (false)
1148
+	 * @return array
1149
+	 * @throws EE_Error
1150
+	 * @throws InvalidArgumentException
1151
+	 * @throws InvalidDataTypeException
1152
+	 * @throws InvalidInterfaceException
1153
+	 */
1154
+	public function get_message_templates(
1155
+		$perpage = 10,
1156
+		$type = 'in_use',
1157
+		$count = false,
1158
+		$all = false,
1159
+		$global = true
1160
+	) {
1161
+
1162
+		$MTP = EEM_Message_Template_Group::instance();
1163
+
1164
+		$this->_req_data['orderby'] = empty($this->_req_data['orderby']) ? 'GRP_ID' : $this->_req_data['orderby'];
1165
+		$orderby = $this->_req_data['orderby'];
1166
+
1167
+		$order = (isset($this->_req_data['order']) && ! empty($this->_req_data['order']))
1168
+			? $this->_req_data['order']
1169
+			: 'ASC';
1170
+
1171
+		$current_page = isset($this->_req_data['paged']) && ! empty($this->_req_data['paged'])
1172
+			? $this->_req_data['paged']
1173
+			: 1;
1174
+		$per_page = isset($this->_req_data['perpage']) && ! empty($this->_req_data['perpage'])
1175
+			? $this->_req_data['perpage']
1176
+			: $perpage;
1177
+
1178
+		$offset = ($current_page - 1) * $per_page;
1179
+		$limit = $all ? null : array($offset, $per_page);
1180
+
1181
+
1182
+		// options will match what is in the _views array property
1183
+		switch ($type) {
1184
+			case 'in_use':
1185
+				$templates = $MTP->get_all_active_message_templates($orderby, $order, $limit, $count, $global, true);
1186
+				break;
1187
+			default:
1188
+				$templates = $MTP->get_all_trashed_grouped_message_templates($orderby, $order, $limit, $count, $global);
1189
+		}
1190
+
1191
+		return $templates;
1192
+	}
1193
+
1194
+
1195
+	/**
1196
+	 * filters etc might need a list of installed message_types
1197
+	 *
1198
+	 * @return array an array of message type objects
1199
+	 */
1200
+	public function get_installed_message_types()
1201
+	{
1202
+		$installed_message_types = $this->_message_resource_manager->installed_message_types();
1203
+		$installed = array();
1204
+
1205
+		foreach ($installed_message_types as $message_type) {
1206
+			$installed[ $message_type->name ] = $message_type;
1207
+		}
1208
+
1209
+		return $installed;
1210
+	}
1211
+
1212
+
1213
+	/**
1214
+	 * _add_message_template
1215
+	 *
1216
+	 * This is used when creating a custom template. All Custom Templates start based off another template.
1217
+	 *
1218
+	 * @param string $message_type
1219
+	 * @param string $messenger
1220
+	 * @param string $GRP_ID
1221
+	 *
1222
+	 * @throws EE_error
1223
+	 */
1224
+	protected function _add_message_template($message_type = '', $messenger = '', $GRP_ID = '')
1225
+	{
1226
+		// set values override any request data
1227
+		$message_type = ! empty($message_type) ? $message_type : '';
1228
+		$message_type = empty($message_type) && ! empty($this->_req_data['message_type'])
1229
+			? $this->_req_data['message_type']
1230
+			: $message_type;
1231
+
1232
+		$messenger = ! empty($messenger) ? $messenger : '';
1233
+		$messenger = empty($messenger) && ! empty($this->_req_data['messenger'])
1234
+			? $this->_req_data['messenger']
1235
+			: $messenger;
1236
+
1237
+		$GRP_ID = ! empty($GRP_ID) ? $GRP_ID : '';
1238
+		$GRP_ID = empty($GRP_ID) && ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : $GRP_ID;
1239
+
1240
+		// we need messenger and message type.  They should be coming from the event editor. If not here then return error
1241
+		if (empty($message_type) || empty($messenger)) {
1242
+			throw new EE_Error(
1243
+				esc_html__(
1244
+					'Sorry, but we can\'t create new templates because we\'re missing the messenger or message type',
1245
+					'event_espresso'
1246
+				)
1247
+			);
1248
+		}
1249
+
1250
+		// we need the GRP_ID for the template being used as the base for the new template
1251
+		if (empty($GRP_ID)) {
1252
+			throw new EE_Error(
1253
+				esc_html__(
1254
+					'In order to create a custom message template the GRP_ID of the template being used as a base is needed',
1255
+					'event_espresso'
1256
+				)
1257
+			);
1258
+		}
1259
+
1260
+		// let's just make sure the template gets generated!
1261
+
1262
+		// we need to reassign some variables for what the insert is expecting
1263
+		$this->_req_data['MTP_messenger'] = $messenger;
1264
+		$this->_req_data['MTP_message_type'] = $message_type;
1265
+		$this->_req_data['GRP_ID'] = $GRP_ID;
1266
+		$this->_insert_or_update_message_template(true);
1267
+	}
1268
+
1269
+
1270
+	/**
1271
+	 * public wrapper for the _add_message_template method
1272
+	 *
1273
+	 * @param string $message_type     message type slug
1274
+	 * @param string $messenger        messenger slug
1275
+	 * @param int    $GRP_ID           GRP_ID for the related message template group this new template will be based
1276
+	 *                                 off of.
1277
+	 * @throws EE_error
1278
+	 */
1279
+	public function add_message_template($message_type, $messenger, $GRP_ID)
1280
+	{
1281
+		$this->_add_message_template($message_type, $messenger, $GRP_ID);
1282
+	}
1283
+
1284
+
1285
+	/**
1286
+	 * _edit_message_template
1287
+	 *
1288
+	 * @access protected
1289
+	 * @return void
1290
+	 * @throws InvalidIdentifierException
1291
+	 * @throws DomainException
1292
+	 * @throws EE_Error
1293
+	 * @throws InvalidArgumentException
1294
+	 * @throws ReflectionException
1295
+	 * @throws InvalidDataTypeException
1296
+	 * @throws InvalidInterfaceException
1297
+	 */
1298
+	protected function _edit_message_template()
1299
+	{
1300
+		do_action('AHEE_log', __FILE__, __FUNCTION__, '');
1301
+		$template_fields = '';
1302
+		$sidebar_fields = '';
1303
+		// we filter the tinyMCE settings to remove the validation since message templates by their nature will not have
1304
+		// valid html in the templates.
1305
+		add_filter('tiny_mce_before_init', array($this, 'filter_tinymce_init'), 10, 2);
1306
+
1307
+		$GRP_ID = isset($this->_req_data['id']) && ! empty($this->_req_data['id'])
1308
+			? absint($this->_req_data['id'])
1309
+			: false;
1310
+
1311
+		$this->_set_shortcodes(); // this also sets the _message_template property.
1312
+		$message_template_group = $this->_message_template_group;
1313
+		$c_label = $message_template_group->context_label();
1314
+		$c_config = $message_template_group->contexts_config();
1315
+
1316
+		reset($c_config);
1317
+		$context = isset($this->_req_data['context']) && ! empty($this->_req_data['context'])
1318
+			? strtolower($this->_req_data['context'])
1319
+			: key($c_config);
1320
+
1321
+
1322
+		if (empty($GRP_ID)) {
1323
+			$action = 'insert_message_template';
1324
+			$edit_message_template_form_url = add_query_arg(
1325
+				array('action' => $action, 'noheader' => true),
1326
+				EE_MSG_ADMIN_URL
1327
+			);
1328
+		} else {
1329
+			$action = 'update_message_template';
1330
+			$edit_message_template_form_url = add_query_arg(
1331
+				array('action' => $action, 'noheader' => true),
1332
+				EE_MSG_ADMIN_URL
1333
+			);
1334
+		}
1335
+
1336
+		// set active messenger for this view
1337
+		$this->_active_messenger = $this->_message_resource_manager->get_active_messenger(
1338
+			$message_template_group->messenger()
1339
+		);
1340
+		$this->_active_message_type_name = $message_template_group->message_type();
1341
+
1342
+
1343
+		// Do we have any validation errors?
1344
+		$validators = $this->_get_transient();
1345
+		$v_fields = ! empty($validators) ? array_keys($validators) : array();
1346
+
1347
+
1348
+		// we need to assemble the title from Various details
1349
+		$context_label = sprintf(
1350
+			esc_html__('(%s %s)', 'event_espresso'),
1351
+			$c_config[ $context ]['label'],
1352
+			ucwords($c_label['label'])
1353
+		);
1354
+
1355
+		$title = sprintf(
1356
+			esc_html__(' %s %s Template %s', 'event_espresso'),
1357
+			ucwords($message_template_group->messenger_obj()->label['singular']),
1358
+			ucwords($message_template_group->message_type_obj()->label['singular']),
1359
+			$context_label
1360
+		);
1361
+
1362
+		$this->_template_args['GRP_ID'] = $GRP_ID;
1363
+		$this->_template_args['message_template'] = $message_template_group;
1364
+		$this->_template_args['is_extra_fields'] = false;
1365
+
1366
+
1367
+		// let's get EEH_MSG_Template so we can get template form fields
1368
+		$template_field_structure = EEH_MSG_Template::get_fields(
1369
+			$message_template_group->messenger(),
1370
+			$message_template_group->message_type()
1371
+		);
1372
+
1373
+		if (! $template_field_structure) {
1374
+			$template_field_structure = false;
1375
+			$template_fields = esc_html__(
1376
+				'There was an error in assembling the fields for this display (you should see an error message)',
1377
+				'event_espresso'
1378
+			);
1379
+		}
1380
+
1381
+
1382
+		$message_templates = $message_template_group->context_templates();
1383
+
1384
+
1385
+		// if we have the extra key.. then we need to remove the content index from the template_field_structure as it
1386
+		// will get handled in the "extra" array.
1387
+		if (is_array($template_field_structure[ $context ]) && isset($template_field_structure[ $context ]['extra'])) {
1388
+			foreach ($template_field_structure[ $context ]['extra'] as $reference_field => $new_fields) {
1389
+				unset($template_field_structure[ $context ][ $reference_field ]);
1390
+			}
1391
+		}
1392
+
1393
+		// let's loop through the template_field_structure and actually assemble the input fields!
1394
+		if (! empty($template_field_structure)) {
1395
+			foreach ($template_field_structure[ $context ] as $template_field => $field_setup_array) {
1396
+				// if this is an 'extra' template field then we need to remove any existing fields that are keyed up in
1397
+				// the extra array and reset them.
1398
+				if ($template_field === 'extra') {
1399
+					$this->_template_args['is_extra_fields'] = true;
1400
+					foreach ($field_setup_array as $reference_field => $new_fields_array) {
1401
+						$message_template = $message_templates[ $context ][ $reference_field ];
1402
+						$content = $message_template instanceof EE_Message_Template
1403
+							? $message_template->get('MTP_content')
1404
+							: '';
1405
+						foreach ($new_fields_array as $extra_field => $extra_array) {
1406
+							// let's verify if we need this extra field via the shortcodes parameter.
1407
+							$continue = false;
1408
+							if (isset($extra_array['shortcodes_required'])) {
1409
+								foreach ((array) $extra_array['shortcodes_required'] as $shortcode) {
1410
+									if (! array_key_exists($shortcode, $this->_shortcodes)) {
1411
+										$continue = true;
1412
+									}
1413
+								}
1414
+								if ($continue) {
1415
+									continue;
1416
+								}
1417
+							}
1418
+
1419
+							$field_id = $reference_field
1420
+										. '-'
1421
+										. $extra_field
1422
+										. '-content';
1423
+							$template_form_fields[ $field_id ] = $extra_array;
1424
+							$template_form_fields[ $field_id ]['name'] = 'MTP_template_fields['
1425
+																		 . $reference_field
1426
+																		 . '][content]['
1427
+																		 . $extra_field . ']';
1428
+							$css_class = isset($extra_array['css_class'])
1429
+								? $extra_array['css_class']
1430
+								: '';
1431
+
1432
+							$template_form_fields[ $field_id ]['css_class'] = ! empty($v_fields)
1433
+																			  && in_array($extra_field, $v_fields, true)
1434
+																			  &&
1435
+																			  (
1436
+																				  is_array($validators[ $extra_field ])
1437
+																				  && isset($validators[ $extra_field ]['msg'])
1438
+																			  )
1439
+								? 'validate-error ' . $css_class
1440
+								: $css_class;
1441
+
1442
+							$template_form_fields[ $field_id ]['value'] = ! empty($message_templates)
1443
+																		  && isset($content[ $extra_field ])
1444
+								? $content[ $extra_field ]
1445
+								: '';
1446
+
1447
+							// do we have a validation error?  if we do then let's use that value instead
1448
+							$template_form_fields[ $field_id ]['value'] = isset($validators[ $extra_field ])
1449
+								? $validators[ $extra_field ]['value']
1450
+								: $template_form_fields[ $field_id ]['value'];
1451
+
1452
+
1453
+							$template_form_fields[ $field_id ]['db-col'] = 'MTP_content';
1454
+
1455
+							// shortcode selector
1456
+							$field_name_to_use = $extra_field === 'main'
1457
+								? 'content'
1458
+								: $extra_field;
1459
+							$template_form_fields[ $field_id ]['append_content'] = $this->_get_shortcode_selector(
1460
+								$field_name_to_use,
1461
+								$field_id
1462
+							);
1463
+
1464
+							if (isset($extra_array['input']) && $extra_array['input'] === 'wp_editor') {
1465
+								// we want to decode the entities
1466
+								$template_form_fields[ $field_id ]['value'] = $template_form_fields[ $field_id ]['value'];
1467
+							}/**/
1468
+						}
1469
+						$templatefield_MTP_id = $reference_field . '-MTP_ID';
1470
+						$templatefield_templatename_id = $reference_field . '-name';
1471
+
1472
+						$template_form_fields[ $templatefield_MTP_id ] = array(
1473
+							'name'       => 'MTP_template_fields[' . $reference_field . '][MTP_ID]',
1474
+							'label'      => null,
1475
+							'input'      => 'hidden',
1476
+							'type'       => 'int',
1477
+							'required'   => false,
1478
+							'validation' => false,
1479
+							'value'      => ! empty($message_templates) ? $message_template->ID() : '',
1480
+							'css_class'  => '',
1481
+							'format'     => '%d',
1482
+							'db-col'     => 'MTP_ID',
1483
+						);
1484
+
1485
+						$template_form_fields[ $templatefield_templatename_id ] = array(
1486
+							'name'       => 'MTP_template_fields[' . $reference_field . '][name]',
1487
+							'label'      => null,
1488
+							'input'      => 'hidden',
1489
+							'type'       => 'string',
1490
+							'required'   => false,
1491
+							'validation' => true,
1492
+							'value'      => $reference_field,
1493
+							'css_class'  => '',
1494
+							'format'     => '%s',
1495
+							'db-col'     => 'MTP_template_field',
1496
+						);
1497
+					}
1498
+					continue; // skip the next stuff, we got the necessary fields here for this dataset.
1499
+				} else {
1500
+					$field_id = $template_field . '-content';
1501
+					$template_form_fields[ $field_id ] = $field_setup_array;
1502
+					$template_form_fields[ $field_id ]['name'] = 'MTP_template_fields[' . $template_field . '][content]';
1503
+					$message_template = isset($message_templates[ $context ][ $template_field ])
1504
+						? $message_templates[ $context ][ $template_field ]
1505
+						: null;
1506
+					$template_form_fields[ $field_id ]['value'] = ! empty($message_templates)
1507
+																  && is_array($message_templates[ $context ])
1508
+																  && $message_template instanceof EE_Message_Template
1509
+						? $message_template->get('MTP_content')
1510
+						: '';
1511
+
1512
+					// do we have a validator error for this field?  if we do then we'll use that value instead
1513
+					$template_form_fields[ $field_id ]['value'] = isset($validators[ $template_field ])
1514
+						? $validators[ $template_field ]['value']
1515
+						: $template_form_fields[ $field_id ]['value'];
1516
+
1517
+
1518
+					$template_form_fields[ $field_id ]['db-col'] = 'MTP_content';
1519
+					$css_class = isset($field_setup_array['css_class'])
1520
+						? $field_setup_array['css_class']
1521
+						: '';
1522
+					$template_form_fields[ $field_id ]['css_class'] = ! empty($v_fields)
1523
+																	  && in_array($template_field, $v_fields, true)
1524
+																	  && isset($validators[ $template_field ]['msg'])
1525
+						? 'validate-error ' . $css_class
1526
+						: $css_class;
1527
+
1528
+					// shortcode selector
1529
+					$template_form_fields[ $field_id ]['append_content'] = $this->_get_shortcode_selector(
1530
+						$template_field,
1531
+						$field_id
1532
+					);
1533
+				}
1534
+
1535
+				// k took care of content field(s) now let's take care of others.
1536
+
1537
+				$templatefield_MTP_id = $template_field . '-MTP_ID';
1538
+				$templatefield_field_templatename_id = $template_field . '-name';
1539
+
1540
+				// foreach template field there are actually two form fields created
1541
+				$template_form_fields[ $templatefield_MTP_id ] = array(
1542
+					'name'       => 'MTP_template_fields[' . $template_field . '][MTP_ID]',
1543
+					'label'      => null,
1544
+					'input'      => 'hidden',
1545
+					'type'       => 'int',
1546
+					'required'   => false,
1547
+					'validation' => true,
1548
+					'value'      => $message_template instanceof EE_Message_Template ? $message_template->ID() : '',
1549
+					'css_class'  => '',
1550
+					'format'     => '%d',
1551
+					'db-col'     => 'MTP_ID',
1552
+				);
1553
+
1554
+				$template_form_fields[ $templatefield_field_templatename_id ] = array(
1555
+					'name'       => 'MTP_template_fields[' . $template_field . '][name]',
1556
+					'label'      => null,
1557
+					'input'      => 'hidden',
1558
+					'type'       => 'string',
1559
+					'required'   => false,
1560
+					'validation' => true,
1561
+					'value'      => $template_field,
1562
+					'css_class'  => '',
1563
+					'format'     => '%s',
1564
+					'db-col'     => 'MTP_template_field',
1565
+				);
1566
+			}
1567
+
1568
+			// add other fields
1569
+			$template_form_fields['ee-msg-current-context'] = array(
1570
+				'name'       => 'MTP_context',
1571
+				'label'      => null,
1572
+				'input'      => 'hidden',
1573
+				'type'       => 'string',
1574
+				'required'   => false,
1575
+				'validation' => true,
1576
+				'value'      => $context,
1577
+				'css_class'  => '',
1578
+				'format'     => '%s',
1579
+				'db-col'     => 'MTP_context',
1580
+			);
1581
+
1582
+			$template_form_fields['ee-msg-grp-id'] = array(
1583
+				'name'       => 'GRP_ID',
1584
+				'label'      => null,
1585
+				'input'      => 'hidden',
1586
+				'type'       => 'int',
1587
+				'required'   => false,
1588
+				'validation' => true,
1589
+				'value'      => $GRP_ID,
1590
+				'css_class'  => '',
1591
+				'format'     => '%d',
1592
+				'db-col'     => 'GRP_ID',
1593
+			);
1594
+
1595
+			$template_form_fields['ee-msg-messenger'] = array(
1596
+				'name'       => 'MTP_messenger',
1597
+				'label'      => null,
1598
+				'input'      => 'hidden',
1599
+				'type'       => 'string',
1600
+				'required'   => false,
1601
+				'validation' => true,
1602
+				'value'      => $message_template_group->messenger(),
1603
+				'css_class'  => '',
1604
+				'format'     => '%s',
1605
+				'db-col'     => 'MTP_messenger',
1606
+			);
1607
+
1608
+			$template_form_fields['ee-msg-message-type'] = array(
1609
+				'name'       => 'MTP_message_type',
1610
+				'label'      => null,
1611
+				'input'      => 'hidden',
1612
+				'type'       => 'string',
1613
+				'required'   => false,
1614
+				'validation' => true,
1615
+				'value'      => $message_template_group->message_type(),
1616
+				'css_class'  => '',
1617
+				'format'     => '%s',
1618
+				'db-col'     => 'MTP_message_type',
1619
+			);
1620
+
1621
+			$sidebar_form_fields['ee-msg-is-global'] = array(
1622
+				'name'       => 'MTP_is_global',
1623
+				'label'      => esc_html__('Global Template', 'event_espresso'),
1624
+				'input'      => 'hidden',
1625
+				'type'       => 'int',
1626
+				'required'   => false,
1627
+				'validation' => true,
1628
+				'value'      => $message_template_group->get('MTP_is_global'),
1629
+				'css_class'  => '',
1630
+				'format'     => '%d',
1631
+				'db-col'     => 'MTP_is_global',
1632
+			);
1633
+
1634
+			$sidebar_form_fields['ee-msg-is-override'] = array(
1635
+				'name'       => 'MTP_is_override',
1636
+				'label'      => esc_html__('Override all custom', 'event_espresso'),
1637
+				'input'      => $message_template_group->is_global() ? 'checkbox' : 'hidden',
1638
+				'type'       => 'int',
1639
+				'required'   => false,
1640
+				'validation' => true,
1641
+				'value'      => $message_template_group->get('MTP_is_override'),
1642
+				'css_class'  => '',
1643
+				'format'     => '%d',
1644
+				'db-col'     => 'MTP_is_override',
1645
+			);
1646
+
1647
+			$sidebar_form_fields['ee-msg-is-active'] = array(
1648
+				'name'       => 'MTP_is_active',
1649
+				'label'      => esc_html__('Active Template', 'event_espresso'),
1650
+				'input'      => 'hidden',
1651
+				'type'       => 'int',
1652
+				'required'   => false,
1653
+				'validation' => true,
1654
+				'value'      => $message_template_group->is_active(),
1655
+				'css_class'  => '',
1656
+				'format'     => '%d',
1657
+				'db-col'     => 'MTP_is_active',
1658
+			);
1659
+
1660
+			$sidebar_form_fields['ee-msg-deleted'] = array(
1661
+				'name'       => 'MTP_deleted',
1662
+				'label'      => null,
1663
+				'input'      => 'hidden',
1664
+				'type'       => 'int',
1665
+				'required'   => false,
1666
+				'validation' => true,
1667
+				'value'      => $message_template_group->get('MTP_deleted'),
1668
+				'css_class'  => '',
1669
+				'format'     => '%d',
1670
+				'db-col'     => 'MTP_deleted',
1671
+			);
1672
+			$sidebar_form_fields['ee-msg-author'] = array(
1673
+				'name'       => 'MTP_user_id',
1674
+				'label'      => esc_html__('Author', 'event_espresso'),
1675
+				'input'      => 'hidden',
1676
+				'type'       => 'int',
1677
+				'required'   => false,
1678
+				'validation' => false,
1679
+				'value'      => $message_template_group->user(),
1680
+				'format'     => '%d',
1681
+				'db-col'     => 'MTP_user_id',
1682
+			);
1683
+
1684
+			$sidebar_form_fields['ee-msg-route'] = array(
1685
+				'name'  => 'action',
1686
+				'input' => 'hidden',
1687
+				'type'  => 'string',
1688
+				'value' => $action,
1689
+			);
1690
+
1691
+			$sidebar_form_fields['ee-msg-id'] = array(
1692
+				'name'  => 'id',
1693
+				'input' => 'hidden',
1694
+				'type'  => 'int',
1695
+				'value' => $GRP_ID,
1696
+			);
1697
+			$sidebar_form_fields['ee-msg-evt-nonce'] = array(
1698
+				'name'  => $action . '_nonce',
1699
+				'input' => 'hidden',
1700
+				'type'  => 'string',
1701
+				'value' => wp_create_nonce($action . '_nonce'),
1702
+			);
1703
+
1704
+			if (isset($this->_req_data['template_switch']) && $this->_req_data['template_switch']) {
1705
+				$sidebar_form_fields['ee-msg-template-switch'] = array(
1706
+					'name'  => 'template_switch',
1707
+					'input' => 'hidden',
1708
+					'type'  => 'int',
1709
+					'value' => 1,
1710
+				);
1711
+			}
1712
+
1713
+
1714
+			$template_fields = $this->_generate_admin_form_fields($template_form_fields);
1715
+			$sidebar_fields = $this->_generate_admin_form_fields($sidebar_form_fields);
1716
+		} //end if ( !empty($template_field_structure) )
1717
+
1718
+		// set extra content for publish box
1719
+		$this->_template_args['publish_box_extra_content'] = $sidebar_fields;
1720
+		$this->_set_publish_post_box_vars(
1721
+			'id',
1722
+			$GRP_ID,
1723
+			false,
1724
+			add_query_arg(
1725
+				array('action' => 'global_mtps'),
1726
+				$this->_admin_base_url
1727
+			)
1728
+		);
1729
+
1730
+		// add preview button
1731
+		$preview_url = parent::add_query_args_and_nonce(
1732
+			array(
1733
+				'message_type' => $message_template_group->message_type(),
1734
+				'messenger'    => $message_template_group->messenger(),
1735
+				'context'      => $context,
1736
+				'GRP_ID'       => $GRP_ID,
1737
+				'action'       => 'preview_message',
1738
+			),
1739
+			$this->_admin_base_url
1740
+		);
1741
+		$preview_button = '<a href="' . $preview_url . '" class="button-secondary messages-preview-button">'
1742
+						  . esc_html__('Preview', 'event_espresso')
1743
+						  . '</a>';
1744
+
1745
+
1746
+		// setup context switcher
1747
+		$context_switcher_args = array(
1748
+			'page'    => 'espresso_messages',
1749
+			'action'  => 'edit_message_template',
1750
+			'id'      => $GRP_ID,
1751
+			'context' => $context,
1752
+			'extra'   => $preview_button,
1753
+		);
1754
+		$this->_set_context_switcher($message_template_group, $context_switcher_args);
1755
+
1756
+
1757
+		// main box
1758
+		$this->_template_args['template_fields'] = $template_fields;
1759
+		$this->_template_args['sidebar_box_id'] = 'details';
1760
+		$this->_template_args['action'] = $action;
1761
+		$this->_template_args['context'] = $context;
1762
+		$this->_template_args['edit_message_template_form_url'] = $edit_message_template_form_url;
1763
+		$this->_template_args['learn_more_about_message_templates_link'] =
1764
+			$this->_learn_more_about_message_templates_link();
1765
+
1766
+
1767
+		$this->_template_args['before_admin_page_content'] = $this->add_context_switcher();
1768
+		$this->_template_args['before_admin_page_content'] .= $this->add_active_context_element(
1769
+			$message_template_group,
1770
+			$context,
1771
+			$context_label
1772
+		);
1773
+		$this->_template_args['before_admin_page_content'] .= $this->_add_form_element_before();
1774
+		$this->_template_args['after_admin_page_content'] = $this->_add_form_element_after();
1775
+
1776
+		$this->_template_path = $this->_template_args['GRP_ID']
1777
+			? EE_MSG_TEMPLATE_PATH . 'ee_msg_details_main_edit_meta_box.template.php'
1778
+			: EE_MSG_TEMPLATE_PATH . 'ee_msg_details_main_add_meta_box.template.php';
1779
+
1780
+		// send along EE_Message_Template_Group object for further template use.
1781
+		$this->_template_args['MTP'] = $message_template_group;
1782
+
1783
+		$this->_template_args['admin_page_content'] = EEH_Template::display_template(
1784
+			$this->_template_path,
1785
+			$this->_template_args,
1786
+			true
1787
+		);
1788
+
1789
+
1790
+		// finally, let's set the admin_page title
1791
+		$this->_admin_page_title = sprintf(__('Editing %s', 'event_espresso'), $title);
1792
+
1793
+
1794
+		// we need to take care of setting the shortcodes property for use elsewhere.
1795
+		$this->_set_shortcodes();
1796
+
1797
+
1798
+		// final template wrapper
1799
+		$this->display_admin_page_with_sidebar();
1800
+	}
1801
+
1802
+
1803
+	public function filter_tinymce_init($mceInit, $editor_id)
1804
+	{
1805
+		return $mceInit;
1806
+	}
1807
+
1808
+
1809
+	public function add_context_switcher()
1810
+	{
1811
+		return $this->_context_switcher;
1812
+	}
1813
+
1814
+
1815
+	/**
1816
+	 * Adds the activation/deactivation toggle for the message template context.
1817
+	 *
1818
+	 * @param EE_Message_Template_Group $message_template_group
1819
+	 * @param string                    $context
1820
+	 * @param string                    $context_label
1821
+	 * @return string
1822
+	 * @throws DomainException
1823
+	 * @throws EE_Error
1824
+	 * @throws InvalidIdentifierException
1825
+	 */
1826
+	protected function add_active_context_element(
1827
+		EE_Message_Template_Group $message_template_group,
1828
+		$context,
1829
+		$context_label
1830
+	) {
1831
+		$template_args = array(
1832
+			'context'                   => $context,
1833
+			'nonce'                     => wp_create_nonce('activate_' . $context . '_toggle_nonce'),
1834
+			'is_active'                 => $message_template_group->is_context_active($context),
1835
+			'on_off_action'             => $message_template_group->is_context_active($context)
1836
+				? 'context-off'
1837
+				: 'context-on',
1838
+			'context_label'             => str_replace(array('(', ')'), '', $context_label),
1839
+			'message_template_group_id' => $message_template_group->ID(),
1840
+		);
1841
+		return EEH_Template::display_template(
1842
+			EE_MSG_TEMPLATE_PATH . 'ee_msg_editor_active_context_element.template.php',
1843
+			$template_args,
1844
+			true
1845
+		);
1846
+	}
1847
+
1848
+
1849
+	/**
1850
+	 * Ajax callback for `toggle_context_template` ajax action.
1851
+	 * Handles toggling the message context on or off.
1852
+	 *
1853
+	 * @throws EE_Error
1854
+	 * @throws InvalidArgumentException
1855
+	 * @throws InvalidDataTypeException
1856
+	 * @throws InvalidIdentifierException
1857
+	 * @throws InvalidInterfaceException
1858
+	 */
1859
+	public function toggle_context_template()
1860
+	{
1861
+		$success = true;
1862
+		// check for required data
1863
+		if (! isset(
1864
+			$this->_req_data['message_template_group_id'],
1865
+			$this->_req_data['context'],
1866
+			$this->_req_data['status']
1867
+		)) {
1868
+			EE_Error::add_error(
1869
+				esc_html__('Required data for doing this action is not available.', 'event_espresso'),
1870
+				__FILE__,
1871
+				__FUNCTION__,
1872
+				__LINE__
1873
+			);
1874
+			$success = false;
1875
+		}
1876
+
1877
+		$nonce = isset($this->_req_data['toggle_context_nonce'])
1878
+			? sanitize_text_field($this->_req_data['toggle_context_nonce'])
1879
+			: '';
1880
+		$nonce_ref = 'activate_' . $this->_req_data['context'] . '_toggle_nonce';
1881
+		$this->_verify_nonce($nonce, $nonce_ref);
1882
+		$status = $this->_req_data['status'];
1883
+		if ($status !== 'off' && $status !== 'on') {
1884
+			EE_Error::add_error(
1885
+				sprintf(
1886
+					esc_html__('The given status (%s) is not valid. Must be "off" or "on"', 'event_espresso'),
1887
+					$this->_req_data['status']
1888
+				),
1889
+				__FILE__,
1890
+				__FUNCTION__,
1891
+				__LINE__
1892
+			);
1893
+			$success = false;
1894
+		}
1895
+		$message_template_group = EEM_Message_Template_Group::instance()->get_one_by_ID(
1896
+			$this->_req_data['message_template_group_id']
1897
+		);
1898
+		if (! $message_template_group instanceof EE_Message_Template_Group) {
1899
+			EE_Error::add_error(
1900
+				sprintf(
1901
+					esc_html__(
1902
+						'Unable to change the active state because the given id "%1$d" does not match a valid "%2$s"',
1903
+						'event_espresso'
1904
+					),
1905
+					$this->_req_data['message_template_group_id'],
1906
+					'EE_Message_Template_Group'
1907
+				),
1908
+				__FILE__,
1909
+				__FUNCTION__,
1910
+				__LINE__
1911
+			);
1912
+			$success = false;
1913
+		}
1914
+		if ($success) {
1915
+			$success = $status === 'off'
1916
+				? $message_template_group->deactivate_context($this->_req_data['context'])
1917
+				: $message_template_group->activate_context($this->_req_data['context']);
1918
+		}
1919
+		$this->_template_args['success'] = $success;
1920
+		$this->_return_json();
1921
+	}
1922
+
1923
+
1924
+	public function _add_form_element_before()
1925
+	{
1926
+		return '<form method="post" action="'
1927
+			   . $this->_template_args["edit_message_template_form_url"]
1928
+			   . '" id="ee-msg-edit-frm">';
1929
+	}
1930
+
1931
+	public function _add_form_element_after()
1932
+	{
1933
+		return '</form>';
1934
+	}
1935
+
1936
+
1937
+	/**
1938
+	 * This executes switching the template pack for a message template.
1939
+	 *
1940
+	 * @since 4.5.0
1941
+	 * @throws EE_Error
1942
+	 * @throws InvalidDataTypeException
1943
+	 * @throws InvalidInterfaceException
1944
+	 * @throws InvalidArgumentException
1945
+	 * @throws ReflectionException
1946
+	 */
1947
+	public function switch_template_pack()
1948
+	{
1949
+		$GRP_ID = ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : 0;
1950
+		$template_pack = ! empty($this->_req_data['template_pack']) ? $this->_req_data['template_pack'] : '';
1951
+
1952
+		// verify we have needed values.
1953
+		if (empty($GRP_ID) || empty($template_pack)) {
1954
+			$this->_template_args['error'] = true;
1955
+			EE_Error::add_error(
1956
+				esc_html__('The required date for switching templates is not available.', 'event_espresso'),
1957
+				__FILE__,
1958
+				__FUNCTION__,
1959
+				__LINE__
1960
+			);
1961
+		} else {
1962
+			// get template, set the new template_pack and then reset to default
1963
+			/** @type EE_Message_Template_Group $message_template_group */
1964
+			$message_template_group = EEM_Message_Template_Group::instance()->get_one_by_ID($GRP_ID);
1965
+
1966
+			$message_template_group->set_template_pack_name($template_pack);
1967
+			$this->_req_data['msgr'] = $message_template_group->messenger();
1968
+			$this->_req_data['mt'] = $message_template_group->message_type();
1969
+
1970
+			$query_args = $this->_reset_to_default_template();
1971
+
1972
+			if (empty($query_args['id'])) {
1973
+				EE_Error::add_error(
1974
+					esc_html__(
1975
+						'Something went wrong with switching the template pack. Please try again or contact EE support',
1976
+						'event_espresso'
1977
+					),
1978
+					__FILE__,
1979
+					__FUNCTION__,
1980
+					__LINE__
1981
+				);
1982
+				$this->_template_args['error'] = true;
1983
+			} else {
1984
+				$template_label = $message_template_group->get_template_pack()->label;
1985
+				$template_pack_labels = $message_template_group->messenger_obj()->get_supports_labels();
1986
+				EE_Error::add_success(
1987
+					sprintf(
1988
+						esc_html__(
1989
+							'This message template has been successfully switched to use the %1$s %2$s.  Please wait while the page reloads with your new template.',
1990
+							'event_espresso'
1991
+						),
1992
+						$template_label,
1993
+						$template_pack_labels->template_pack
1994
+					)
1995
+				);
1996
+				// generate the redirect url for js.
1997
+				$url = self::add_query_args_and_nonce(
1998
+					$query_args,
1999
+					$this->_admin_base_url
2000
+				);
2001
+				$this->_template_args['data']['redirect_url'] = $url;
2002
+				$this->_template_args['success'] = true;
2003
+			}
2004
+
2005
+			$this->_return_json();
2006
+		}
2007
+	}
2008
+
2009
+
2010
+	/**
2011
+	 * This handles resetting the template for the given messenger/message_type so that users can start from scratch if
2012
+	 * they want.
2013
+	 *
2014
+	 * @access protected
2015
+	 * @return array|null
2016
+	 * @throws EE_Error
2017
+	 * @throws InvalidArgumentException
2018
+	 * @throws InvalidDataTypeException
2019
+	 * @throws InvalidInterfaceException
2020
+	 */
2021
+	protected function _reset_to_default_template()
2022
+	{
2023
+
2024
+		$templates = array();
2025
+		$GRP_ID = ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : 0;
2026
+		// we need to make sure we've got the info we need.
2027
+		if (! isset($this->_req_data['msgr'], $this->_req_data['mt'], $this->_req_data['GRP_ID'])) {
2028
+			EE_Error::add_error(
2029
+				esc_html__(
2030
+					'In order to reset the template to its default we require the messenger, message type, and message template GRP_ID to know what is being reset.  At least one of these is missing.',
2031
+					'event_espresso'
2032
+				),
2033
+				__FILE__,
2034
+				__FUNCTION__,
2035
+				__LINE__
2036
+			);
2037
+		}
2038
+
2039
+		// all templates will be reset to whatever the defaults are
2040
+		// for the global template matching the messenger and message type.
2041
+		$success = ! empty($GRP_ID) ? true : false;
2042
+
2043
+		if ($success) {
2044
+			// let's first determine if the incoming template is a global template,
2045
+			// if it isn't then we need to get the global template matching messenger and message type.
2046
+			// $MTPG = EEM_Message_Template_Group::instance()->get_one_by_ID( $GRP_ID );
2047
+
2048
+
2049
+			// note this is ONLY deleting the template fields (Message Template rows) NOT the message template group.
2050
+			$success = $this->_delete_mtp_permanently($GRP_ID, false);
2051
+
2052
+			if ($success) {
2053
+				// if successfully deleted, lets generate the new ones.
2054
+				// Note. We set GLOBAL to true, because resets on ANY template
2055
+				// will use the related global template defaults for regeneration.
2056
+				// This means that if a custom template is reset it resets to whatever the related global template is.
2057
+				// HOWEVER, we DO keep the template pack and template variation set
2058
+				// for the current custom template when resetting.
2059
+				$templates = $this->_generate_new_templates(
2060
+					$this->_req_data['msgr'],
2061
+					$this->_req_data['mt'],
2062
+					$GRP_ID,
2063
+					true
2064
+				);
2065
+			}
2066
+		}
2067
+
2068
+		// any error messages?
2069
+		if (! $success) {
2070
+			EE_Error::add_error(
2071
+				esc_html__(
2072
+					'Something went wrong with deleting existing templates. Unable to reset to default',
2073
+					'event_espresso'
2074
+				),
2075
+				__FILE__,
2076
+				__FUNCTION__,
2077
+				__LINE__
2078
+			);
2079
+		}
2080
+
2081
+		// all good, let's add a success message!
2082
+		if ($success && ! empty($templates)) {
2083
+			// the info for the template we generated is the first element in the returned array
2084
+			// $templates = $templates[0];
2085
+			EE_Error::overwrite_success();
2086
+			EE_Error::add_success(__('Templates have been reset to defaults.', 'event_espresso'));
2087
+		}
2088
+
2089
+
2090
+		$query_args = array(
2091
+			'id'      => isset($templates['GRP_ID']) ? $templates['GRP_ID'] : null,
2092
+			'context' => isset($templates['MTP_context']) ? $templates['MTP_context'] : null,
2093
+			'action'  => isset($templates['GRP_ID']) ? 'edit_message_template' : 'global_mtps',
2094
+		);
2095
+
2096
+		// if called via ajax then we return query args otherwise redirect
2097
+		if (defined('DOING_AJAX') && DOING_AJAX) {
2098
+			return $query_args;
2099
+		} else {
2100
+			$this->_redirect_after_action(false, '', '', $query_args, true);
2101
+
2102
+			return null;
2103
+		}
2104
+	}
2105
+
2106
+
2107
+	/**
2108
+	 * Retrieve and set the message preview for display.
2109
+	 *
2110
+	 * @param bool $send if TRUE then we are doing an actual TEST send with the results of the preview.
2111
+	 * @return string
2112
+	 * @throws ReflectionException
2113
+	 * @throws EE_Error
2114
+	 * @throws InvalidArgumentException
2115
+	 * @throws InvalidDataTypeException
2116
+	 * @throws InvalidInterfaceException
2117
+	 */
2118
+	public function _preview_message($send = false)
2119
+	{
2120
+		// first make sure we've got the necessary parameters
2121
+		if (! isset(
2122
+			$this->_req_data['message_type'],
2123
+			$this->_req_data['messenger'],
2124
+			$this->_req_data['messenger'],
2125
+			$this->_req_data['GRP_ID']
2126
+		)) {
2127
+			EE_Error::add_error(
2128
+				esc_html__('Missing necessary parameters for displaying preview', 'event_espresso'),
2129
+				__FILE__,
2130
+				__FUNCTION__,
2131
+				__LINE__
2132
+			);
2133
+		}
2134
+
2135
+		EE_Registry::instance()->REQ->set('GRP_ID', $this->_req_data['GRP_ID']);
2136
+
2137
+
2138
+		// get the preview!
2139
+		$preview = EED_Messages::preview_message(
2140
+			$this->_req_data['message_type'],
2141
+			$this->_req_data['context'],
2142
+			$this->_req_data['messenger'],
2143
+			$send
2144
+		);
2145
+
2146
+		if ($send) {
2147
+			return $preview;
2148
+		}
2149
+
2150
+		// let's add a button to go back to the edit view
2151
+		$query_args = array(
2152
+			'id'      => $this->_req_data['GRP_ID'],
2153
+			'context' => $this->_req_data['context'],
2154
+			'action'  => 'edit_message_template',
2155
+		);
2156
+		$go_back_url = parent::add_query_args_and_nonce($query_args, $this->_admin_base_url);
2157
+		$preview_button = '<a href="'
2158
+						  . $go_back_url
2159
+						  . '" class="button-secondary messages-preview-go-back-button">'
2160
+						  . esc_html__('Go Back to Edit', 'event_espresso')
2161
+						  . '</a>';
2162
+		$message_types = $this->get_installed_message_types();
2163
+		$active_messenger = $this->_message_resource_manager->get_active_messenger(
2164
+			$this->_req_data['messenger']
2165
+		);
2166
+		$active_messenger_label = $active_messenger instanceof EE_messenger
2167
+			? ucwords($active_messenger->label['singular'])
2168
+			: esc_html__('Unknown Messenger', 'event_espresso');
2169
+		// let's provide a helpful title for context
2170
+		$preview_title = sprintf(
2171
+			esc_html__('Viewing Preview for %s %s Message Template', 'event_espresso'),
2172
+			$active_messenger_label,
2173
+			ucwords($message_types[ $this->_req_data['message_type'] ]->label['singular'])
2174
+		);
2175
+		// setup display of preview.
2176
+		$this->_admin_page_title = $preview_title;
2177
+		$this->_template_args['admin_page_content'] = $preview_button . '<br />' . $preview;
2178
+		$this->_template_args['data']['force_json'] = true;
2179
+
2180
+		return '';
2181
+	}
2182
+
2183
+
2184
+	/**
2185
+	 * The initial _preview_message is on a no headers route.  It will optionally call this if necessary otherwise it
2186
+	 * gets called automatically.
2187
+	 *
2188
+	 * @since 4.5.0
2189
+	 *
2190
+	 * @return string
2191
+	 */
2192
+	protected function _display_preview_message()
2193
+	{
2194
+		$this->display_admin_page_with_no_sidebar();
2195
+	}
2196
+
2197
+
2198
+	/**
2199
+	 * registers metaboxes that should show up on the "edit_message_template" page
2200
+	 *
2201
+	 * @access protected
2202
+	 * @return void
2203
+	 */
2204
+	protected function _register_edit_meta_boxes()
2205
+	{
2206
+		add_meta_box(
2207
+			'mtp_valid_shortcodes',
2208
+			esc_html__('Valid Shortcodes', 'event_espresso'),
2209
+			array($this, 'shortcode_meta_box'),
2210
+			$this->_current_screen->id,
2211
+			'side',
2212
+			'default'
2213
+		);
2214
+		add_meta_box(
2215
+			'mtp_extra_actions',
2216
+			esc_html__('Extra Actions', 'event_espresso'),
2217
+			array($this, 'extra_actions_meta_box'),
2218
+			$this->_current_screen->id,
2219
+			'side',
2220
+			'high'
2221
+		);
2222
+		add_meta_box(
2223
+			'mtp_templates',
2224
+			esc_html__('Template Styles', 'event_espresso'),
2225
+			array($this, 'template_pack_meta_box'),
2226
+			$this->_current_screen->id,
2227
+			'side',
2228
+			'high'
2229
+		);
2230
+	}
2231
+
2232
+
2233
+	/**
2234
+	 * metabox content for all template pack and variation selection.
2235
+	 *
2236
+	 * @since 4.5.0
2237
+	 * @return string
2238
+	 * @throws DomainException
2239
+	 * @throws EE_Error
2240
+	 * @throws InvalidArgumentException
2241
+	 * @throws ReflectionException
2242
+	 * @throws InvalidDataTypeException
2243
+	 * @throws InvalidInterfaceException
2244
+	 */
2245
+	public function template_pack_meta_box()
2246
+	{
2247
+		$this->_set_message_template_group();
2248
+
2249
+		$tp_collection = EEH_MSG_Template::get_template_pack_collection();
2250
+
2251
+		$tp_select_values = array();
2252
+
2253
+		foreach ($tp_collection as $tp) {
2254
+			// only include template packs that support this messenger and message type!
2255
+			$supports = $tp->get_supports();
2256
+			if (! isset($supports[ $this->_message_template_group->messenger() ])
2257
+				|| ! in_array(
2258
+					$this->_message_template_group->message_type(),
2259
+					$supports[ $this->_message_template_group->messenger() ],
2260
+					true
2261
+				)
2262
+			) {
2263
+				// not supported
2264
+				continue;
2265
+			}
2266
+
2267
+			$tp_select_values[] = array(
2268
+				'text' => $tp->label,
2269
+				'id'   => $tp->dbref,
2270
+			);
2271
+		}
2272
+
2273
+		// if empty $tp_select_values then we make sure default is set because EVERY message type should be supported by
2274
+		// the default template pack.  This still allows for the odd template pack to override.
2275
+		if (empty($tp_select_values)) {
2276
+			$tp_select_values[] = array(
2277
+				'text' => esc_html__('Default', 'event_espresso'),
2278
+				'id'   => 'default',
2279
+			);
2280
+		}
2281
+
2282
+		// setup variation select values for the currently selected template.
2283
+		$variations = $this->_message_template_group->get_template_pack()->get_variations(
2284
+			$this->_message_template_group->messenger(),
2285
+			$this->_message_template_group->message_type()
2286
+		);
2287
+		$variations_select_values = array();
2288
+		foreach ($variations as $variation => $label) {
2289
+			$variations_select_values[] = array(
2290
+				'text' => $label,
2291
+				'id'   => $variation,
2292
+			);
2293
+		}
2294
+
2295
+		$template_pack_labels = $this->_message_template_group->messenger_obj()->get_supports_labels();
2296
+
2297
+		$template_args['template_packs_selector'] = EEH_Form_Fields::select_input(
2298
+			'MTP_template_pack',
2299
+			$tp_select_values,
2300
+			$this->_message_template_group->get_template_pack_name()
2301
+		);
2302
+		$template_args['variations_selector'] = EEH_Form_Fields::select_input(
2303
+			'MTP_template_variation',
2304
+			$variations_select_values,
2305
+			$this->_message_template_group->get_template_pack_variation()
2306
+		);
2307
+		$template_args['template_pack_label'] = $template_pack_labels->template_pack;
2308
+		$template_args['template_variation_label'] = $template_pack_labels->template_variation;
2309
+		$template_args['template_pack_description'] = $template_pack_labels->template_pack_description;
2310
+		$template_args['template_variation_description'] = $template_pack_labels->template_variation_description;
2311
+
2312
+		$template = EE_MSG_TEMPLATE_PATH . 'template_pack_and_variations_metabox.template.php';
2313
+
2314
+		EEH_Template::display_template($template, $template_args);
2315
+	}
2316
+
2317
+
2318
+	/**
2319
+	 * This meta box holds any extra actions related to Message Templates
2320
+	 * For now, this includes Resetting templates to defaults and sending a test email.
2321
+	 *
2322
+	 * @access  public
2323
+	 * @return void
2324
+	 * @throws EE_Error
2325
+	 */
2326
+	public function extra_actions_meta_box()
2327
+	{
2328
+		$template_form_fields = array();
2329
+
2330
+		$extra_args = array(
2331
+			'msgr'   => $this->_message_template_group->messenger(),
2332
+			'mt'     => $this->_message_template_group->message_type(),
2333
+			'GRP_ID' => $this->_message_template_group->GRP_ID(),
2334
+		);
2335
+		// first we need to see if there are any fields
2336
+		$fields = $this->_message_template_group->messenger_obj()->get_test_settings_fields();
2337
+
2338
+		if (! empty($fields)) {
2339
+			// yup there be fields
2340
+			foreach ($fields as $field => $config) {
2341
+				$field_id = $this->_message_template_group->messenger() . '_' . $field;
2342
+				$existing = $this->_message_template_group->messenger_obj()->get_existing_test_settings();
2343
+				$default = isset($config['default']) ? $config['default'] : '';
2344
+				$default = isset($config['value']) ? $config['value'] : $default;
2345
+
2346
+				// if type is hidden and the value is empty
2347
+				// something may have gone wrong so let's correct with the defaults
2348
+				$fix = $config['input'] === 'hidden'
2349
+					   && isset($existing[ $field ])
2350
+					   && empty($existing[ $field ])
2351
+					? $default
2352
+					: '';
2353
+				$existing[ $field ] = isset($existing[ $field ]) && empty($fix)
2354
+					? $existing[ $field ]
2355
+					: $fix;
2356
+
2357
+				$template_form_fields[ $field_id ] = array(
2358
+					'name'       => 'test_settings_fld[' . $field . ']',
2359
+					'label'      => $config['label'],
2360
+					'input'      => $config['input'],
2361
+					'type'       => $config['type'],
2362
+					'required'   => $config['required'],
2363
+					'validation' => $config['validation'],
2364
+					'value'      => isset($existing[ $field ]) ? $existing[ $field ] : $default,
2365
+					'css_class'  => $config['css_class'],
2366
+					'options'    => isset($config['options']) ? $config['options'] : array(),
2367
+					'default'    => $default,
2368
+					'format'     => $config['format'],
2369
+				);
2370
+			}
2371
+		}
2372
+
2373
+		$test_settings_fields = ! empty($template_form_fields)
2374
+			? $this->_generate_admin_form_fields($template_form_fields, 'string', 'ee_tst_settings_flds')
2375
+			: '';
2376
+
2377
+		$test_settings_html = '';
2378
+		// print out $test_settings_fields
2379
+		if (! empty($test_settings_fields)) {
2380
+			echo $test_settings_fields;
2381
+			$test_settings_html = '<input type="submit" class="button-primary mtp-test-button alignright" ';
2382
+			$test_settings_html .= 'name="test_button" value="';
2383
+			$test_settings_html .= esc_html__('Test Send', 'event_espresso');
2384
+			$test_settings_html .= '" /><div style="clear:both"></div>';
2385
+		}
2386
+
2387
+		// and button
2388
+		$test_settings_html .= '<p>'
2389
+							   . esc_html__('Need to reset this message type and start over?', 'event_espresso')
2390
+							   . '</p>';
2391
+		$test_settings_html .= '<div class="publishing-action alignright resetbutton">';
2392
+		$test_settings_html .= $this->get_action_link_or_button(
2393
+			'reset_to_default',
2394
+			'reset',
2395
+			$extra_args,
2396
+			'button-primary reset-default-button'
2397
+		);
2398
+		$test_settings_html .= '</div><div style="clear:both"></div>';
2399
+		echo $test_settings_html;
2400
+	}
2401
+
2402
+
2403
+	/**
2404
+	 * This returns the shortcode selector skeleton for a given context and field.
2405
+	 *
2406
+	 * @since 4.9.rc.000
2407
+	 * @param string $field           The name of the field retrieving shortcodes for.
2408
+	 * @param string $linked_input_id The css id of the input that the shortcodes get added to.
2409
+	 * @return string
2410
+	 * @throws DomainException
2411
+	 * @throws EE_Error
2412
+	 * @throws InvalidArgumentException
2413
+	 * @throws ReflectionException
2414
+	 * @throws InvalidDataTypeException
2415
+	 * @throws InvalidInterfaceException
2416
+	 */
2417
+	protected function _get_shortcode_selector($field, $linked_input_id)
2418
+	{
2419
+		$template_args = array(
2420
+			'shortcodes'      => $this->_get_shortcodes(array($field), true),
2421
+			'fieldname'       => $field,
2422
+			'linked_input_id' => $linked_input_id,
2423
+		);
2424
+
2425
+		return EEH_Template::display_template(
2426
+			EE_MSG_TEMPLATE_PATH . 'shortcode_selector_skeleton.template.php',
2427
+			$template_args,
2428
+			true
2429
+		);
2430
+	}
2431
+
2432
+
2433
+	/**
2434
+	 * This just takes care of returning the meta box content for shortcodes (only used on the edit message template
2435
+	 * page)
2436
+	 *
2437
+	 * @access public
2438
+	 * @return void
2439
+	 * @throws EE_Error
2440
+	 * @throws InvalidArgumentException
2441
+	 * @throws ReflectionException
2442
+	 * @throws InvalidDataTypeException
2443
+	 * @throws InvalidInterfaceException
2444
+	 */
2445
+	public function shortcode_meta_box()
2446
+	{
2447
+		$shortcodes = $this->_get_shortcodes(array(), false); // just make sure shortcodes property is set
2448
+		// $messenger = $this->_message_template_group->messenger_obj();
2449
+		// now let's set the content depending on the status of the shortcodes array
2450
+		if (empty($shortcodes)) {
2451
+			$content = '<p>' . esc_html__('There are no valid shortcodes available', 'event_espresso') . '</p>';
2452
+			echo $content;
2453
+		} else {
2454
+			// $alt = 0;
2455
+			?>
2456 2456
             <div style="float:right; margin-top:10px"><?php
2457
-                            echo $this->_get_help_tab_link('message_template_shortcodes');
2458
-                            ?></div>
2457
+							echo $this->_get_help_tab_link('message_template_shortcodes');
2458
+							?></div>
2459 2459
             <p class="small-text"><?php
2460
-                                  printf(
2461
-                                      esc_html__(
2462
-                                          'You can view the shortcodes usable in your template by clicking the %s icon next to each field.',
2463
-                                          'event_espresso'
2464
-                                      ),
2465
-                                      '<span class="dashicons dashicons-menu"></span>'
2466
-                                  );
2467
-                                ?>
2460
+								  printf(
2461
+									  esc_html__(
2462
+										  'You can view the shortcodes usable in your template by clicking the %s icon next to each field.',
2463
+										  'event_espresso'
2464
+									  ),
2465
+									  '<span class="dashicons dashicons-menu"></span>'
2466
+								  );
2467
+								?>
2468 2468
             </p>
2469 2469
             <?php
2470
-        }
2471
-    }
2472
-
2473
-
2474
-    /**
2475
-     * used to set the $_shortcodes property for when its needed elsewhere.
2476
-     *
2477
-     * @access protected
2478
-     * @return void
2479
-     * @throws EE_Error
2480
-     * @throws InvalidArgumentException
2481
-     * @throws ReflectionException
2482
-     * @throws InvalidDataTypeException
2483
-     * @throws InvalidInterfaceException
2484
-     */
2485
-    protected function _set_shortcodes()
2486
-    {
2487
-
2488
-        // no need to run this if the property is already set
2489
-        if (! empty($this->_shortcodes)) {
2490
-            return;
2491
-        }
2492
-
2493
-        $this->_shortcodes = $this->_get_shortcodes();
2494
-    }
2495
-
2496
-
2497
-    /**
2498
-     * get's all shortcodes for a given template group. (typically used by _set_shortcodes to set the $_shortcodes
2499
-     * property)
2500
-     *
2501
-     * @access  protected
2502
-     * @param  array   $fields include an array of specific field names that you want to be used to get the shortcodes
2503
-     *                         for. Defaults to all (for the given context)
2504
-     * @param  boolean $merged Whether to merge all the shortcodes into one list of unique shortcodes
2505
-     * @return array Shortcodes indexed by fieldname and the an array of shortcode/label pairs OR if merged is
2506
-     *                         true just an array of shortcode/label pairs.
2507
-     * @throws EE_Error
2508
-     * @throws InvalidArgumentException
2509
-     * @throws ReflectionException
2510
-     * @throws InvalidDataTypeException
2511
-     * @throws InvalidInterfaceException
2512
-     */
2513
-    protected function _get_shortcodes($fields = array(), $merged = true)
2514
-    {
2515
-        $this->_set_message_template_group();
2516
-
2517
-        // we need the messenger and message template to retrieve the valid shortcodes array.
2518
-        $GRP_ID = isset($this->_req_data['id']) && ! empty($this->_req_data['id'])
2519
-            ? absint($this->_req_data['id'])
2520
-            : false;
2521
-        $context = isset($this->_req_data['context'])
2522
-            ? $this->_req_data['context']
2523
-            : key($this->_message_template_group->contexts_config());
2524
-
2525
-        return ! empty($GRP_ID) ? $this->_message_template_group->get_shortcodes($context, $fields, $merged) : array();
2526
-    }
2527
-
2528
-
2529
-    /**
2530
-     * This sets the _message_template property (containing the called message_template object)
2531
-     *
2532
-     * @access protected
2533
-     * @return void
2534
-     * @throws EE_Error
2535
-     * @throws InvalidArgumentException
2536
-     * @throws ReflectionException
2537
-     * @throws InvalidDataTypeException
2538
-     * @throws InvalidInterfaceException
2539
-     */
2540
-    protected function _set_message_template_group()
2541
-    {
2542
-
2543
-        if (! empty($this->_message_template_group)) {
2544
-            return;
2545
-        } //get out if this is already set.
2546
-
2547
-        $GRP_ID = ! empty($this->_req_data['GRP_ID']) ? absint($this->_req_data['GRP_ID']) : false;
2548
-        $GRP_ID = empty($GRP_ID) && ! empty($this->_req_data['id']) ? $this->_req_data['id'] : $GRP_ID;
2549
-
2550
-        // let's get the message templates
2551
-        $MTP = EEM_Message_Template_Group::instance();
2552
-
2553
-        if (empty($GRP_ID)) {
2554
-            $this->_message_template_group = $MTP->create_default_object();
2555
-        } else {
2556
-            $this->_message_template_group = $MTP->get_one_by_ID($GRP_ID);
2557
-        }
2558
-
2559
-        $this->_template_pack = $this->_message_template_group->get_template_pack();
2560
-        $this->_variation = $this->_message_template_group->get_template_pack_variation();
2561
-    }
2562
-
2563
-
2564
-    /**
2565
-     * sets up a context switcher for edit forms
2566
-     *
2567
-     * @access  protected
2568
-     * @param  EE_Message_Template_Group $template_group_object the template group object being displayed on the form
2569
-     * @param array                      $args                  various things the context switcher needs.
2570
-     * @throws EE_Error
2571
-     */
2572
-    protected function _set_context_switcher(EE_Message_Template_Group $template_group_object, $args)
2573
-    {
2574
-        $context_details = $template_group_object->contexts_config();
2575
-        $context_label = $template_group_object->context_label();
2576
-        ob_start();
2577
-        ?>
2470
+		}
2471
+	}
2472
+
2473
+
2474
+	/**
2475
+	 * used to set the $_shortcodes property for when its needed elsewhere.
2476
+	 *
2477
+	 * @access protected
2478
+	 * @return void
2479
+	 * @throws EE_Error
2480
+	 * @throws InvalidArgumentException
2481
+	 * @throws ReflectionException
2482
+	 * @throws InvalidDataTypeException
2483
+	 * @throws InvalidInterfaceException
2484
+	 */
2485
+	protected function _set_shortcodes()
2486
+	{
2487
+
2488
+		// no need to run this if the property is already set
2489
+		if (! empty($this->_shortcodes)) {
2490
+			return;
2491
+		}
2492
+
2493
+		$this->_shortcodes = $this->_get_shortcodes();
2494
+	}
2495
+
2496
+
2497
+	/**
2498
+	 * get's all shortcodes for a given template group. (typically used by _set_shortcodes to set the $_shortcodes
2499
+	 * property)
2500
+	 *
2501
+	 * @access  protected
2502
+	 * @param  array   $fields include an array of specific field names that you want to be used to get the shortcodes
2503
+	 *                         for. Defaults to all (for the given context)
2504
+	 * @param  boolean $merged Whether to merge all the shortcodes into one list of unique shortcodes
2505
+	 * @return array Shortcodes indexed by fieldname and the an array of shortcode/label pairs OR if merged is
2506
+	 *                         true just an array of shortcode/label pairs.
2507
+	 * @throws EE_Error
2508
+	 * @throws InvalidArgumentException
2509
+	 * @throws ReflectionException
2510
+	 * @throws InvalidDataTypeException
2511
+	 * @throws InvalidInterfaceException
2512
+	 */
2513
+	protected function _get_shortcodes($fields = array(), $merged = true)
2514
+	{
2515
+		$this->_set_message_template_group();
2516
+
2517
+		// we need the messenger and message template to retrieve the valid shortcodes array.
2518
+		$GRP_ID = isset($this->_req_data['id']) && ! empty($this->_req_data['id'])
2519
+			? absint($this->_req_data['id'])
2520
+			: false;
2521
+		$context = isset($this->_req_data['context'])
2522
+			? $this->_req_data['context']
2523
+			: key($this->_message_template_group->contexts_config());
2524
+
2525
+		return ! empty($GRP_ID) ? $this->_message_template_group->get_shortcodes($context, $fields, $merged) : array();
2526
+	}
2527
+
2528
+
2529
+	/**
2530
+	 * This sets the _message_template property (containing the called message_template object)
2531
+	 *
2532
+	 * @access protected
2533
+	 * @return void
2534
+	 * @throws EE_Error
2535
+	 * @throws InvalidArgumentException
2536
+	 * @throws ReflectionException
2537
+	 * @throws InvalidDataTypeException
2538
+	 * @throws InvalidInterfaceException
2539
+	 */
2540
+	protected function _set_message_template_group()
2541
+	{
2542
+
2543
+		if (! empty($this->_message_template_group)) {
2544
+			return;
2545
+		} //get out if this is already set.
2546
+
2547
+		$GRP_ID = ! empty($this->_req_data['GRP_ID']) ? absint($this->_req_data['GRP_ID']) : false;
2548
+		$GRP_ID = empty($GRP_ID) && ! empty($this->_req_data['id']) ? $this->_req_data['id'] : $GRP_ID;
2549
+
2550
+		// let's get the message templates
2551
+		$MTP = EEM_Message_Template_Group::instance();
2552
+
2553
+		if (empty($GRP_ID)) {
2554
+			$this->_message_template_group = $MTP->create_default_object();
2555
+		} else {
2556
+			$this->_message_template_group = $MTP->get_one_by_ID($GRP_ID);
2557
+		}
2558
+
2559
+		$this->_template_pack = $this->_message_template_group->get_template_pack();
2560
+		$this->_variation = $this->_message_template_group->get_template_pack_variation();
2561
+	}
2562
+
2563
+
2564
+	/**
2565
+	 * sets up a context switcher for edit forms
2566
+	 *
2567
+	 * @access  protected
2568
+	 * @param  EE_Message_Template_Group $template_group_object the template group object being displayed on the form
2569
+	 * @param array                      $args                  various things the context switcher needs.
2570
+	 * @throws EE_Error
2571
+	 */
2572
+	protected function _set_context_switcher(EE_Message_Template_Group $template_group_object, $args)
2573
+	{
2574
+		$context_details = $template_group_object->contexts_config();
2575
+		$context_label = $template_group_object->context_label();
2576
+		ob_start();
2577
+		?>
2578 2578
         <div class="ee-msg-switcher-container">
2579 2579
             <form method="get" action="<?php echo EE_MSG_ADMIN_URL; ?>" id="ee-msg-context-switcher-frm">
2580 2580
                 <?php
2581
-                foreach ($args as $name => $value) {
2582
-                    if ($name === 'context' || empty($value) || $name === 'extra') {
2583
-                        continue;
2584
-                    }
2585
-                    ?>
2581
+				foreach ($args as $name => $value) {
2582
+					if ($name === 'context' || empty($value) || $name === 'extra') {
2583
+						continue;
2584
+					}
2585
+					?>
2586 2586
                     <input type="hidden" name="<?php echo $name; ?>" value="<?php echo $value; ?>"/>
2587 2587
                     <?php
2588
-                }
2589
-                // setup nonce_url
2590
-                wp_nonce_field($args['action'] . '_nonce', $args['action'] . '_nonce', false);
2591
-                ?>
2588
+				}
2589
+				// setup nonce_url
2590
+				wp_nonce_field($args['action'] . '_nonce', $args['action'] . '_nonce', false);
2591
+				?>
2592 2592
                 <select name="context">
2593 2593
                     <?php
2594
-                    $context_templates = $template_group_object->context_templates();
2595
-                    if (is_array($context_templates)) :
2596
-                        foreach ($context_templates as $context => $template_fields) :
2597
-                            $checked = ($context === $args['context']) ? 'selected="selected"' : '';
2598
-                            ?>
2594
+					$context_templates = $template_group_object->context_templates();
2595
+					if (is_array($context_templates)) :
2596
+						foreach ($context_templates as $context => $template_fields) :
2597
+							$checked = ($context === $args['context']) ? 'selected="selected"' : '';
2598
+							?>
2599 2599
                             <option value="<?php echo $context; ?>" <?php echo $checked; ?>>
2600 2600
                                 <?php echo $context_details[ $context ]['label']; ?>
2601 2601
                             </option>
2602 2602
                         <?php endforeach;
2603
-                    endif; ?>
2603
+					endif; ?>
2604 2604
                 </select>
2605 2605
                 <?php $button_text = sprintf(__('Switch %s', 'event_espresso'), ucwords($context_label['label'])); ?>
2606 2606
                 <input id="submit-msg-context-switcher-sbmt" class="button-secondary" type="submit"
@@ -2609,1921 +2609,1921 @@  discard block
 block discarded – undo
2609 2609
             <?php echo $args['extra']; ?>
2610 2610
         </div> <!-- end .ee-msg-switcher-container -->
2611 2611
         <?php
2612
-        $output = ob_get_contents();
2613
-        ob_clean();
2614
-        $this->_context_switcher = $output;
2615
-    }
2616
-
2617
-
2618
-    /**
2619
-     * utility for sanitizing new values coming in.
2620
-     * Note: this is only used when updating a context.
2621
-     *
2622
-     * @access protected
2623
-     *
2624
-     * @param int $index This helps us know which template field to select from the request array.
2625
-     *
2626
-     * @return array
2627
-     */
2628
-    protected function _set_message_template_column_values($index)
2629
-    {
2630
-        if (is_array($this->_req_data['MTP_template_fields'][ $index ]['content'])) {
2631
-            foreach ($this->_req_data['MTP_template_fields'][ $index ]['content'] as $field => $value) {
2632
-                $this->_req_data['MTP_template_fields'][ $index ]['content'][ $field ] = $value;
2633
-            }
2634
-        }
2635
-
2636
-
2637
-        $set_column_values = array(
2638
-            'MTP_ID'             => absint($this->_req_data['MTP_template_fields'][ $index ]['MTP_ID']),
2639
-            'GRP_ID'             => absint($this->_req_data['GRP_ID']),
2640
-            'MTP_user_id'        => absint($this->_req_data['MTP_user_id']),
2641
-            'MTP_messenger'      => strtolower($this->_req_data['MTP_messenger']),
2642
-            'MTP_message_type'   => strtolower($this->_req_data['MTP_message_type']),
2643
-            'MTP_template_field' => strtolower($this->_req_data['MTP_template_fields'][ $index ]['name']),
2644
-            'MTP_context'        => strtolower($this->_req_data['MTP_context']),
2645
-            'MTP_content'        => $this->_req_data['MTP_template_fields'][ $index ]['content'],
2646
-            'MTP_is_global'      => isset($this->_req_data['MTP_is_global'])
2647
-                ? absint($this->_req_data['MTP_is_global'])
2648
-                : 0,
2649
-            'MTP_is_override'    => isset($this->_req_data['MTP_is_override'])
2650
-                ? absint($this->_req_data['MTP_is_override'])
2651
-                : 0,
2652
-            'MTP_deleted'        => absint($this->_req_data['MTP_deleted']),
2653
-            'MTP_is_active'      => absint($this->_req_data['MTP_is_active']),
2654
-        );
2655
-
2656
-
2657
-        return $set_column_values;
2658
-    }
2659
-
2660
-
2661
-    protected function _insert_or_update_message_template($new = false)
2662
-    {
2663
-
2664
-        do_action('AHEE_log', __FILE__, __FUNCTION__, '');
2665
-        $success = 0;
2666
-        $override = false;
2667
-
2668
-        // setup notices description
2669
-        $messenger_slug = ! empty($this->_req_data['MTP_messenger']) ? $this->_req_data['MTP_messenger'] : '';
2670
-
2671
-        // need the message type and messenger objects to be able to use the labels for the notices
2672
-        $messenger_object = $this->_message_resource_manager->get_messenger($messenger_slug);
2673
-        $messenger_label = $messenger_object instanceof EE_messenger
2674
-            ? ucwords($messenger_object->label['singular'])
2675
-            : '';
2676
-
2677
-        $message_type_slug = ! empty($this->_req_data['MTP_message_type'])
2678
-            ? $this->_req_data['MTP_message_type']
2679
-            : '';
2680
-        $message_type_object = $this->_message_resource_manager->get_message_type($message_type_slug);
2681
-
2682
-        $message_type_label = $message_type_object instanceof EE_message_type
2683
-            ? ucwords($message_type_object->label['singular'])
2684
-            : '';
2685
-
2686
-        $context_slug = ! empty($this->_req_data['MTP_context'])
2687
-            ? $this->_req_data['MTP_context']
2688
-            : '';
2689
-        $context = ucwords(str_replace('_', ' ', $context_slug));
2690
-
2691
-        $item_desc = $messenger_label && $message_type_label
2692
-            ? $messenger_label . ' ' . $message_type_label . ' ' . $context . ' '
2693
-            : '';
2694
-        $item_desc .= 'Message Template';
2695
-        $query_args = array();
2696
-        $edit_array = array();
2697
-        $action_desc = '';
2698
-
2699
-        // if this is "new" then we need to generate the default contexts for the selected messenger/message_type for
2700
-        // user to edit.
2701
-        if ($new) {
2702
-            $GRP_ID = ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : 0;
2703
-            if ($edit_array = $this->_generate_new_templates($messenger_slug, $message_type_slug, $GRP_ID)) {
2704
-                if (empty($edit_array)) {
2705
-                    $success = 0;
2706
-                } else {
2707
-                    $success = 1;
2708
-                    $edit_array = $edit_array[0];
2709
-                    $query_args = array(
2710
-                        'id'      => $edit_array['GRP_ID'],
2711
-                        'context' => $edit_array['MTP_context'],
2712
-                        'action'  => 'edit_message_template',
2713
-                    );
2714
-                }
2715
-            }
2716
-            $action_desc = 'created';
2717
-        } else {
2718
-            $MTPG = EEM_Message_Template_Group::instance();
2719
-            $MTP = EEM_Message_Template::instance();
2720
-
2721
-
2722
-            // run update for each template field in displayed context
2723
-            if (! isset($this->_req_data['MTP_template_fields']) && empty($this->_req_data['MTP_template_fields'])) {
2724
-                EE_Error::add_error(
2725
-                    esc_html__(
2726
-                        'There was a problem saving the template fields from the form because I didn\'t receive any actual template field data.',
2727
-                        'event_espresso'
2728
-                    ),
2729
-                    __FILE__,
2730
-                    __FUNCTION__,
2731
-                    __LINE__
2732
-                );
2733
-                $success = 0;
2734
-            } else {
2735
-                // first validate all fields!
2736
-                // this filter allows client code to add its own validation to the template fields as well.
2737
-                // returning an empty array means everything passed validation.
2738
-                // errors in validation should be represented in an array with the following shape:
2739
-                // array(
2740
-                //   'fieldname' => array(
2741
-                //          'msg' => 'error message'
2742
-                //          'value' => 'value for field producing error'
2743
-                // )
2744
-                $custom_validation = (array) apply_filters(
2745
-                    'FHEE__Messages_Admin_Page___insert_or_update_message_template__validates',
2746
-                    array(),
2747
-                    $this->_req_data['MTP_template_fields'],
2748
-                    $context_slug,
2749
-                    $messenger_slug,
2750
-                    $message_type_slug
2751
-                );
2752
-
2753
-                $system_validation = $MTPG->validate(
2754
-                    $this->_req_data['MTP_template_fields'],
2755
-                    $context_slug,
2756
-                    $messenger_slug,
2757
-                    $message_type_slug
2758
-                );
2759
-
2760
-                $system_validation = ! is_array($system_validation) && $system_validation ? array()
2761
-                    : $system_validation;
2762
-                $validates = array_merge($custom_validation, $system_validation);
2763
-
2764
-                // if $validate returned error messages (i.e. is_array()) then we need to process them and setup an
2765
-                // appropriate response. HMM, dang this isn't correct, $validates will ALWAYS be an array.
2766
-                //  WE need to make sure there is no actual error messages in validates.
2767
-                if (is_array($validates) && ! empty($validates)) {
2768
-                    // add the transient so when the form loads we know which fields to highlight
2769
-                    $this->_add_transient('edit_message_template', $validates);
2770
-
2771
-                    $success = 0;
2772
-
2773
-                    // setup notices
2774
-                    foreach ($validates as $field => $error) {
2775
-                        if (isset($error['msg'])) {
2776
-                            EE_Error::add_error($error['msg'], __FILE__, __FUNCTION__, __LINE__);
2777
-                        }
2778
-                    }
2779
-                } else {
2780
-                    $set_column_values = array();
2781
-                    foreach ($this->_req_data['MTP_template_fields'] as $template_field => $content) {
2782
-                        $set_column_values = $this->_set_message_template_column_values($template_field);
2783
-
2784
-                        $where_cols_n_values = array(
2785
-                            'MTP_ID' => $this->_req_data['MTP_template_fields'][ $template_field ]['MTP_ID'],
2786
-                        );
2787
-                        // if they aren't allowed to use all JS, restrict them to just posty-y tags
2788
-                        if (! current_user_can('unfiltered_html')) {
2789
-                            if (is_array($set_column_values['MTP_content'])) {
2790
-                                foreach ($set_column_values['MTP_content'] as $key => $value) {
2791
-                                    // remove slashes so wp_kses works properly (its wp_kses_stripslashes() function
2792
-                                    // only removes slashes from double-quotes, so attributes using single quotes always
2793
-                                    // appear invalid.) But currently the models expect slashed data, so after wp_kses
2794
-                                    // runs we need to re-slash the data. Sheesh. See
2795
-                                    // https://events.codebasehq.com/projects/event-espresso/tickets/11211#update-47321587
2796
-                                    $set_column_values['MTP_content'][ $key ] = addslashes(
2797
-                                        wp_kses(
2798
-                                            stripslashes($value),
2799
-                                            wp_kses_allowed_html('post')
2800
-                                        )
2801
-                                    );
2802
-                                }
2803
-                            } else {
2804
-                                $set_column_values['MTP_content'] = wp_kses(
2805
-                                    $set_column_values['MTP_content'],
2806
-                                    wp_kses_allowed_html('post')
2807
-                                );
2808
-                            }
2809
-                        }
2810
-                        $message_template_fields = array(
2811
-                            'GRP_ID'             => $set_column_values['GRP_ID'],
2812
-                            'MTP_template_field' => $set_column_values['MTP_template_field'],
2813
-                            'MTP_context'        => $set_column_values['MTP_context'],
2814
-                            'MTP_content'        => $set_column_values['MTP_content'],
2815
-                        );
2816
-                        if ($updated = $MTP->update($message_template_fields, array($where_cols_n_values))) {
2817
-                            if ($updated === false) {
2818
-                                EE_Error::add_error(
2819
-                                    sprintf(
2820
-                                        esc_html__('%s field was NOT updated for some reason', 'event_espresso'),
2821
-                                        $template_field
2822
-                                    ),
2823
-                                    __FILE__,
2824
-                                    __FUNCTION__,
2825
-                                    __LINE__
2826
-                                );
2827
-                            } else {
2828
-                                $success = 1;
2829
-                            }
2830
-                        } else {
2831
-                            // only do this logic if we don't have a MTP_ID for this field
2832
-                            if (empty($this->_req_data['MTP_template_fields'][ $template_field ]['MTP_ID'])) {
2833
-                                // this has already been through the template field validator and sanitized, so it will be
2834
-                                // safe to insert this field.  Why insert?  This typically happens when we introduce a new
2835
-                                // message template field in a messenger/message type and existing users don't have the
2836
-                                // default setup for it.
2837
-                                // @link https://events.codebasehq.com/projects/event-espresso/tickets/9465
2838
-                                $updated = $MTP->insert($message_template_fields);
2839
-                                if (! $updated || is_wp_error($updated)) {
2840
-                                    EE_Error::add_error(
2841
-                                        sprintf(
2842
-                                            esc_html__('%s field could not be updated.', 'event_espresso'),
2843
-                                            $template_field
2844
-                                        ),
2845
-                                        __FILE__,
2846
-                                        __FUNCTION__,
2847
-                                        __LINE__
2848
-                                    );
2849
-                                    $success = 0;
2850
-                                } else {
2851
-                                    $success = 1;
2852
-                                }
2853
-                            }
2854
-                        }
2855
-                        $action_desc = 'updated';
2856
-                    }
2857
-
2858
-                    // we can use the last set_column_values for the MTPG update (because its the same for all of these specific MTPs)
2859
-                    $mtpg_fields = array(
2860
-                        'MTP_user_id'      => $set_column_values['MTP_user_id'],
2861
-                        'MTP_messenger'    => $set_column_values['MTP_messenger'],
2862
-                        'MTP_message_type' => $set_column_values['MTP_message_type'],
2863
-                        'MTP_is_global'    => $set_column_values['MTP_is_global'],
2864
-                        'MTP_is_override'  => $set_column_values['MTP_is_override'],
2865
-                        'MTP_deleted'      => $set_column_values['MTP_deleted'],
2866
-                        'MTP_is_active'    => $set_column_values['MTP_is_active'],
2867
-                        'MTP_name'         => ! empty($this->_req_data['ee_msg_non_global_fields']['MTP_name'])
2868
-                            ? $this->_req_data['ee_msg_non_global_fields']['MTP_name']
2869
-                            : '',
2870
-                        'MTP_description'  => ! empty($this->_req_data['ee_msg_non_global_fields']['MTP_description'])
2871
-                            ? $this->_req_data['ee_msg_non_global_fields']['MTP_description']
2872
-                            : '',
2873
-                    );
2874
-
2875
-                    $mtpg_where = array('GRP_ID' => $set_column_values['GRP_ID']);
2876
-                    $updated = $MTPG->update($mtpg_fields, array($mtpg_where));
2877
-
2878
-                    if ($updated === false) {
2879
-                        EE_Error::add_error(
2880
-                            sprintf(
2881
-                                esc_html__(
2882
-                                    'The Message Template Group (%d) was NOT updated for some reason',
2883
-                                    'event_espresso'
2884
-                                ),
2885
-                                $set_column_values['GRP_ID']
2886
-                            ),
2887
-                            __FILE__,
2888
-                            __FUNCTION__,
2889
-                            __LINE__
2890
-                        );
2891
-                    } else {
2892
-                        // k now we need to ensure the template_pack and template_variation fields are set.
2893
-                        $template_pack = ! empty($this->_req_data['MTP_template_pack'])
2894
-                            ? $this->_req_data['MTP_template_pack']
2895
-                            : 'default';
2896
-
2897
-                        $template_variation = ! empty($this->_req_data['MTP_template_variation'])
2898
-                            ? $this->_req_data['MTP_template_variation']
2899
-                            : 'default';
2900
-
2901
-                        $mtpg_obj = $MTPG->get_one_by_ID($set_column_values['GRP_ID']);
2902
-                        if ($mtpg_obj instanceof EE_Message_Template_Group) {
2903
-                            $mtpg_obj->set_template_pack_name($template_pack);
2904
-                            $mtpg_obj->set_template_pack_variation($template_variation);
2905
-                        }
2906
-                        $success = 1;
2907
-                    }
2908
-                }
2909
-            }
2910
-        }
2911
-
2912
-        // we return things differently if doing ajax
2913
-        if (defined('DOING_AJAX') && DOING_AJAX) {
2914
-            $this->_template_args['success'] = $success;
2915
-            $this->_template_args['error'] = ! $success ? true : false;
2916
-            $this->_template_args['content'] = '';
2917
-            $this->_template_args['data'] = array(
2918
-                'grpID'        => $edit_array['GRP_ID'],
2919
-                'templateName' => $edit_array['template_name'],
2920
-            );
2921
-            if ($success) {
2922
-                EE_Error::overwrite_success();
2923
-                EE_Error::add_success(
2924
-                    esc_html__(
2925
-                        'The new template has been created and automatically selected for this event.  You can edit the new template by clicking the edit button.  Note before this template is assigned to this event, the event must be saved.',
2926
-                        'event_espresso'
2927
-                    )
2928
-                );
2929
-            }
2930
-
2931
-            $this->_return_json();
2932
-        }
2933
-
2934
-
2935
-        // was a test send triggered?
2936
-        if (isset($this->_req_data['test_button'])) {
2937
-            EE_Error::overwrite_success();
2938
-            $this->_do_test_send($context_slug, $messenger_slug, $message_type_slug);
2939
-            $override = true;
2940
-        }
2941
-
2942
-        if (empty($query_args)) {
2943
-            $query_args = array(
2944
-                'id'      => $this->_req_data['GRP_ID'],
2945
-                'context' => $context_slug,
2946
-                'action'  => 'edit_message_template',
2947
-            );
2948
-        }
2949
-
2950
-        $this->_redirect_after_action($success, $item_desc, $action_desc, $query_args, $override);
2951
-    }
2952
-
2953
-
2954
-    /**
2955
-     * processes a test send request to do an actual messenger delivery test for the given message template being tested
2956
-     *
2957
-     * @param  string $context      what context being tested
2958
-     * @param  string $messenger    messenger being tested
2959
-     * @param  string $message_type message type being tested
2960
-     * @throws EE_Error
2961
-     * @throws InvalidArgumentException
2962
-     * @throws InvalidDataTypeException
2963
-     * @throws InvalidInterfaceException
2964
-     */
2965
-    protected function _do_test_send($context, $messenger, $message_type)
2966
-    {
2967
-        // set things up for preview
2968
-        $this->_req_data['messenger'] = $messenger;
2969
-        $this->_req_data['message_type'] = $message_type;
2970
-        $this->_req_data['context'] = $context;
2971
-        $this->_req_data['GRP_ID'] = isset($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : '';
2972
-        $active_messenger = $this->_message_resource_manager->get_active_messenger($messenger);
2973
-
2974
-        // let's save any existing fields that might be required by the messenger
2975
-        if (isset($this->_req_data['test_settings_fld'])
2976
-            && $active_messenger instanceof EE_messenger
2977
-            && apply_filters(
2978
-                'FHEE__Messages_Admin_Page__do_test_send__set_existing_test_settings',
2979
-                true,
2980
-                $this->_req_data['test_settings_fld'],
2981
-                $active_messenger
2982
-            )
2983
-        ) {
2984
-            $active_messenger->set_existing_test_settings($this->_req_data['test_settings_fld']);
2985
-        }
2986
-
2987
-        /**
2988
-         * Use filter to add additional controls on whether message can send or not
2989
-         */
2990
-        if (apply_filters(
2991
-            'FHEE__Messages_Admin_Page__do_test_send__can_send',
2992
-            true,
2993
-            $context,
2994
-            $this->_req_data,
2995
-            $messenger,
2996
-            $message_type
2997
-        )) {
2998
-            $success = $this->_preview_message(true);
2999
-            if ($success) {
3000
-                EE_Error::add_success(__('Test message sent', 'event_espresso'));
3001
-            } else {
3002
-                EE_Error::add_error(
3003
-                    esc_html__('The test message was not sent', 'event_espresso'),
3004
-                    __FILE__,
3005
-                    __FUNCTION__,
3006
-                    __LINE__
3007
-                );
3008
-            }
3009
-        }
3010
-    }
3011
-
3012
-
3013
-    /**
3014
-     * _generate_new_templates
3015
-     * This will handle the messenger, message_type selection when "adding a new custom template" for an event and will
3016
-     * automatically create the defaults for the event.  The user would then be redirected to edit the default context
3017
-     * for the event.
3018
-     *
3019
-     *
3020
-     * @param  string $messenger     the messenger we are generating templates for
3021
-     * @param array   $message_types array of message types that the templates are generated for.
3022
-     * @param int     $GRP_ID        If this is a custom template being generated then a GRP_ID needs to be included to
3023
-     *                               indicate the message_template_group being used as the base.
3024
-     *
3025
-     * @param bool    $global
3026
-     *
3027
-     * @return array|bool array of data required for the redirect to the correct edit page or bool if
3028
-     *                               encountering problems.
3029
-     * @throws EE_Error
3030
-     */
3031
-    protected function _generate_new_templates($messenger, $message_types, $GRP_ID = 0, $global = false)
3032
-    {
3033
-
3034
-        // if no $message_types are given then that's okay... this may be a messenger that just adds shortcodes, so we
3035
-        // just don't generate any templates.
3036
-        if (empty($message_types)) {
3037
-            return true;
3038
-        }
3039
-
3040
-        return EEH_MSG_Template::generate_new_templates($messenger, $message_types, $GRP_ID, $global);
3041
-    }
3042
-
3043
-
3044
-    /**
3045
-     * [_trash_or_restore_message_template]
3046
-     *
3047
-     * @param  boolean $trash whether to move an item to trash/restore (TRUE) or restore it (FALSE)
3048
-     * @param boolean  $all   whether this is going to trash/restore all contexts within a template group (TRUE) OR just
3049
-     *                        an individual context (FALSE).
3050
-     * @return void
3051
-     * @throws EE_Error
3052
-     * @throws InvalidArgumentException
3053
-     * @throws InvalidDataTypeException
3054
-     * @throws InvalidInterfaceException
3055
-     */
3056
-    protected function _trash_or_restore_message_template($trash = true, $all = false)
3057
-    {
3058
-        do_action('AHEE_log', __FILE__, __FUNCTION__, '');
3059
-        $MTP = EEM_Message_Template_Group::instance();
3060
-
3061
-        $success = 1;
3062
-
3063
-        // incoming GRP_IDs
3064
-        if ($all) {
3065
-            // Checkboxes
3066
-            if (! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3067
-                // if array has more than one element then success message should be plural.
3068
-                // todo: what about nonce?
3069
-                $success = count($this->_req_data['checkbox']) > 1 ? 2 : 1;
3070
-
3071
-                // cycle through checkboxes
3072
-                while (list($GRP_ID, $value) = each($this->_req_data['checkbox'])) {
3073
-                    $trashed_or_restored = $trash ? $MTP->delete_by_ID($GRP_ID) : $MTP->restore_by_ID($GRP_ID);
3074
-                    if (! $trashed_or_restored) {
3075
-                        $success = 0;
3076
-                    }
3077
-                }
3078
-            } else {
3079
-                // grab single GRP_ID and handle
3080
-                $GRP_ID = isset($this->_req_data['id']) ? absint($this->_req_data['id']) : 0;
3081
-                if (! empty($GRP_ID)) {
3082
-                    $trashed_or_restored = $trash ? $MTP->delete_by_ID($GRP_ID) : $MTP->restore_by_ID($GRP_ID);
3083
-                    if (! $trashed_or_restored) {
3084
-                        $success = 0;
3085
-                    }
3086
-                } else {
3087
-                    $success = 0;
3088
-                }
3089
-            }
3090
-        }
3091
-
3092
-        $action_desc = $trash
3093
-            ? esc_html__('moved to the trash', 'event_espresso')
3094
-            : esc_html__('restored', 'event_espresso');
3095
-
3096
-        $action_desc = ! empty($this->_req_data['template_switch']) ? esc_html__('switched', 'event_espresso') : $action_desc;
3097
-
3098
-        $item_desc = $all ? _n(
3099
-            'Message Template Group',
3100
-            'Message Template Groups',
3101
-            $success,
3102
-            'event_espresso'
3103
-        ) : _n('Message Template Context', 'Message Template Contexts', $success, 'event_espresso');
3104
-
3105
-        $item_desc = ! empty($this->_req_data['template_switch']) ? _n(
3106
-            'template',
3107
-            'templates',
3108
-            $success,
3109
-            'event_espresso'
3110
-        ) : $item_desc;
3111
-
3112
-        $this->_redirect_after_action($success, $item_desc, $action_desc, array());
3113
-    }
3114
-
3115
-
3116
-    /**
3117
-     * [_delete_message_template]
3118
-     * NOTE: this handles not only the deletion of the groups but also all the templates belonging to that group.
3119
-     *
3120
-     * @return void
3121
-     * @throws EE_Error
3122
-     * @throws InvalidArgumentException
3123
-     * @throws InvalidDataTypeException
3124
-     * @throws InvalidInterfaceException
3125
-     */
3126
-    protected function _delete_message_template()
3127
-    {
3128
-        do_action('AHEE_log', __FILE__, __FUNCTION__, '');
3129
-
3130
-        // checkboxes
3131
-        if (! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3132
-            // if array has more than one element then success message should be plural
3133
-            $success = count($this->_req_data['checkbox']) > 1 ? 2 : 1;
3134
-
3135
-            // cycle through bulk action checkboxes
3136
-            while (list($GRP_ID, $value) = each($this->_req_data['checkbox'])) {
3137
-                $success = $this->_delete_mtp_permanently($GRP_ID);
3138
-            }
3139
-        } else {
3140
-            // grab single grp_id and delete
3141
-            $GRP_ID = absint($this->_req_data['id']);
3142
-            $success = $this->_delete_mtp_permanently($GRP_ID);
3143
-        }
3144
-
3145
-        $this->_redirect_after_action($success, 'Message Templates', 'deleted', array());
3146
-    }
3147
-
3148
-
3149
-    /**
3150
-     * helper for permanently deleting a mtP group and all related message_templates
3151
-     *
3152
-     * @param  int  $GRP_ID        The group being deleted
3153
-     * @param  bool $include_group whether to delete the Message Template Group as well.
3154
-     * @return bool boolean to indicate the success of the deletes or not.
3155
-     * @throws EE_Error
3156
-     * @throws InvalidArgumentException
3157
-     * @throws InvalidDataTypeException
3158
-     * @throws InvalidInterfaceException
3159
-     */
3160
-    private function _delete_mtp_permanently($GRP_ID, $include_group = true)
3161
-    {
3162
-        $success = 1;
3163
-        $MTPG = EEM_Message_Template_Group::instance();
3164
-        // first let's GET this group
3165
-        $MTG = $MTPG->get_one_by_ID($GRP_ID);
3166
-        // then delete permanently all the related Message Templates
3167
-        $deleted = $MTG->delete_related_permanently('Message_Template');
3168
-
3169
-        if ($deleted === 0) {
3170
-            $success = 0;
3171
-        }
3172
-
3173
-        // now delete permanently this particular group
3174
-
3175
-        if ($include_group && ! $MTG->delete_permanently()) {
3176
-            $success = 0;
3177
-        }
3178
-
3179
-        return $success;
3180
-    }
3181
-
3182
-
3183
-    /**
3184
-     *    _learn_more_about_message_templates_link
3185
-     *
3186
-     * @access protected
3187
-     * @return string
3188
-     */
3189
-    protected function _learn_more_about_message_templates_link()
3190
-    {
3191
-        return '<a class="hidden" style="margin:0 20px; cursor:pointer; font-size:12px;" >'
3192
-               . esc_html__('learn more about how message templates works', 'event_espresso')
3193
-               . '</a>';
3194
-    }
3195
-
3196
-
3197
-    /**
3198
-     * Used for setting up messenger/message type activation.  This loads up the initial view.  The rest is handled by
3199
-     * ajax and other routes.
3200
-     *
3201
-     * @return void
3202
-     * @throws DomainException
3203
-     */
3204
-    protected function _settings()
3205
-    {
3206
-
3207
-
3208
-        $this->_set_m_mt_settings();
3209
-
3210
-        $selected_messenger = isset($this->_req_data['selected_messenger'])
3211
-            ? $this->_req_data['selected_messenger']
3212
-            : 'email';
3213
-
3214
-        // let's setup the messenger tabs
3215
-        $this->_template_args['admin_page_header'] = EEH_Tabbed_Content::tab_text_links(
3216
-            $this->_m_mt_settings['messenger_tabs'],
3217
-            'messenger_links',
3218
-            '|',
3219
-            $selected_messenger
3220
-        );
3221
-        $this->_template_args['before_admin_page_content'] = '<div class="ui-widget ui-helper-clearfix">';
3222
-        $this->_template_args['after_admin_page_content'] = '</div><!-- end .ui-widget -->';
3223
-
3224
-        $this->display_admin_page_with_sidebar();
3225
-    }
3226
-
3227
-
3228
-    /**
3229
-     * This sets the $_m_mt_settings property for when needed (used on the Messages settings page)
3230
-     *
3231
-     * @access protected
3232
-     * @return void
3233
-     * @throws DomainException
3234
-     */
3235
-    protected function _set_m_mt_settings()
3236
-    {
3237
-        // first if this is already set then lets get out no need to regenerate data.
3238
-        if (! empty($this->_m_mt_settings)) {
3239
-            return;
3240
-        }
3241
-
3242
-        // get all installed messengers and message_types
3243
-        /** @type EE_messenger[] $messengers */
3244
-        $messengers = $this->_message_resource_manager->installed_messengers();
3245
-        /** @type EE_message_type[] $message_types */
3246
-        $message_types = $this->_message_resource_manager->installed_message_types();
3247
-
3248
-
3249
-        // assemble the array for the _tab_text_links helper
3250
-
3251
-        foreach ($messengers as $messenger) {
3252
-            $this->_m_mt_settings['messenger_tabs'][ $messenger->name ] = array(
3253
-                'label' => ucwords($messenger->label['singular']),
3254
-                'class' => $this->_message_resource_manager->is_messenger_active($messenger->name)
3255
-                    ? 'messenger-active'
3256
-                    : '',
3257
-                'href'  => $messenger->name,
3258
-                'title' => esc_html__('Modify this Messenger', 'event_espresso'),
3259
-                'slug'  => $messenger->name,
3260
-                'obj'   => $messenger,
3261
-            );
3262
-
3263
-
3264
-            $message_types_for_messenger = $messenger->get_valid_message_types();
3265
-
3266
-            foreach ($message_types as $message_type) {
3267
-                // first we need to verify that this message type is valid with this messenger. Cause if it isn't then
3268
-                // it shouldn't show in either the inactive OR active metabox.
3269
-                if (! in_array($message_type->name, $message_types_for_messenger, true)) {
3270
-                    continue;
3271
-                }
3272
-
3273
-                $a_or_i = $this->_message_resource_manager->is_message_type_active_for_messenger(
3274
-                    $messenger->name,
3275
-                    $message_type->name
3276
-                )
3277
-                    ? 'active'
3278
-                    : 'inactive';
3279
-
3280
-                $this->_m_mt_settings['message_type_tabs'][ $messenger->name ][ $a_or_i ][ $message_type->name ] = array(
3281
-                    'label'    => ucwords($message_type->label['singular']),
3282
-                    'class'    => 'message-type-' . $a_or_i,
3283
-                    'slug_id'  => $message_type->name . '-messagetype-' . $messenger->name,
3284
-                    'mt_nonce' => wp_create_nonce($message_type->name . '_nonce'),
3285
-                    'href'     => 'espresso_' . $message_type->name . '_message_type_settings',
3286
-                    'title'    => $a_or_i === 'active'
3287
-                        ? esc_html__('Drag this message type to the Inactive window to deactivate', 'event_espresso')
3288
-                        : esc_html__('Drag this message type to the messenger to activate', 'event_espresso'),
3289
-                    'content'  => $a_or_i === 'active'
3290
-                        ? $this->_message_type_settings_content($message_type, $messenger, true)
3291
-                        : $this->_message_type_settings_content($message_type, $messenger),
3292
-                    'slug'     => $message_type->name,
3293
-                    'active'   => $a_or_i === 'active',
3294
-                    'obj'      => $message_type,
3295
-                );
3296
-            }
3297
-        }
3298
-    }
3299
-
3300
-
3301
-    /**
3302
-     * This just prepares the content for the message type settings
3303
-     *
3304
-     * @param  EE_message_type $message_type The message type object
3305
-     * @param  EE_messenger    $messenger    The messenger object
3306
-     * @param  boolean         $active       Whether the message type is active or not
3307
-     * @return string html output for the content
3308
-     * @throws DomainException
3309
-     */
3310
-    protected function _message_type_settings_content($message_type, $messenger, $active = false)
3311
-    {
3312
-        // get message type fields
3313
-        $fields = $message_type->get_admin_settings_fields();
3314
-        $settings_template_args['template_form_fields'] = '';
3315
-
3316
-        if (! empty($fields) && $active) {
3317
-            $existing_settings = $message_type->get_existing_admin_settings($messenger->name);
3318
-            foreach ($fields as $fldname => $fldprops) {
3319
-                $field_id = $messenger->name . '-' . $message_type->name . '-' . $fldname;
3320
-                $template_form_field[ $field_id ] = array(
3321
-                    'name'       => 'message_type_settings[' . $fldname . ']',
3322
-                    'label'      => $fldprops['label'],
3323
-                    'input'      => $fldprops['field_type'],
3324
-                    'type'       => $fldprops['value_type'],
3325
-                    'required'   => $fldprops['required'],
3326
-                    'validation' => $fldprops['validation'],
3327
-                    'value'      => isset($existing_settings[ $fldname ])
3328
-                        ? $existing_settings[ $fldname ]
3329
-                        : $fldprops['default'],
3330
-                    'options'    => isset($fldprops['options'])
3331
-                        ? $fldprops['options']
3332
-                        : array(),
3333
-                    'default'    => isset($existing_settings[ $fldname ])
3334
-                        ? $existing_settings[ $fldname ]
3335
-                        : $fldprops['default'],
3336
-                    'css_class'  => 'no-drag',
3337
-                    'format'     => $fldprops['format'],
3338
-                );
3339
-            }
3340
-
3341
-
3342
-            $settings_template_args['template_form_fields'] = ! empty($template_form_field)
3343
-                ? $this->_generate_admin_form_fields(
3344
-                    $template_form_field,
3345
-                    'string',
3346
-                    'ee_mt_activate_form'
3347
-                )
3348
-                : '';
3349
-        }
3350
-
3351
-        $settings_template_args['description'] = $message_type->description;
3352
-        // we also need some hidden fields
3353
-        $settings_template_args['hidden_fields'] = array(
3354
-            'message_type_settings[messenger]'    => array(
3355
-                'type'  => 'hidden',
3356
-                'value' => $messenger->name,
3357
-            ),
3358
-            'message_type_settings[message_type]' => array(
3359
-                'type'  => 'hidden',
3360
-                'value' => $message_type->name,
3361
-            ),
3362
-            'type'                                => array(
3363
-                'type'  => 'hidden',
3364
-                'value' => 'message_type',
3365
-            ),
3366
-        );
3367
-
3368
-        $settings_template_args['hidden_fields'] = $this->_generate_admin_form_fields(
3369
-            $settings_template_args['hidden_fields'],
3370
-            'array'
3371
-        );
3372
-        $settings_template_args['show_form'] = empty($settings_template_args['template_form_fields'])
3373
-            ? ' hidden'
3374
-            : '';
3375
-
3376
-
3377
-        $template = EE_MSG_TEMPLATE_PATH . 'ee_msg_mt_settings_content.template.php';
3378
-        $content = EEH_Template::display_template($template, $settings_template_args, true);
3379
-
3380
-        return $content;
3381
-    }
3382
-
3383
-
3384
-    /**
3385
-     * Generate all the metaboxes for the message types and register them for the messages settings page.
3386
-     *
3387
-     * @access protected
3388
-     * @return void
3389
-     * @throws DomainException
3390
-     */
3391
-    protected function _messages_settings_metaboxes()
3392
-    {
3393
-        $this->_set_m_mt_settings();
3394
-        $m_boxes = $mt_boxes = array();
3395
-        $m_template_args = $mt_template_args = array();
3396
-
3397
-        $selected_messenger = isset($this->_req_data['selected_messenger'])
3398
-            ? $this->_req_data['selected_messenger']
3399
-            : 'email';
3400
-
3401
-        if (isset($this->_m_mt_settings['messenger_tabs'])) {
3402
-            foreach ($this->_m_mt_settings['messenger_tabs'] as $messenger => $tab_array) {
3403
-                $hide_on_message = $this->_message_resource_manager->is_messenger_active($messenger) ? '' : 'hidden';
3404
-                $hide_off_message = $this->_message_resource_manager->is_messenger_active($messenger) ? 'hidden' : '';
3405
-                // messenger meta boxes
3406
-                $active = $selected_messenger === $messenger;
3407
-                $active_mt_tabs = isset(
3408
-                    $this->_m_mt_settings['message_type_tabs'][ $messenger ]['active']
3409
-                )
3410
-                    ? $this->_m_mt_settings['message_type_tabs'][ $messenger ]['active']
3411
-                    : '';
3412
-                $m_boxes[ $messenger . '_a_box' ] = sprintf(
3413
-                    esc_html__('%s Settings', 'event_espresso'),
3414
-                    $tab_array['label']
3415
-                );
3416
-                $m_template_args[ $messenger . '_a_box' ] = array(
3417
-                    'active_message_types'   => ! empty($active_mt_tabs) ? $this->_get_mt_tabs($active_mt_tabs) : '',
3418
-                    'inactive_message_types' => isset(
3419
-                        $this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive']
3420
-                    )
3421
-                        ? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive'])
3422
-                        : '',
3423
-                    'content'                => $this->_get_messenger_box_content($tab_array['obj']),
3424
-                    'hidden'                 => $active ? '' : ' hidden',
3425
-                    'hide_on_message'        => $hide_on_message,
3426
-                    'messenger'              => $messenger,
3427
-                    'active'                 => $active,
3428
-                );
3429
-                // message type meta boxes
3430
-                // (which is really just the inactive container for each messenger
3431
-                // showing inactive message types for that messenger)
3432
-                $mt_boxes[ $messenger . '_i_box' ] = esc_html__('Inactive Message Types', 'event_espresso');
3433
-                $mt_template_args[ $messenger . '_i_box' ] = array(
3434
-                    'active_message_types'   => ! empty($active_mt_tabs) ? $this->_get_mt_tabs($active_mt_tabs) : '',
3435
-                    'inactive_message_types' => isset(
3436
-                        $this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive']
3437
-                    )
3438
-                        ? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive'])
3439
-                        : '',
3440
-                    'hidden'                 => $active ? '' : ' hidden',
3441
-                    'hide_on_message'        => $hide_on_message,
3442
-                    'hide_off_message'       => $hide_off_message,
3443
-                    'messenger'              => $messenger,
3444
-                    'active'                 => $active,
3445
-                );
3446
-            }
3447
-        }
3448
-
3449
-
3450
-        // register messenger metaboxes
3451
-        $m_template_path = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_messenger_mt_meta_box.template.php';
3452
-        foreach ($m_boxes as $box => $label) {
3453
-            $callback_args = array('template_path' => $m_template_path, 'template_args' => $m_template_args[ $box ]);
3454
-            $msgr = str_replace('_a_box', '', $box);
3455
-            add_meta_box(
3456
-                'espresso_' . $msgr . '_settings',
3457
-                $label,
3458
-                function ($post, $metabox) {
3459
-                    echo EEH_Template::display_template(
3460
-                        $metabox["args"]["template_path"],
3461
-                        $metabox["args"]["template_args"],
3462
-                        true
3463
-                    );
3464
-                },
3465
-                $this->_current_screen->id,
3466
-                'normal',
3467
-                'high',
3468
-                $callback_args
3469
-            );
3470
-        }
3471
-
3472
-        // register message type metaboxes
3473
-        $mt_template_path = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_messenger_meta_box.template.php';
3474
-        foreach ($mt_boxes as $box => $label) {
3475
-            $callback_args = array(
3476
-                'template_path' => $mt_template_path,
3477
-                'template_args' => $mt_template_args[ $box ],
3478
-            );
3479
-            $mt = str_replace('_i_box', '', $box);
3480
-            add_meta_box(
3481
-                'espresso_' . $mt . '_inactive_mts',
3482
-                $label,
3483
-                function ($post, $metabox) {
3484
-                    echo EEH_Template::display_template(
3485
-                        $metabox["args"]["template_path"],
3486
-                        $metabox["args"]["template_args"],
3487
-                        true
3488
-                    );
3489
-                },
3490
-                $this->_current_screen->id,
3491
-                'side',
3492
-                'high',
3493
-                $callback_args
3494
-            );
3495
-        }
3496
-
3497
-        // register metabox for global messages settings but only when on the main site.  On single site installs this
3498
-        // will always result in the metabox showing, on multisite installs the metabox will only show on the main site.
3499
-        if (is_main_site()) {
3500
-            add_meta_box(
3501
-                'espresso_global_message_settings',
3502
-                esc_html__('Global Message Settings', 'event_espresso'),
3503
-                array($this, 'global_messages_settings_metabox_content'),
3504
-                $this->_current_screen->id,
3505
-                'normal',
3506
-                'low',
3507
-                array()
3508
-            );
3509
-        }
3510
-    }
3511
-
3512
-
3513
-    /**
3514
-     *  This generates the content for the global messages settings metabox.
3515
-     *
3516
-     * @return string
3517
-     * @throws EE_Error
3518
-     * @throws InvalidArgumentException
3519
-     * @throws ReflectionException
3520
-     * @throws InvalidDataTypeException
3521
-     * @throws InvalidInterfaceException
3522
-     */
3523
-    public function global_messages_settings_metabox_content()
3524
-    {
3525
-        $form = $this->_generate_global_settings_form();
3526
-        echo $form->form_open(
3527
-            $this->add_query_args_and_nonce(array('action' => 'update_global_settings'), EE_MSG_ADMIN_URL),
3528
-            'POST'
3529
-        )
3530
-             . $form->get_html()
3531
-             . $form->form_close();
3532
-    }
3533
-
3534
-
3535
-    /**
3536
-     * This generates and returns the form object for the global messages settings.
3537
-     *
3538
-     * @return EE_Form_Section_Proper
3539
-     * @throws EE_Error
3540
-     * @throws InvalidArgumentException
3541
-     * @throws ReflectionException
3542
-     * @throws InvalidDataTypeException
3543
-     * @throws InvalidInterfaceException
3544
-     */
3545
-    protected function _generate_global_settings_form()
3546
-    {
3547
-        EE_Registry::instance()->load_helper('HTML');
3548
-        /** @var EE_Network_Core_Config $network_config */
3549
-        $network_config = EE_Registry::instance()->NET_CFG->core;
3550
-
3551
-        return new EE_Form_Section_Proper(
3552
-            array(
3553
-                'name'            => 'global_messages_settings',
3554
-                'html_id'         => 'global_messages_settings',
3555
-                'html_class'      => 'form-table',
3556
-                'layout_strategy' => new EE_Admin_Two_Column_Layout(),
3557
-                'subsections'     => apply_filters(
3558
-                    'FHEE__Messages_Admin_Page__global_messages_settings_metabox_content__form_subsections',
3559
-                    array(
3560
-                        'do_messages_on_same_request' => new EE_Select_Input(
3561
-                            array(
3562
-                                true  => esc_html__("On the same request", "event_espresso"),
3563
-                                false => esc_html__("On a separate request", "event_espresso"),
3564
-                            ),
3565
-                            array(
3566
-                                'default'         => $network_config->do_messages_on_same_request,
3567
-                                'html_label_text' => esc_html__(
3568
-                                    'Generate and send all messages:',
3569
-                                    'event_espresso'
3570
-                                ),
3571
-                                'html_help_text'  => esc_html__(
3572
-                                    'By default the messages system uses a more efficient means of processing messages on separate requests and utilizes the wp-cron scheduling system.  This makes things execute faster for people registering for your events.  However, if the wp-cron system is disabled on your site and there is no alternative in place, then you can change this so messages are always executed on the same request.',
3573
-                                    'event_espresso'
3574
-                                ),
3575
-                            )
3576
-                        ),
3577
-                        'delete_threshold'            => new EE_Select_Input(
3578
-                            array(
3579
-                                0  => esc_html__('Forever', 'event_espresso'),
3580
-                                3  => esc_html__('3 Months', 'event_espresso'),
3581
-                                6  => esc_html__('6 Months', 'event_espresso'),
3582
-                                9  => esc_html__('9 Months', 'event_espresso'),
3583
-                                12 => esc_html__('12 Months', 'event_espresso'),
3584
-                                24 => esc_html__('24 Months', 'event_espresso'),
3585
-                                36 => esc_html__('36 Months', 'event_espresso'),
3586
-                            ),
3587
-                            array(
3588
-                                'default'         => EE_Registry::instance()->CFG->messages->delete_threshold,
3589
-                                'html_label_text' => esc_html__('Cleanup of old messages:', 'event_espresso'),
3590
-                                'html_help_text'  => esc_html__(
3591
-                                    'You can control how long a record of processed messages is kept via this option.',
3592
-                                    'event_espresso'
3593
-                                ),
3594
-                            )
3595
-                        ),
3596
-                        'update_settings'             => new EE_Submit_Input(
3597
-                            array(
3598
-                                'default'         => esc_html__('Update', 'event_espresso'),
3599
-                                'html_label_text' => '&nbsp',
3600
-                            )
3601
-                        ),
3602
-                    )
3603
-                ),
3604
-            )
3605
-        );
3606
-    }
3607
-
3608
-
3609
-    /**
3610
-     * This handles updating the global settings set on the admin page.
3611
-     *
3612
-     * @throws EE_Error
3613
-     * @throws InvalidDataTypeException
3614
-     * @throws InvalidInterfaceException
3615
-     * @throws InvalidArgumentException
3616
-     * @throws ReflectionException
3617
-     */
3618
-    protected function _update_global_settings()
3619
-    {
3620
-        /** @var EE_Network_Core_Config $network_config */
3621
-        $network_config = EE_Registry::instance()->NET_CFG->core;
3622
-        $messages_config = EE_Registry::instance()->CFG->messages;
3623
-        $form = $this->_generate_global_settings_form();
3624
-        if ($form->was_submitted()) {
3625
-            $form->receive_form_submission();
3626
-            if ($form->is_valid()) {
3627
-                $valid_data = $form->valid_data();
3628
-                foreach ($valid_data as $property => $value) {
3629
-                    $setter = 'set_' . $property;
3630
-                    if (method_exists($network_config, $setter)) {
3631
-                        $network_config->{$setter}($value);
3632
-                    } elseif (property_exists($network_config, $property)
3633
-                        && $network_config->{$property} !== $value
3634
-                    ) {
3635
-                        $network_config->{$property} = $value;
3636
-                    } elseif (property_exists($messages_config, $property)
3637
-                        && $messages_config->{$property} !== $value
3638
-                    ) {
3639
-                        $messages_config->{$property} = $value;
3640
-                    }
3641
-                }
3642
-                // only update if the form submission was valid!
3643
-                EE_Registry::instance()->NET_CFG->update_config(true, false);
3644
-                EE_Registry::instance()->CFG->update_espresso_config();
3645
-                EE_Error::overwrite_success();
3646
-                EE_Error::add_success(__('Global message settings were updated', 'event_espresso'));
3647
-            }
3648
-        }
3649
-        $this->_redirect_after_action(0, '', '', array('action' => 'settings'), true);
3650
-    }
3651
-
3652
-
3653
-    /**
3654
-     * this prepares the messenger tabs that can be dragged in and out of messenger boxes to activate/deactivate
3655
-     *
3656
-     * @param  array $tab_array This is an array of message type tab details used to generate the tabs
3657
-     * @return string html formatted tabs
3658
-     * @throws DomainException
3659
-     */
3660
-    protected function _get_mt_tabs($tab_array)
3661
-    {
3662
-        $tab_array = (array) $tab_array;
3663
-        $template = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_mt_settings_tab_item.template.php';
3664
-        $tabs = '';
3665
-
3666
-        foreach ($tab_array as $tab) {
3667
-            $tabs .= EEH_Template::display_template($template, $tab, true);
3668
-        }
3669
-
3670
-        return $tabs;
3671
-    }
3672
-
3673
-
3674
-    /**
3675
-     * This prepares the content of the messenger meta box admin settings
3676
-     *
3677
-     * @param  EE_messenger $messenger The messenger we're setting up content for
3678
-     * @return string html formatted content
3679
-     * @throws DomainException
3680
-     */
3681
-    protected function _get_messenger_box_content(EE_messenger $messenger)
3682
-    {
3683
-
3684
-        $fields = $messenger->get_admin_settings_fields();
3685
-        $settings_template_args['template_form_fields'] = '';
3686
-
3687
-        // is $messenger active?
3688
-        $settings_template_args['active'] = $this->_message_resource_manager->is_messenger_active($messenger->name);
3689
-
3690
-
3691
-        if (! empty($fields)) {
3692
-            $existing_settings = $messenger->get_existing_admin_settings();
3693
-
3694
-            foreach ($fields as $fldname => $fldprops) {
3695
-                $field_id = $messenger->name . '-' . $fldname;
3696
-                $template_form_field[ $field_id ] = array(
3697
-                    'name'       => 'messenger_settings[' . $field_id . ']',
3698
-                    'label'      => $fldprops['label'],
3699
-                    'input'      => $fldprops['field_type'],
3700
-                    'type'       => $fldprops['value_type'],
3701
-                    'required'   => $fldprops['required'],
3702
-                    'validation' => $fldprops['validation'],
3703
-                    'value'      => isset($existing_settings[ $field_id ])
3704
-                        ? $existing_settings[ $field_id ]
3705
-                        : $fldprops['default'],
3706
-                    'css_class'  => '',
3707
-                    'format'     => $fldprops['format'],
3708
-                );
3709
-            }
3710
-
3711
-
3712
-            $settings_template_args['template_form_fields'] = ! empty($template_form_field)
3713
-                ? $this->_generate_admin_form_fields($template_form_field, 'string', 'ee_m_activate_form')
3714
-                : '';
3715
-        }
3716
-
3717
-        // we also need some hidden fields
3718
-        $settings_template_args['hidden_fields'] = array(
3719
-            'messenger_settings[messenger]' => array(
3720
-                'type'  => 'hidden',
3721
-                'value' => $messenger->name,
3722
-            ),
3723
-            'type'                          => array(
3724
-                'type'  => 'hidden',
3725
-                'value' => 'messenger',
3726
-            ),
3727
-        );
3728
-
3729
-        // make sure any active message types that are existing are included in the hidden fields
3730
-        if (isset($this->_m_mt_settings['message_type_tabs'][ $messenger->name ]['active'])) {
3731
-            foreach ($this->_m_mt_settings['message_type_tabs'][ $messenger->name ]['active'] as $mt => $values) {
3732
-                $settings_template_args['hidden_fields'][ 'messenger_settings[message_types][' . $mt . ']' ] = array(
3733
-                    'type'  => 'hidden',
3734
-                    'value' => $mt,
3735
-                );
3736
-            }
3737
-        }
3738
-        $settings_template_args['hidden_fields'] = $this->_generate_admin_form_fields(
3739
-            $settings_template_args['hidden_fields'],
3740
-            'array'
3741
-        );
3742
-        $active = $this->_message_resource_manager->is_messenger_active($messenger->name);
3743
-
3744
-        $settings_template_args['messenger'] = $messenger->name;
3745
-        $settings_template_args['description'] = $messenger->description;
3746
-        $settings_template_args['show_hide_edit_form'] = $active ? '' : ' hidden';
3747
-
3748
-
3749
-        $settings_template_args['show_hide_edit_form'] = $this->_message_resource_manager->is_messenger_active(
3750
-            $messenger->name
3751
-        )
3752
-            ? $settings_template_args['show_hide_edit_form']
3753
-            : ' hidden';
3754
-
3755
-        $settings_template_args['show_hide_edit_form'] = empty($settings_template_args['template_form_fields'])
3756
-            ? ' hidden'
3757
-            : $settings_template_args['show_hide_edit_form'];
3758
-
3759
-
3760
-        $settings_template_args['on_off_action'] = $active ? 'messenger-off' : 'messenger-on';
3761
-        $settings_template_args['nonce'] = wp_create_nonce('activate_' . $messenger->name . '_toggle_nonce');
3762
-        $settings_template_args['on_off_status'] = $active ? true : false;
3763
-        $template = EE_MSG_TEMPLATE_PATH . 'ee_msg_m_settings_content.template.php';
3764
-        $content = EEH_Template::display_template(
3765
-            $template,
3766
-            $settings_template_args,
3767
-            true
3768
-        );
3769
-
3770
-        return $content;
3771
-    }
3772
-
3773
-
3774
-    /**
3775
-     * used by ajax on the messages settings page to activate|deactivate the messenger
3776
-     *
3777
-     * @throws DomainException
3778
-     * @throws EE_Error
3779
-     * @throws InvalidDataTypeException
3780
-     * @throws InvalidInterfaceException
3781
-     * @throws InvalidArgumentException
3782
-     * @throws ReflectionException
3783
-     */
3784
-    public function activate_messenger_toggle()
3785
-    {
3786
-        $success = true;
3787
-        $this->_prep_default_response_for_messenger_or_message_type_toggle();
3788
-        // let's check that we have required data
3789
-        if (! isset($this->_req_data['messenger'])) {
3790
-            EE_Error::add_error(
3791
-                esc_html__('Messenger name needed to toggle activation. None given', 'event_espresso'),
3792
-                __FILE__,
3793
-                __FUNCTION__,
3794
-                __LINE__
3795
-            );
3796
-            $success = false;
3797
-        }
3798
-
3799
-        // do a nonce check here since we're not arriving via a normal route
3800
-        $nonce = isset($this->_req_data['activate_nonce'])
3801
-            ? sanitize_text_field($this->_req_data['activate_nonce'])
3802
-            : '';
3803
-        $nonce_ref = 'activate_' . $this->_req_data['messenger'] . '_toggle_nonce';
3804
-
3805
-        $this->_verify_nonce($nonce, $nonce_ref);
3806
-
3807
-
3808
-        if (! isset($this->_req_data['status'])) {
3809
-            EE_Error::add_error(
3810
-                esc_html__(
3811
-                    'Messenger status needed to know whether activation or deactivation is happening. No status is given',
3812
-                    'event_espresso'
3813
-                ),
3814
-                __FILE__,
3815
-                __FUNCTION__,
3816
-                __LINE__
3817
-            );
3818
-            $success = false;
3819
-        }
3820
-
3821
-        // do check to verify we have a valid status.
3822
-        $status = $this->_req_data['status'];
3823
-
3824
-        if ($status !== 'off' && $status !== 'on') {
3825
-            EE_Error::add_error(
3826
-                sprintf(
3827
-                    esc_html__('The given status (%s) is not valid. Must be "off" or "on"', 'event_espresso'),
3828
-                    $this->_req_data['status']
3829
-                ),
3830
-                __FILE__,
3831
-                __FUNCTION__,
3832
-                __LINE__
3833
-            );
3834
-            $success = false;
3835
-        }
3836
-
3837
-        if ($success) {
3838
-            // made it here?  Stop dawdling then!!
3839
-            $success = $status === 'off'
3840
-                ? $this->_deactivate_messenger($this->_req_data['messenger'])
3841
-                : $this->_activate_messenger($this->_req_data['messenger']);
3842
-        }
3843
-
3844
-        $this->_template_args['success'] = $success;
3845
-
3846
-        // no special instructions so let's just do the json return (which should automatically do all the special stuff).
3847
-        $this->_return_json();
3848
-    }
3849
-
3850
-
3851
-    /**
3852
-     * used by ajax from the messages settings page to activate|deactivate a message type
3853
-     *
3854
-     * @throws DomainException
3855
-     * @throws EE_Error
3856
-     * @throws ReflectionException
3857
-     * @throws InvalidDataTypeException
3858
-     * @throws InvalidInterfaceException
3859
-     * @throws InvalidArgumentException
3860
-     */
3861
-    public function activate_mt_toggle()
3862
-    {
3863
-        $success = true;
3864
-        $this->_prep_default_response_for_messenger_or_message_type_toggle();
3865
-
3866
-        // let's make sure we have the necessary data
3867
-        if (! isset($this->_req_data['message_type'])) {
3868
-            EE_Error::add_error(
3869
-                esc_html__('Message Type name needed to toggle activation. None given', 'event_espresso'),
3870
-                __FILE__,
3871
-                __FUNCTION__,
3872
-                __LINE__
3873
-            );
3874
-            $success = false;
3875
-        }
3876
-
3877
-        if (! isset($this->_req_data['messenger'])) {
3878
-            EE_Error::add_error(
3879
-                esc_html__('Messenger name needed to toggle activation. None given', 'event_espresso'),
3880
-                __FILE__,
3881
-                __FUNCTION__,
3882
-                __LINE__
3883
-            );
3884
-            $success = false;
3885
-        }
3886
-
3887
-        if (! isset($this->_req_data['status'])) {
3888
-            EE_Error::add_error(
3889
-                esc_html__(
3890
-                    'Messenger status needed to know whether activation or deactivation is happening. No status is given',
3891
-                    'event_espresso'
3892
-                ),
3893
-                __FILE__,
3894
-                __FUNCTION__,
3895
-                __LINE__
3896
-            );
3897
-            $success = false;
3898
-        }
3899
-
3900
-
3901
-        // do check to verify we have a valid status.
3902
-        $status = $this->_req_data['status'];
3903
-
3904
-        if ($status !== 'activate' && $status !== 'deactivate') {
3905
-            EE_Error::add_error(
3906
-                sprintf(
3907
-                    esc_html__('The given status (%s) is not valid. Must be "active" or "inactive"', 'event_espresso'),
3908
-                    $this->_req_data['status']
3909
-                ),
3910
-                __FILE__,
3911
-                __FUNCTION__,
3912
-                __LINE__
3913
-            );
3914
-            $success = false;
3915
-        }
3916
-
3917
-
3918
-        // do a nonce check here since we're not arriving via a normal route
3919
-        $nonce = isset($this->_req_data['mt_nonce']) ? sanitize_text_field($this->_req_data['mt_nonce']) : '';
3920
-        $nonce_ref = $this->_req_data['message_type'] . '_nonce';
3921
-
3922
-        $this->_verify_nonce($nonce, $nonce_ref);
3923
-
3924
-        if ($success) {
3925
-            // made it here? um, what are you waiting for then?
3926
-            $success = $status === 'deactivate'
3927
-                ? $this->_deactivate_message_type_for_messenger(
3928
-                    $this->_req_data['messenger'],
3929
-                    $this->_req_data['message_type']
3930
-                )
3931
-                : $this->_activate_message_type_for_messenger(
3932
-                    $this->_req_data['messenger'],
3933
-                    $this->_req_data['message_type']
3934
-                );
3935
-        }
3936
-
3937
-        $this->_template_args['success'] = $success;
3938
-        $this->_return_json();
3939
-    }
3940
-
3941
-
3942
-    /**
3943
-     * Takes care of processing activating a messenger and preparing the appropriate response.
3944
-     *
3945
-     * @param string $messenger_name The name of the messenger being activated
3946
-     * @return bool
3947
-     * @throws DomainException
3948
-     * @throws EE_Error
3949
-     * @throws InvalidArgumentException
3950
-     * @throws ReflectionException
3951
-     * @throws InvalidDataTypeException
3952
-     * @throws InvalidInterfaceException
3953
-     */
3954
-    protected function _activate_messenger($messenger_name)
3955
-    {
3956
-        /** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
3957
-        $active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
3958
-        $message_types_to_activate = $active_messenger instanceof EE_Messenger
3959
-            ? $active_messenger->get_default_message_types()
3960
-            : array();
3961
-
3962
-        // ensure is active
3963
-        $this->_message_resource_manager->activate_messenger($messenger_name, $message_types_to_activate);
3964
-
3965
-        // set response_data for reload
3966
-        foreach ($message_types_to_activate as $message_type_name) {
3967
-            /** @var EE_message_type $message_type */
3968
-            $message_type = $this->_message_resource_manager->get_message_type($message_type_name);
3969
-            if ($this->_message_resource_manager->is_message_type_active_for_messenger(
3970
-                $messenger_name,
3971
-                $message_type_name
3972
-            )
3973
-                && $message_type instanceof EE_message_type
3974
-            ) {
3975
-                $this->_template_args['data']['active_mts'][] = $message_type_name;
3976
-                if ($message_type->get_admin_settings_fields()) {
3977
-                    $this->_template_args['data']['mt_reload'][] = $message_type_name;
3978
-                }
3979
-            }
3980
-        }
3981
-
3982
-        // add success message for activating messenger
3983
-        return $this->_setup_response_message_for_activating_messenger_with_message_types($active_messenger);
3984
-    }
3985
-
3986
-
3987
-    /**
3988
-     * Takes care of processing deactivating a messenger and preparing the appropriate response.
3989
-     *
3990
-     * @param string $messenger_name The name of the messenger being activated
3991
-     * @return bool
3992
-     * @throws DomainException
3993
-     * @throws EE_Error
3994
-     * @throws InvalidArgumentException
3995
-     * @throws ReflectionException
3996
-     * @throws InvalidDataTypeException
3997
-     * @throws InvalidInterfaceException
3998
-     */
3999
-    protected function _deactivate_messenger($messenger_name)
4000
-    {
4001
-        /** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
4002
-        $active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
4003
-        $this->_message_resource_manager->deactivate_messenger($messenger_name);
4004
-
4005
-        return $this->_setup_response_message_for_deactivating_messenger_with_message_types($active_messenger);
4006
-    }
4007
-
4008
-
4009
-    /**
4010
-     * Takes care of processing activating a message type for a messenger and preparing the appropriate response.
4011
-     *
4012
-     * @param string $messenger_name    The name of the messenger the message type is being activated for.
4013
-     * @param string $message_type_name The name of the message type being activated for the messenger
4014
-     * @return bool
4015
-     * @throws DomainException
4016
-     * @throws EE_Error
4017
-     * @throws InvalidArgumentException
4018
-     * @throws ReflectionException
4019
-     * @throws InvalidDataTypeException
4020
-     * @throws InvalidInterfaceException
4021
-     */
4022
-    protected function _activate_message_type_for_messenger($messenger_name, $message_type_name)
4023
-    {
4024
-        /** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
4025
-        $active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
4026
-        /** @var EE_message_type $message_type_to_activate This will be present because it can't be toggled if it isn't */
4027
-        $message_type_to_activate = $this->_message_resource_manager->get_message_type($message_type_name);
4028
-
4029
-        // ensure is active
4030
-        $this->_message_resource_manager->activate_messenger($messenger_name, $message_type_name);
4031
-
4032
-        // set response for load
4033
-        if ($this->_message_resource_manager->is_message_type_active_for_messenger(
4034
-            $messenger_name,
4035
-            $message_type_name
4036
-        )
4037
-        ) {
4038
-            $this->_template_args['data']['active_mts'][] = $message_type_name;
4039
-            if ($message_type_to_activate->get_admin_settings_fields()) {
4040
-                $this->_template_args['data']['mt_reload'][] = $message_type_name;
4041
-            }
4042
-        }
4043
-
4044
-        return $this->_setup_response_message_for_activating_messenger_with_message_types(
4045
-            $active_messenger,
4046
-            $message_type_to_activate
4047
-        );
4048
-    }
4049
-
4050
-
4051
-    /**
4052
-     * Takes care of processing deactivating a message type for a messenger and preparing the appropriate response.
4053
-     *
4054
-     * @param string $messenger_name    The name of the messenger the message type is being deactivated for.
4055
-     * @param string $message_type_name The name of the message type being deactivated for the messenger
4056
-     * @return bool
4057
-     * @throws DomainException
4058
-     * @throws EE_Error
4059
-     * @throws InvalidArgumentException
4060
-     * @throws ReflectionException
4061
-     * @throws InvalidDataTypeException
4062
-     * @throws InvalidInterfaceException
4063
-     */
4064
-    protected function _deactivate_message_type_for_messenger($messenger_name, $message_type_name)
4065
-    {
4066
-        /** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
4067
-        $active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
4068
-        /** @var EE_message_type $message_type_to_activate This will be present because it can't be toggled if it isn't */
4069
-        $message_type_to_deactivate = $this->_message_resource_manager->get_message_type($message_type_name);
4070
-        $this->_message_resource_manager->deactivate_message_type_for_messenger($message_type_name, $messenger_name);
4071
-
4072
-        return $this->_setup_response_message_for_deactivating_messenger_with_message_types(
4073
-            $active_messenger,
4074
-            $message_type_to_deactivate
4075
-        );
4076
-    }
4077
-
4078
-
4079
-    /**
4080
-     * This just initializes the defaults for activating messenger and message type responses.
4081
-     */
4082
-    protected function _prep_default_response_for_messenger_or_message_type_toggle()
4083
-    {
4084
-        $this->_template_args['data']['active_mts'] = array();
4085
-        $this->_template_args['data']['mt_reload'] = array();
4086
-    }
4087
-
4088
-
4089
-    /**
4090
-     * Setup appropriate response for activating a messenger and/or message types
4091
-     *
4092
-     * @param EE_messenger         $messenger
4093
-     * @param EE_message_type|null $message_type
4094
-     * @return bool
4095
-     * @throws DomainException
4096
-     * @throws EE_Error
4097
-     * @throws InvalidArgumentException
4098
-     * @throws ReflectionException
4099
-     * @throws InvalidDataTypeException
4100
-     * @throws InvalidInterfaceException
4101
-     */
4102
-    protected function _setup_response_message_for_activating_messenger_with_message_types(
4103
-        $messenger,
4104
-        EE_Message_Type $message_type = null
4105
-    ) {
4106
-        // if $messenger isn't a valid messenger object then get out.
4107
-        if (! $messenger instanceof EE_Messenger) {
4108
-            EE_Error::add_error(
4109
-                esc_html__('The messenger being activated is not a valid messenger', 'event_espresso'),
4110
-                __FILE__,
4111
-                __FUNCTION__,
4112
-                __LINE__
4113
-            );
4114
-
4115
-            return false;
4116
-        }
4117
-        // activated
4118
-        if ($this->_template_args['data']['active_mts']) {
4119
-            EE_Error::overwrite_success();
4120
-            // activated a message type with the messenger
4121
-            if ($message_type instanceof EE_message_type) {
4122
-                EE_Error::add_success(
4123
-                    sprintf(
4124
-                        esc_html__(
4125
-                            '%s message type has been successfully activated with the %s messenger',
4126
-                            'event_espresso'
4127
-                        ),
4128
-                        ucwords($message_type->label['singular']),
4129
-                        ucwords($messenger->label['singular'])
4130
-                    )
4131
-                );
4132
-
4133
-                // if message type was invoice then let's make sure we activate the invoice payment method.
4134
-                if ($message_type->name === 'invoice') {
4135
-                    EE_Registry::instance()->load_lib('Payment_Method_Manager');
4136
-                    $pm = EE_Payment_Method_Manager::instance()->activate_a_payment_method_of_type('Invoice');
4137
-                    if ($pm instanceof EE_Payment_Method) {
4138
-                        EE_Error::add_attention(
4139
-                            esc_html__(
4140
-                                'Activating the invoice message type also automatically activates the invoice payment method.  If you do not wish the invoice payment method to be active, or to change its settings, visit the payment method admin page.',
4141
-                                'event_espresso'
4142
-                            )
4143
-                        );
4144
-                    }
4145
-                }
4146
-                // just toggles the entire messenger
4147
-            } else {
4148
-                EE_Error::add_success(
4149
-                    sprintf(
4150
-                        esc_html__('%s messenger has been successfully activated', 'event_espresso'),
4151
-                        ucwords($messenger->label['singular'])
4152
-                    )
4153
-                );
4154
-            }
4155
-
4156
-            return true;
4157
-
4158
-            // possible error condition. This will happen when our active_mts data is empty because it is validated for actual active
4159
-            // message types after the activation process.  However its possible some messengers don't HAVE any default_message_types
4160
-            // in which case we just give a success message for the messenger being successfully activated.
4161
-        } else {
4162
-            if (! $messenger->get_default_message_types()) {
4163
-                // messenger doesn't have any default message types so still a success.
4164
-                EE_Error::add_success(
4165
-                    sprintf(
4166
-                        esc_html__('%s messenger was successfully activated.', 'event_espresso'),
4167
-                        ucwords($messenger->label['singular'])
4168
-                    )
4169
-                );
4170
-
4171
-                return true;
4172
-            } else {
4173
-                EE_Error::add_error(
4174
-                    $message_type instanceof EE_message_type
4175
-                        ? sprintf(
4176
-                            esc_html__(
4177
-                                '%s message type was not successfully activated with the %s messenger',
4178
-                                'event_espresso'
4179
-                            ),
4180
-                            ucwords($message_type->label['singular']),
4181
-                            ucwords($messenger->label['singular'])
4182
-                        )
4183
-                        : sprintf(
4184
-                            esc_html__('%s messenger was not successfully activated', 'event_espresso'),
4185
-                            ucwords($messenger->label['singular'])
4186
-                        ),
4187
-                    __FILE__,
4188
-                    __FUNCTION__,
4189
-                    __LINE__
4190
-                );
4191
-
4192
-                return false;
4193
-            }
4194
-        }
4195
-    }
4196
-
4197
-
4198
-    /**
4199
-     * This sets up the appropriate response for deactivating a messenger and/or message type.
4200
-     *
4201
-     * @param EE_messenger         $messenger
4202
-     * @param EE_message_type|null $message_type
4203
-     * @return bool
4204
-     * @throws DomainException
4205
-     * @throws EE_Error
4206
-     * @throws InvalidArgumentException
4207
-     * @throws ReflectionException
4208
-     * @throws InvalidDataTypeException
4209
-     * @throws InvalidInterfaceException
4210
-     */
4211
-    protected function _setup_response_message_for_deactivating_messenger_with_message_types(
4212
-        $messenger,
4213
-        EE_message_type $message_type = null
4214
-    ) {
4215
-        EE_Error::overwrite_success();
4216
-
4217
-        // if $messenger isn't a valid messenger object then get out.
4218
-        if (! $messenger instanceof EE_Messenger) {
4219
-            EE_Error::add_error(
4220
-                esc_html__('The messenger being deactivated is not a valid messenger', 'event_espresso'),
4221
-                __FILE__,
4222
-                __FUNCTION__,
4223
-                __LINE__
4224
-            );
4225
-
4226
-            return false;
4227
-        }
4228
-
4229
-        if ($message_type instanceof EE_message_type) {
4230
-            $message_type_name = $message_type->name;
4231
-            EE_Error::add_success(
4232
-                sprintf(
4233
-                    esc_html__(
4234
-                        '%s message type has been successfully deactivated for the %s messenger.',
4235
-                        'event_espresso'
4236
-                    ),
4237
-                    ucwords($message_type->label['singular']),
4238
-                    ucwords($messenger->label['singular'])
4239
-                )
4240
-            );
4241
-        } else {
4242
-            $message_type_name = '';
4243
-            EE_Error::add_success(
4244
-                sprintf(
4245
-                    esc_html__('%s messenger has been successfully deactivated.', 'event_espresso'),
4246
-                    ucwords($messenger->label['singular'])
4247
-                )
4248
-            );
4249
-        }
4250
-
4251
-        // if messenger was html or message type was invoice then let's make sure we deactivate invoice payment method.
4252
-        if ($messenger->name === 'html' || $message_type_name === 'invoice') {
4253
-            EE_Registry::instance()->load_lib('Payment_Method_Manager');
4254
-            $count_updated = EE_Payment_Method_Manager::instance()->deactivate_payment_method('invoice');
4255
-            if ($count_updated > 0) {
4256
-                $msg = $message_type_name === 'invoice'
4257
-                    ? esc_html__(
4258
-                        'Deactivating the invoice message type also automatically deactivates the invoice payment method. In order for invoices to be generated the invoice message type must be active. If you completed this action by mistake, simply reactivate the invoice message type and then visit the payment methods admin page to reactivate the invoice payment method.',
4259
-                        'event_espresso'
4260
-                    )
4261
-                    : esc_html__(
4262
-                        'Deactivating the html messenger also automatically deactivates the invoice payment method.  In order for invoices to be generated the html messenger must be be active.  If you completed this action by mistake, simply reactivate the html messenger, then visit the payment methods admin page to reactivate the invoice payment method.',
4263
-                        'event_espresso'
4264
-                    );
4265
-                EE_Error::add_attention($msg);
4266
-            }
4267
-        }
4268
-
4269
-        return true;
4270
-    }
4271
-
4272
-
4273
-    /**
4274
-     * handles updating a message type form on messenger activation IF the message type has settings fields. (via ajax)
4275
-     *
4276
-     * @throws DomainException
4277
-     */
4278
-    public function update_mt_form()
4279
-    {
4280
-        if (! isset($this->_req_data['messenger']) || ! isset($this->_req_data['message_type'])) {
4281
-            EE_Error::add_error(
4282
-                esc_html__('Require message type or messenger to send an updated form', 'event_espresso'),
4283
-                __FILE__,
4284
-                __FUNCTION__,
4285
-                __LINE__
4286
-            );
4287
-            $this->_return_json();
4288
-        }
4289
-
4290
-        $message_types = $this->get_installed_message_types();
4291
-
4292
-        $message_type = $message_types[ $this->_req_data['message_type'] ];
4293
-        $messenger = $this->_message_resource_manager->get_active_messenger($this->_req_data['messenger']);
4294
-
4295
-        $content = $this->_message_type_settings_content(
4296
-            $message_type,
4297
-            $messenger,
4298
-            true
4299
-        );
4300
-        $this->_template_args['success'] = true;
4301
-        $this->_template_args['content'] = $content;
4302
-        $this->_return_json();
4303
-    }
4304
-
4305
-
4306
-    /**
4307
-     * this handles saving the settings for a messenger or message type
4308
-     *
4309
-     */
4310
-    public function save_settings()
4311
-    {
4312
-        if (! isset($this->_req_data['type'])) {
4313
-            EE_Error::add_error(
4314
-                esc_html__(
4315
-                    'Cannot save settings because type is unknown (messenger settings or messsage type settings?)',
4316
-                    'event_espresso'
4317
-                ),
4318
-                __FILE__,
4319
-                __FUNCTION__,
4320
-                __LINE__
4321
-            );
4322
-            $this->_template_args['error'] = true;
4323
-            $this->_return_json();
4324
-        }
4325
-
4326
-
4327
-        if ($this->_req_data['type'] === 'messenger') {
4328
-            // this should be an array.
4329
-            $settings = $this->_req_data['messenger_settings'];
4330
-            $messenger = $settings['messenger'];
4331
-            // let's setup the settings data
4332
-            foreach ($settings as $key => $value) {
4333
-                switch ($key) {
4334
-                    case 'messenger':
4335
-                        unset($settings['messenger']);
4336
-                        break;
4337
-                    case 'message_types':
4338
-                        unset($settings['message_types']);
4339
-                        break;
4340
-                    default:
4341
-                        $settings[ $key ] = $value;
4342
-                        break;
4343
-                }
4344
-            }
4345
-            $this->_message_resource_manager->add_settings_for_messenger($messenger, $settings);
4346
-        } elseif ($this->_req_data['type'] === 'message_type') {
4347
-            $settings = $this->_req_data['message_type_settings'];
4348
-            $messenger = $settings['messenger'];
4349
-            $message_type = $settings['message_type'];
4350
-
4351
-            foreach ($settings as $key => $value) {
4352
-                switch ($key) {
4353
-                    case 'messenger':
4354
-                        unset($settings['messenger']);
4355
-                        break;
4356
-                    case 'message_type':
4357
-                        unset($settings['message_type']);
4358
-                        break;
4359
-                    default:
4360
-                        $settings[ $key ] = $value;
4361
-                        break;
4362
-                }
4363
-            }
4364
-
4365
-            $this->_message_resource_manager->add_settings_for_message_type($messenger, $message_type, $settings);
4366
-        }
4367
-
4368
-        // okay we should have the data all setup.  Now we just update!
4369
-        $success = $this->_message_resource_manager->update_active_messengers_option();
4370
-
4371
-        if ($success) {
4372
-            EE_Error::add_success(__('Settings updated', 'event_espresso'));
4373
-        } else {
4374
-            EE_Error::add_error(
4375
-                esc_html__(
4376
-                    'Settings did not get updated',
4377
-                    'event_espresso'
4378
-                ),
4379
-                __FILE__,
4380
-                __FUNCTION__,
4381
-                __LINE__
4382
-            );
4383
-        }
4384
-
4385
-        $this->_template_args['success'] = $success;
4386
-        $this->_return_json();
4387
-    }
4388
-
4389
-
4390
-
4391
-
4392
-    /**  EE MESSAGE PROCESSING ACTIONS **/
4393
-
4394
-
4395
-    /**
4396
-     * This immediately generates any EE_Message ID's that are selected that are EEM_Message::status_incomplete
4397
-     * However, this does not send immediately, it just queues for sending.
4398
-     *
4399
-     * @since 4.9.0
4400
-     * @throws EE_Error
4401
-     * @throws InvalidDataTypeException
4402
-     * @throws InvalidInterfaceException
4403
-     * @throws InvalidArgumentException
4404
-     * @throws ReflectionException
4405
-     */
4406
-    protected function _generate_now()
4407
-    {
4408
-        EED_Messages::generate_now($this->_get_msg_ids_from_request());
4409
-        $this->_redirect_after_action(false, '', '', array(), true);
4410
-    }
4411
-
4412
-
4413
-    /**
4414
-     * This immediately generates AND sends any EE_Message's selected that are EEM_Message::status_incomplete or that
4415
-     * are EEM_Message::status_resend or EEM_Message::status_idle
4416
-     *
4417
-     * @since 4.9.0
4418
-     * @throws EE_Error
4419
-     * @throws InvalidDataTypeException
4420
-     * @throws InvalidInterfaceException
4421
-     * @throws InvalidArgumentException
4422
-     * @throws ReflectionException
4423
-     */
4424
-    protected function _generate_and_send_now()
4425
-    {
4426
-        EED_Messages::generate_and_send_now($this->_get_msg_ids_from_request());
4427
-        $this->_redirect_after_action(false, '', '', array(), true);
4428
-    }
4429
-
4430
-
4431
-    /**
4432
-     * This queues any EEM_Message::status_sent EE_Message ids in the request for resending.
4433
-     *
4434
-     * @since 4.9.0
4435
-     * @throws EE_Error
4436
-     * @throws InvalidDataTypeException
4437
-     * @throws InvalidInterfaceException
4438
-     * @throws InvalidArgumentException
4439
-     * @throws ReflectionException
4440
-     */
4441
-    protected function _queue_for_resending()
4442
-    {
4443
-        EED_Messages::queue_for_resending($this->_get_msg_ids_from_request());
4444
-        $this->_redirect_after_action(false, '', '', array(), true);
4445
-    }
4446
-
4447
-
4448
-    /**
4449
-     *  This sends immediately any EEM_Message::status_idle or EEM_Message::status_resend messages in the queue
4450
-     *
4451
-     * @since 4.9.0
4452
-     * @throws EE_Error
4453
-     * @throws InvalidDataTypeException
4454
-     * @throws InvalidInterfaceException
4455
-     * @throws InvalidArgumentException
4456
-     * @throws ReflectionException
4457
-     */
4458
-    protected function _send_now()
4459
-    {
4460
-        EED_Messages::send_now($this->_get_msg_ids_from_request());
4461
-        $this->_redirect_after_action(false, '', '', array(), true);
4462
-    }
4463
-
4464
-
4465
-    /**
4466
-     * Deletes EE_messages for IDs in the request.
4467
-     *
4468
-     * @since 4.9.0
4469
-     * @throws EE_Error
4470
-     * @throws InvalidDataTypeException
4471
-     * @throws InvalidInterfaceException
4472
-     * @throws InvalidArgumentException
4473
-     */
4474
-    protected function _delete_ee_messages()
4475
-    {
4476
-        $msg_ids = $this->_get_msg_ids_from_request();
4477
-        $deleted_count = 0;
4478
-        foreach ($msg_ids as $msg_id) {
4479
-            if (EEM_Message::instance()->delete_by_ID($msg_id)) {
4480
-                $deleted_count++;
4481
-            }
4482
-        }
4483
-        if ($deleted_count) {
4484
-            EE_Error::add_success(
4485
-                esc_html(
4486
-                    _n(
4487
-                        'Message successfully deleted',
4488
-                        'Messages successfully deleted',
4489
-                        $deleted_count,
4490
-                        'event_espresso'
4491
-                    )
4492
-                )
4493
-            );
4494
-            $this->_redirect_after_action(
4495
-                false,
4496
-                '',
4497
-                '',
4498
-                array(),
4499
-                true
4500
-            );
4501
-        } else {
4502
-            EE_Error::add_error(
4503
-                _n('The message was not deleted.', 'The messages were not deleted', count($msg_ids), 'event_espresso'),
4504
-                __FILE__,
4505
-                __FUNCTION__,
4506
-                __LINE__
4507
-            );
4508
-            $this->_redirect_after_action(false, '', '', array(), true);
4509
-        }
4510
-    }
4511
-
4512
-
4513
-    /**
4514
-     *  This looks for 'MSG_ID' key in the request and returns an array of MSG_ID's if present.
4515
-     *
4516
-     * @since 4.9.0
4517
-     * @return array
4518
-     */
4519
-    protected function _get_msg_ids_from_request()
4520
-    {
4521
-        if (! isset($this->_req_data['MSG_ID'])) {
4522
-            return array();
4523
-        }
4524
-
4525
-        return is_array($this->_req_data['MSG_ID'])
4526
-            ? array_keys($this->_req_data['MSG_ID'])
4527
-            : array($this->_req_data['MSG_ID']);
4528
-    }
2612
+		$output = ob_get_contents();
2613
+		ob_clean();
2614
+		$this->_context_switcher = $output;
2615
+	}
2616
+
2617
+
2618
+	/**
2619
+	 * utility for sanitizing new values coming in.
2620
+	 * Note: this is only used when updating a context.
2621
+	 *
2622
+	 * @access protected
2623
+	 *
2624
+	 * @param int $index This helps us know which template field to select from the request array.
2625
+	 *
2626
+	 * @return array
2627
+	 */
2628
+	protected function _set_message_template_column_values($index)
2629
+	{
2630
+		if (is_array($this->_req_data['MTP_template_fields'][ $index ]['content'])) {
2631
+			foreach ($this->_req_data['MTP_template_fields'][ $index ]['content'] as $field => $value) {
2632
+				$this->_req_data['MTP_template_fields'][ $index ]['content'][ $field ] = $value;
2633
+			}
2634
+		}
2635
+
2636
+
2637
+		$set_column_values = array(
2638
+			'MTP_ID'             => absint($this->_req_data['MTP_template_fields'][ $index ]['MTP_ID']),
2639
+			'GRP_ID'             => absint($this->_req_data['GRP_ID']),
2640
+			'MTP_user_id'        => absint($this->_req_data['MTP_user_id']),
2641
+			'MTP_messenger'      => strtolower($this->_req_data['MTP_messenger']),
2642
+			'MTP_message_type'   => strtolower($this->_req_data['MTP_message_type']),
2643
+			'MTP_template_field' => strtolower($this->_req_data['MTP_template_fields'][ $index ]['name']),
2644
+			'MTP_context'        => strtolower($this->_req_data['MTP_context']),
2645
+			'MTP_content'        => $this->_req_data['MTP_template_fields'][ $index ]['content'],
2646
+			'MTP_is_global'      => isset($this->_req_data['MTP_is_global'])
2647
+				? absint($this->_req_data['MTP_is_global'])
2648
+				: 0,
2649
+			'MTP_is_override'    => isset($this->_req_data['MTP_is_override'])
2650
+				? absint($this->_req_data['MTP_is_override'])
2651
+				: 0,
2652
+			'MTP_deleted'        => absint($this->_req_data['MTP_deleted']),
2653
+			'MTP_is_active'      => absint($this->_req_data['MTP_is_active']),
2654
+		);
2655
+
2656
+
2657
+		return $set_column_values;
2658
+	}
2659
+
2660
+
2661
+	protected function _insert_or_update_message_template($new = false)
2662
+	{
2663
+
2664
+		do_action('AHEE_log', __FILE__, __FUNCTION__, '');
2665
+		$success = 0;
2666
+		$override = false;
2667
+
2668
+		// setup notices description
2669
+		$messenger_slug = ! empty($this->_req_data['MTP_messenger']) ? $this->_req_data['MTP_messenger'] : '';
2670
+
2671
+		// need the message type and messenger objects to be able to use the labels for the notices
2672
+		$messenger_object = $this->_message_resource_manager->get_messenger($messenger_slug);
2673
+		$messenger_label = $messenger_object instanceof EE_messenger
2674
+			? ucwords($messenger_object->label['singular'])
2675
+			: '';
2676
+
2677
+		$message_type_slug = ! empty($this->_req_data['MTP_message_type'])
2678
+			? $this->_req_data['MTP_message_type']
2679
+			: '';
2680
+		$message_type_object = $this->_message_resource_manager->get_message_type($message_type_slug);
2681
+
2682
+		$message_type_label = $message_type_object instanceof EE_message_type
2683
+			? ucwords($message_type_object->label['singular'])
2684
+			: '';
2685
+
2686
+		$context_slug = ! empty($this->_req_data['MTP_context'])
2687
+			? $this->_req_data['MTP_context']
2688
+			: '';
2689
+		$context = ucwords(str_replace('_', ' ', $context_slug));
2690
+
2691
+		$item_desc = $messenger_label && $message_type_label
2692
+			? $messenger_label . ' ' . $message_type_label . ' ' . $context . ' '
2693
+			: '';
2694
+		$item_desc .= 'Message Template';
2695
+		$query_args = array();
2696
+		$edit_array = array();
2697
+		$action_desc = '';
2698
+
2699
+		// if this is "new" then we need to generate the default contexts for the selected messenger/message_type for
2700
+		// user to edit.
2701
+		if ($new) {
2702
+			$GRP_ID = ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : 0;
2703
+			if ($edit_array = $this->_generate_new_templates($messenger_slug, $message_type_slug, $GRP_ID)) {
2704
+				if (empty($edit_array)) {
2705
+					$success = 0;
2706
+				} else {
2707
+					$success = 1;
2708
+					$edit_array = $edit_array[0];
2709
+					$query_args = array(
2710
+						'id'      => $edit_array['GRP_ID'],
2711
+						'context' => $edit_array['MTP_context'],
2712
+						'action'  => 'edit_message_template',
2713
+					);
2714
+				}
2715
+			}
2716
+			$action_desc = 'created';
2717
+		} else {
2718
+			$MTPG = EEM_Message_Template_Group::instance();
2719
+			$MTP = EEM_Message_Template::instance();
2720
+
2721
+
2722
+			// run update for each template field in displayed context
2723
+			if (! isset($this->_req_data['MTP_template_fields']) && empty($this->_req_data['MTP_template_fields'])) {
2724
+				EE_Error::add_error(
2725
+					esc_html__(
2726
+						'There was a problem saving the template fields from the form because I didn\'t receive any actual template field data.',
2727
+						'event_espresso'
2728
+					),
2729
+					__FILE__,
2730
+					__FUNCTION__,
2731
+					__LINE__
2732
+				);
2733
+				$success = 0;
2734
+			} else {
2735
+				// first validate all fields!
2736
+				// this filter allows client code to add its own validation to the template fields as well.
2737
+				// returning an empty array means everything passed validation.
2738
+				// errors in validation should be represented in an array with the following shape:
2739
+				// array(
2740
+				//   'fieldname' => array(
2741
+				//          'msg' => 'error message'
2742
+				//          'value' => 'value for field producing error'
2743
+				// )
2744
+				$custom_validation = (array) apply_filters(
2745
+					'FHEE__Messages_Admin_Page___insert_or_update_message_template__validates',
2746
+					array(),
2747
+					$this->_req_data['MTP_template_fields'],
2748
+					$context_slug,
2749
+					$messenger_slug,
2750
+					$message_type_slug
2751
+				);
2752
+
2753
+				$system_validation = $MTPG->validate(
2754
+					$this->_req_data['MTP_template_fields'],
2755
+					$context_slug,
2756
+					$messenger_slug,
2757
+					$message_type_slug
2758
+				);
2759
+
2760
+				$system_validation = ! is_array($system_validation) && $system_validation ? array()
2761
+					: $system_validation;
2762
+				$validates = array_merge($custom_validation, $system_validation);
2763
+
2764
+				// if $validate returned error messages (i.e. is_array()) then we need to process them and setup an
2765
+				// appropriate response. HMM, dang this isn't correct, $validates will ALWAYS be an array.
2766
+				//  WE need to make sure there is no actual error messages in validates.
2767
+				if (is_array($validates) && ! empty($validates)) {
2768
+					// add the transient so when the form loads we know which fields to highlight
2769
+					$this->_add_transient('edit_message_template', $validates);
2770
+
2771
+					$success = 0;
2772
+
2773
+					// setup notices
2774
+					foreach ($validates as $field => $error) {
2775
+						if (isset($error['msg'])) {
2776
+							EE_Error::add_error($error['msg'], __FILE__, __FUNCTION__, __LINE__);
2777
+						}
2778
+					}
2779
+				} else {
2780
+					$set_column_values = array();
2781
+					foreach ($this->_req_data['MTP_template_fields'] as $template_field => $content) {
2782
+						$set_column_values = $this->_set_message_template_column_values($template_field);
2783
+
2784
+						$where_cols_n_values = array(
2785
+							'MTP_ID' => $this->_req_data['MTP_template_fields'][ $template_field ]['MTP_ID'],
2786
+						);
2787
+						// if they aren't allowed to use all JS, restrict them to just posty-y tags
2788
+						if (! current_user_can('unfiltered_html')) {
2789
+							if (is_array($set_column_values['MTP_content'])) {
2790
+								foreach ($set_column_values['MTP_content'] as $key => $value) {
2791
+									// remove slashes so wp_kses works properly (its wp_kses_stripslashes() function
2792
+									// only removes slashes from double-quotes, so attributes using single quotes always
2793
+									// appear invalid.) But currently the models expect slashed data, so after wp_kses
2794
+									// runs we need to re-slash the data. Sheesh. See
2795
+									// https://events.codebasehq.com/projects/event-espresso/tickets/11211#update-47321587
2796
+									$set_column_values['MTP_content'][ $key ] = addslashes(
2797
+										wp_kses(
2798
+											stripslashes($value),
2799
+											wp_kses_allowed_html('post')
2800
+										)
2801
+									);
2802
+								}
2803
+							} else {
2804
+								$set_column_values['MTP_content'] = wp_kses(
2805
+									$set_column_values['MTP_content'],
2806
+									wp_kses_allowed_html('post')
2807
+								);
2808
+							}
2809
+						}
2810
+						$message_template_fields = array(
2811
+							'GRP_ID'             => $set_column_values['GRP_ID'],
2812
+							'MTP_template_field' => $set_column_values['MTP_template_field'],
2813
+							'MTP_context'        => $set_column_values['MTP_context'],
2814
+							'MTP_content'        => $set_column_values['MTP_content'],
2815
+						);
2816
+						if ($updated = $MTP->update($message_template_fields, array($where_cols_n_values))) {
2817
+							if ($updated === false) {
2818
+								EE_Error::add_error(
2819
+									sprintf(
2820
+										esc_html__('%s field was NOT updated for some reason', 'event_espresso'),
2821
+										$template_field
2822
+									),
2823
+									__FILE__,
2824
+									__FUNCTION__,
2825
+									__LINE__
2826
+								);
2827
+							} else {
2828
+								$success = 1;
2829
+							}
2830
+						} else {
2831
+							// only do this logic if we don't have a MTP_ID for this field
2832
+							if (empty($this->_req_data['MTP_template_fields'][ $template_field ]['MTP_ID'])) {
2833
+								// this has already been through the template field validator and sanitized, so it will be
2834
+								// safe to insert this field.  Why insert?  This typically happens when we introduce a new
2835
+								// message template field in a messenger/message type and existing users don't have the
2836
+								// default setup for it.
2837
+								// @link https://events.codebasehq.com/projects/event-espresso/tickets/9465
2838
+								$updated = $MTP->insert($message_template_fields);
2839
+								if (! $updated || is_wp_error($updated)) {
2840
+									EE_Error::add_error(
2841
+										sprintf(
2842
+											esc_html__('%s field could not be updated.', 'event_espresso'),
2843
+											$template_field
2844
+										),
2845
+										__FILE__,
2846
+										__FUNCTION__,
2847
+										__LINE__
2848
+									);
2849
+									$success = 0;
2850
+								} else {
2851
+									$success = 1;
2852
+								}
2853
+							}
2854
+						}
2855
+						$action_desc = 'updated';
2856
+					}
2857
+
2858
+					// we can use the last set_column_values for the MTPG update (because its the same for all of these specific MTPs)
2859
+					$mtpg_fields = array(
2860
+						'MTP_user_id'      => $set_column_values['MTP_user_id'],
2861
+						'MTP_messenger'    => $set_column_values['MTP_messenger'],
2862
+						'MTP_message_type' => $set_column_values['MTP_message_type'],
2863
+						'MTP_is_global'    => $set_column_values['MTP_is_global'],
2864
+						'MTP_is_override'  => $set_column_values['MTP_is_override'],
2865
+						'MTP_deleted'      => $set_column_values['MTP_deleted'],
2866
+						'MTP_is_active'    => $set_column_values['MTP_is_active'],
2867
+						'MTP_name'         => ! empty($this->_req_data['ee_msg_non_global_fields']['MTP_name'])
2868
+							? $this->_req_data['ee_msg_non_global_fields']['MTP_name']
2869
+							: '',
2870
+						'MTP_description'  => ! empty($this->_req_data['ee_msg_non_global_fields']['MTP_description'])
2871
+							? $this->_req_data['ee_msg_non_global_fields']['MTP_description']
2872
+							: '',
2873
+					);
2874
+
2875
+					$mtpg_where = array('GRP_ID' => $set_column_values['GRP_ID']);
2876
+					$updated = $MTPG->update($mtpg_fields, array($mtpg_where));
2877
+
2878
+					if ($updated === false) {
2879
+						EE_Error::add_error(
2880
+							sprintf(
2881
+								esc_html__(
2882
+									'The Message Template Group (%d) was NOT updated for some reason',
2883
+									'event_espresso'
2884
+								),
2885
+								$set_column_values['GRP_ID']
2886
+							),
2887
+							__FILE__,
2888
+							__FUNCTION__,
2889
+							__LINE__
2890
+						);
2891
+					} else {
2892
+						// k now we need to ensure the template_pack and template_variation fields are set.
2893
+						$template_pack = ! empty($this->_req_data['MTP_template_pack'])
2894
+							? $this->_req_data['MTP_template_pack']
2895
+							: 'default';
2896
+
2897
+						$template_variation = ! empty($this->_req_data['MTP_template_variation'])
2898
+							? $this->_req_data['MTP_template_variation']
2899
+							: 'default';
2900
+
2901
+						$mtpg_obj = $MTPG->get_one_by_ID($set_column_values['GRP_ID']);
2902
+						if ($mtpg_obj instanceof EE_Message_Template_Group) {
2903
+							$mtpg_obj->set_template_pack_name($template_pack);
2904
+							$mtpg_obj->set_template_pack_variation($template_variation);
2905
+						}
2906
+						$success = 1;
2907
+					}
2908
+				}
2909
+			}
2910
+		}
2911
+
2912
+		// we return things differently if doing ajax
2913
+		if (defined('DOING_AJAX') && DOING_AJAX) {
2914
+			$this->_template_args['success'] = $success;
2915
+			$this->_template_args['error'] = ! $success ? true : false;
2916
+			$this->_template_args['content'] = '';
2917
+			$this->_template_args['data'] = array(
2918
+				'grpID'        => $edit_array['GRP_ID'],
2919
+				'templateName' => $edit_array['template_name'],
2920
+			);
2921
+			if ($success) {
2922
+				EE_Error::overwrite_success();
2923
+				EE_Error::add_success(
2924
+					esc_html__(
2925
+						'The new template has been created and automatically selected for this event.  You can edit the new template by clicking the edit button.  Note before this template is assigned to this event, the event must be saved.',
2926
+						'event_espresso'
2927
+					)
2928
+				);
2929
+			}
2930
+
2931
+			$this->_return_json();
2932
+		}
2933
+
2934
+
2935
+		// was a test send triggered?
2936
+		if (isset($this->_req_data['test_button'])) {
2937
+			EE_Error::overwrite_success();
2938
+			$this->_do_test_send($context_slug, $messenger_slug, $message_type_slug);
2939
+			$override = true;
2940
+		}
2941
+
2942
+		if (empty($query_args)) {
2943
+			$query_args = array(
2944
+				'id'      => $this->_req_data['GRP_ID'],
2945
+				'context' => $context_slug,
2946
+				'action'  => 'edit_message_template',
2947
+			);
2948
+		}
2949
+
2950
+		$this->_redirect_after_action($success, $item_desc, $action_desc, $query_args, $override);
2951
+	}
2952
+
2953
+
2954
+	/**
2955
+	 * processes a test send request to do an actual messenger delivery test for the given message template being tested
2956
+	 *
2957
+	 * @param  string $context      what context being tested
2958
+	 * @param  string $messenger    messenger being tested
2959
+	 * @param  string $message_type message type being tested
2960
+	 * @throws EE_Error
2961
+	 * @throws InvalidArgumentException
2962
+	 * @throws InvalidDataTypeException
2963
+	 * @throws InvalidInterfaceException
2964
+	 */
2965
+	protected function _do_test_send($context, $messenger, $message_type)
2966
+	{
2967
+		// set things up for preview
2968
+		$this->_req_data['messenger'] = $messenger;
2969
+		$this->_req_data['message_type'] = $message_type;
2970
+		$this->_req_data['context'] = $context;
2971
+		$this->_req_data['GRP_ID'] = isset($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : '';
2972
+		$active_messenger = $this->_message_resource_manager->get_active_messenger($messenger);
2973
+
2974
+		// let's save any existing fields that might be required by the messenger
2975
+		if (isset($this->_req_data['test_settings_fld'])
2976
+			&& $active_messenger instanceof EE_messenger
2977
+			&& apply_filters(
2978
+				'FHEE__Messages_Admin_Page__do_test_send__set_existing_test_settings',
2979
+				true,
2980
+				$this->_req_data['test_settings_fld'],
2981
+				$active_messenger
2982
+			)
2983
+		) {
2984
+			$active_messenger->set_existing_test_settings($this->_req_data['test_settings_fld']);
2985
+		}
2986
+
2987
+		/**
2988
+		 * Use filter to add additional controls on whether message can send or not
2989
+		 */
2990
+		if (apply_filters(
2991
+			'FHEE__Messages_Admin_Page__do_test_send__can_send',
2992
+			true,
2993
+			$context,
2994
+			$this->_req_data,
2995
+			$messenger,
2996
+			$message_type
2997
+		)) {
2998
+			$success = $this->_preview_message(true);
2999
+			if ($success) {
3000
+				EE_Error::add_success(__('Test message sent', 'event_espresso'));
3001
+			} else {
3002
+				EE_Error::add_error(
3003
+					esc_html__('The test message was not sent', 'event_espresso'),
3004
+					__FILE__,
3005
+					__FUNCTION__,
3006
+					__LINE__
3007
+				);
3008
+			}
3009
+		}
3010
+	}
3011
+
3012
+
3013
+	/**
3014
+	 * _generate_new_templates
3015
+	 * This will handle the messenger, message_type selection when "adding a new custom template" for an event and will
3016
+	 * automatically create the defaults for the event.  The user would then be redirected to edit the default context
3017
+	 * for the event.
3018
+	 *
3019
+	 *
3020
+	 * @param  string $messenger     the messenger we are generating templates for
3021
+	 * @param array   $message_types array of message types that the templates are generated for.
3022
+	 * @param int     $GRP_ID        If this is a custom template being generated then a GRP_ID needs to be included to
3023
+	 *                               indicate the message_template_group being used as the base.
3024
+	 *
3025
+	 * @param bool    $global
3026
+	 *
3027
+	 * @return array|bool array of data required for the redirect to the correct edit page or bool if
3028
+	 *                               encountering problems.
3029
+	 * @throws EE_Error
3030
+	 */
3031
+	protected function _generate_new_templates($messenger, $message_types, $GRP_ID = 0, $global = false)
3032
+	{
3033
+
3034
+		// if no $message_types are given then that's okay... this may be a messenger that just adds shortcodes, so we
3035
+		// just don't generate any templates.
3036
+		if (empty($message_types)) {
3037
+			return true;
3038
+		}
3039
+
3040
+		return EEH_MSG_Template::generate_new_templates($messenger, $message_types, $GRP_ID, $global);
3041
+	}
3042
+
3043
+
3044
+	/**
3045
+	 * [_trash_or_restore_message_template]
3046
+	 *
3047
+	 * @param  boolean $trash whether to move an item to trash/restore (TRUE) or restore it (FALSE)
3048
+	 * @param boolean  $all   whether this is going to trash/restore all contexts within a template group (TRUE) OR just
3049
+	 *                        an individual context (FALSE).
3050
+	 * @return void
3051
+	 * @throws EE_Error
3052
+	 * @throws InvalidArgumentException
3053
+	 * @throws InvalidDataTypeException
3054
+	 * @throws InvalidInterfaceException
3055
+	 */
3056
+	protected function _trash_or_restore_message_template($trash = true, $all = false)
3057
+	{
3058
+		do_action('AHEE_log', __FILE__, __FUNCTION__, '');
3059
+		$MTP = EEM_Message_Template_Group::instance();
3060
+
3061
+		$success = 1;
3062
+
3063
+		// incoming GRP_IDs
3064
+		if ($all) {
3065
+			// Checkboxes
3066
+			if (! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3067
+				// if array has more than one element then success message should be plural.
3068
+				// todo: what about nonce?
3069
+				$success = count($this->_req_data['checkbox']) > 1 ? 2 : 1;
3070
+
3071
+				// cycle through checkboxes
3072
+				while (list($GRP_ID, $value) = each($this->_req_data['checkbox'])) {
3073
+					$trashed_or_restored = $trash ? $MTP->delete_by_ID($GRP_ID) : $MTP->restore_by_ID($GRP_ID);
3074
+					if (! $trashed_or_restored) {
3075
+						$success = 0;
3076
+					}
3077
+				}
3078
+			} else {
3079
+				// grab single GRP_ID and handle
3080
+				$GRP_ID = isset($this->_req_data['id']) ? absint($this->_req_data['id']) : 0;
3081
+				if (! empty($GRP_ID)) {
3082
+					$trashed_or_restored = $trash ? $MTP->delete_by_ID($GRP_ID) : $MTP->restore_by_ID($GRP_ID);
3083
+					if (! $trashed_or_restored) {
3084
+						$success = 0;
3085
+					}
3086
+				} else {
3087
+					$success = 0;
3088
+				}
3089
+			}
3090
+		}
3091
+
3092
+		$action_desc = $trash
3093
+			? esc_html__('moved to the trash', 'event_espresso')
3094
+			: esc_html__('restored', 'event_espresso');
3095
+
3096
+		$action_desc = ! empty($this->_req_data['template_switch']) ? esc_html__('switched', 'event_espresso') : $action_desc;
3097
+
3098
+		$item_desc = $all ? _n(
3099
+			'Message Template Group',
3100
+			'Message Template Groups',
3101
+			$success,
3102
+			'event_espresso'
3103
+		) : _n('Message Template Context', 'Message Template Contexts', $success, 'event_espresso');
3104
+
3105
+		$item_desc = ! empty($this->_req_data['template_switch']) ? _n(
3106
+			'template',
3107
+			'templates',
3108
+			$success,
3109
+			'event_espresso'
3110
+		) : $item_desc;
3111
+
3112
+		$this->_redirect_after_action($success, $item_desc, $action_desc, array());
3113
+	}
3114
+
3115
+
3116
+	/**
3117
+	 * [_delete_message_template]
3118
+	 * NOTE: this handles not only the deletion of the groups but also all the templates belonging to that group.
3119
+	 *
3120
+	 * @return void
3121
+	 * @throws EE_Error
3122
+	 * @throws InvalidArgumentException
3123
+	 * @throws InvalidDataTypeException
3124
+	 * @throws InvalidInterfaceException
3125
+	 */
3126
+	protected function _delete_message_template()
3127
+	{
3128
+		do_action('AHEE_log', __FILE__, __FUNCTION__, '');
3129
+
3130
+		// checkboxes
3131
+		if (! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3132
+			// if array has more than one element then success message should be plural
3133
+			$success = count($this->_req_data['checkbox']) > 1 ? 2 : 1;
3134
+
3135
+			// cycle through bulk action checkboxes
3136
+			while (list($GRP_ID, $value) = each($this->_req_data['checkbox'])) {
3137
+				$success = $this->_delete_mtp_permanently($GRP_ID);
3138
+			}
3139
+		} else {
3140
+			// grab single grp_id and delete
3141
+			$GRP_ID = absint($this->_req_data['id']);
3142
+			$success = $this->_delete_mtp_permanently($GRP_ID);
3143
+		}
3144
+
3145
+		$this->_redirect_after_action($success, 'Message Templates', 'deleted', array());
3146
+	}
3147
+
3148
+
3149
+	/**
3150
+	 * helper for permanently deleting a mtP group and all related message_templates
3151
+	 *
3152
+	 * @param  int  $GRP_ID        The group being deleted
3153
+	 * @param  bool $include_group whether to delete the Message Template Group as well.
3154
+	 * @return bool boolean to indicate the success of the deletes or not.
3155
+	 * @throws EE_Error
3156
+	 * @throws InvalidArgumentException
3157
+	 * @throws InvalidDataTypeException
3158
+	 * @throws InvalidInterfaceException
3159
+	 */
3160
+	private function _delete_mtp_permanently($GRP_ID, $include_group = true)
3161
+	{
3162
+		$success = 1;
3163
+		$MTPG = EEM_Message_Template_Group::instance();
3164
+		// first let's GET this group
3165
+		$MTG = $MTPG->get_one_by_ID($GRP_ID);
3166
+		// then delete permanently all the related Message Templates
3167
+		$deleted = $MTG->delete_related_permanently('Message_Template');
3168
+
3169
+		if ($deleted === 0) {
3170
+			$success = 0;
3171
+		}
3172
+
3173
+		// now delete permanently this particular group
3174
+
3175
+		if ($include_group && ! $MTG->delete_permanently()) {
3176
+			$success = 0;
3177
+		}
3178
+
3179
+		return $success;
3180
+	}
3181
+
3182
+
3183
+	/**
3184
+	 *    _learn_more_about_message_templates_link
3185
+	 *
3186
+	 * @access protected
3187
+	 * @return string
3188
+	 */
3189
+	protected function _learn_more_about_message_templates_link()
3190
+	{
3191
+		return '<a class="hidden" style="margin:0 20px; cursor:pointer; font-size:12px;" >'
3192
+			   . esc_html__('learn more about how message templates works', 'event_espresso')
3193
+			   . '</a>';
3194
+	}
3195
+
3196
+
3197
+	/**
3198
+	 * Used for setting up messenger/message type activation.  This loads up the initial view.  The rest is handled by
3199
+	 * ajax and other routes.
3200
+	 *
3201
+	 * @return void
3202
+	 * @throws DomainException
3203
+	 */
3204
+	protected function _settings()
3205
+	{
3206
+
3207
+
3208
+		$this->_set_m_mt_settings();
3209
+
3210
+		$selected_messenger = isset($this->_req_data['selected_messenger'])
3211
+			? $this->_req_data['selected_messenger']
3212
+			: 'email';
3213
+
3214
+		// let's setup the messenger tabs
3215
+		$this->_template_args['admin_page_header'] = EEH_Tabbed_Content::tab_text_links(
3216
+			$this->_m_mt_settings['messenger_tabs'],
3217
+			'messenger_links',
3218
+			'|',
3219
+			$selected_messenger
3220
+		);
3221
+		$this->_template_args['before_admin_page_content'] = '<div class="ui-widget ui-helper-clearfix">';
3222
+		$this->_template_args['after_admin_page_content'] = '</div><!-- end .ui-widget -->';
3223
+
3224
+		$this->display_admin_page_with_sidebar();
3225
+	}
3226
+
3227
+
3228
+	/**
3229
+	 * This sets the $_m_mt_settings property for when needed (used on the Messages settings page)
3230
+	 *
3231
+	 * @access protected
3232
+	 * @return void
3233
+	 * @throws DomainException
3234
+	 */
3235
+	protected function _set_m_mt_settings()
3236
+	{
3237
+		// first if this is already set then lets get out no need to regenerate data.
3238
+		if (! empty($this->_m_mt_settings)) {
3239
+			return;
3240
+		}
3241
+
3242
+		// get all installed messengers and message_types
3243
+		/** @type EE_messenger[] $messengers */
3244
+		$messengers = $this->_message_resource_manager->installed_messengers();
3245
+		/** @type EE_message_type[] $message_types */
3246
+		$message_types = $this->_message_resource_manager->installed_message_types();
3247
+
3248
+
3249
+		// assemble the array for the _tab_text_links helper
3250
+
3251
+		foreach ($messengers as $messenger) {
3252
+			$this->_m_mt_settings['messenger_tabs'][ $messenger->name ] = array(
3253
+				'label' => ucwords($messenger->label['singular']),
3254
+				'class' => $this->_message_resource_manager->is_messenger_active($messenger->name)
3255
+					? 'messenger-active'
3256
+					: '',
3257
+				'href'  => $messenger->name,
3258
+				'title' => esc_html__('Modify this Messenger', 'event_espresso'),
3259
+				'slug'  => $messenger->name,
3260
+				'obj'   => $messenger,
3261
+			);
3262
+
3263
+
3264
+			$message_types_for_messenger = $messenger->get_valid_message_types();
3265
+
3266
+			foreach ($message_types as $message_type) {
3267
+				// first we need to verify that this message type is valid with this messenger. Cause if it isn't then
3268
+				// it shouldn't show in either the inactive OR active metabox.
3269
+				if (! in_array($message_type->name, $message_types_for_messenger, true)) {
3270
+					continue;
3271
+				}
3272
+
3273
+				$a_or_i = $this->_message_resource_manager->is_message_type_active_for_messenger(
3274
+					$messenger->name,
3275
+					$message_type->name
3276
+				)
3277
+					? 'active'
3278
+					: 'inactive';
3279
+
3280
+				$this->_m_mt_settings['message_type_tabs'][ $messenger->name ][ $a_or_i ][ $message_type->name ] = array(
3281
+					'label'    => ucwords($message_type->label['singular']),
3282
+					'class'    => 'message-type-' . $a_or_i,
3283
+					'slug_id'  => $message_type->name . '-messagetype-' . $messenger->name,
3284
+					'mt_nonce' => wp_create_nonce($message_type->name . '_nonce'),
3285
+					'href'     => 'espresso_' . $message_type->name . '_message_type_settings',
3286
+					'title'    => $a_or_i === 'active'
3287
+						? esc_html__('Drag this message type to the Inactive window to deactivate', 'event_espresso')
3288
+						: esc_html__('Drag this message type to the messenger to activate', 'event_espresso'),
3289
+					'content'  => $a_or_i === 'active'
3290
+						? $this->_message_type_settings_content($message_type, $messenger, true)
3291
+						: $this->_message_type_settings_content($message_type, $messenger),
3292
+					'slug'     => $message_type->name,
3293
+					'active'   => $a_or_i === 'active',
3294
+					'obj'      => $message_type,
3295
+				);
3296
+			}
3297
+		}
3298
+	}
3299
+
3300
+
3301
+	/**
3302
+	 * This just prepares the content for the message type settings
3303
+	 *
3304
+	 * @param  EE_message_type $message_type The message type object
3305
+	 * @param  EE_messenger    $messenger    The messenger object
3306
+	 * @param  boolean         $active       Whether the message type is active or not
3307
+	 * @return string html output for the content
3308
+	 * @throws DomainException
3309
+	 */
3310
+	protected function _message_type_settings_content($message_type, $messenger, $active = false)
3311
+	{
3312
+		// get message type fields
3313
+		$fields = $message_type->get_admin_settings_fields();
3314
+		$settings_template_args['template_form_fields'] = '';
3315
+
3316
+		if (! empty($fields) && $active) {
3317
+			$existing_settings = $message_type->get_existing_admin_settings($messenger->name);
3318
+			foreach ($fields as $fldname => $fldprops) {
3319
+				$field_id = $messenger->name . '-' . $message_type->name . '-' . $fldname;
3320
+				$template_form_field[ $field_id ] = array(
3321
+					'name'       => 'message_type_settings[' . $fldname . ']',
3322
+					'label'      => $fldprops['label'],
3323
+					'input'      => $fldprops['field_type'],
3324
+					'type'       => $fldprops['value_type'],
3325
+					'required'   => $fldprops['required'],
3326
+					'validation' => $fldprops['validation'],
3327
+					'value'      => isset($existing_settings[ $fldname ])
3328
+						? $existing_settings[ $fldname ]
3329
+						: $fldprops['default'],
3330
+					'options'    => isset($fldprops['options'])
3331
+						? $fldprops['options']
3332
+						: array(),
3333
+					'default'    => isset($existing_settings[ $fldname ])
3334
+						? $existing_settings[ $fldname ]
3335
+						: $fldprops['default'],
3336
+					'css_class'  => 'no-drag',
3337
+					'format'     => $fldprops['format'],
3338
+				);
3339
+			}
3340
+
3341
+
3342
+			$settings_template_args['template_form_fields'] = ! empty($template_form_field)
3343
+				? $this->_generate_admin_form_fields(
3344
+					$template_form_field,
3345
+					'string',
3346
+					'ee_mt_activate_form'
3347
+				)
3348
+				: '';
3349
+		}
3350
+
3351
+		$settings_template_args['description'] = $message_type->description;
3352
+		// we also need some hidden fields
3353
+		$settings_template_args['hidden_fields'] = array(
3354
+			'message_type_settings[messenger]'    => array(
3355
+				'type'  => 'hidden',
3356
+				'value' => $messenger->name,
3357
+			),
3358
+			'message_type_settings[message_type]' => array(
3359
+				'type'  => 'hidden',
3360
+				'value' => $message_type->name,
3361
+			),
3362
+			'type'                                => array(
3363
+				'type'  => 'hidden',
3364
+				'value' => 'message_type',
3365
+			),
3366
+		);
3367
+
3368
+		$settings_template_args['hidden_fields'] = $this->_generate_admin_form_fields(
3369
+			$settings_template_args['hidden_fields'],
3370
+			'array'
3371
+		);
3372
+		$settings_template_args['show_form'] = empty($settings_template_args['template_form_fields'])
3373
+			? ' hidden'
3374
+			: '';
3375
+
3376
+
3377
+		$template = EE_MSG_TEMPLATE_PATH . 'ee_msg_mt_settings_content.template.php';
3378
+		$content = EEH_Template::display_template($template, $settings_template_args, true);
3379
+
3380
+		return $content;
3381
+	}
3382
+
3383
+
3384
+	/**
3385
+	 * Generate all the metaboxes for the message types and register them for the messages settings page.
3386
+	 *
3387
+	 * @access protected
3388
+	 * @return void
3389
+	 * @throws DomainException
3390
+	 */
3391
+	protected function _messages_settings_metaboxes()
3392
+	{
3393
+		$this->_set_m_mt_settings();
3394
+		$m_boxes = $mt_boxes = array();
3395
+		$m_template_args = $mt_template_args = array();
3396
+
3397
+		$selected_messenger = isset($this->_req_data['selected_messenger'])
3398
+			? $this->_req_data['selected_messenger']
3399
+			: 'email';
3400
+
3401
+		if (isset($this->_m_mt_settings['messenger_tabs'])) {
3402
+			foreach ($this->_m_mt_settings['messenger_tabs'] as $messenger => $tab_array) {
3403
+				$hide_on_message = $this->_message_resource_manager->is_messenger_active($messenger) ? '' : 'hidden';
3404
+				$hide_off_message = $this->_message_resource_manager->is_messenger_active($messenger) ? 'hidden' : '';
3405
+				// messenger meta boxes
3406
+				$active = $selected_messenger === $messenger;
3407
+				$active_mt_tabs = isset(
3408
+					$this->_m_mt_settings['message_type_tabs'][ $messenger ]['active']
3409
+				)
3410
+					? $this->_m_mt_settings['message_type_tabs'][ $messenger ]['active']
3411
+					: '';
3412
+				$m_boxes[ $messenger . '_a_box' ] = sprintf(
3413
+					esc_html__('%s Settings', 'event_espresso'),
3414
+					$tab_array['label']
3415
+				);
3416
+				$m_template_args[ $messenger . '_a_box' ] = array(
3417
+					'active_message_types'   => ! empty($active_mt_tabs) ? $this->_get_mt_tabs($active_mt_tabs) : '',
3418
+					'inactive_message_types' => isset(
3419
+						$this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive']
3420
+					)
3421
+						? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive'])
3422
+						: '',
3423
+					'content'                => $this->_get_messenger_box_content($tab_array['obj']),
3424
+					'hidden'                 => $active ? '' : ' hidden',
3425
+					'hide_on_message'        => $hide_on_message,
3426
+					'messenger'              => $messenger,
3427
+					'active'                 => $active,
3428
+				);
3429
+				// message type meta boxes
3430
+				// (which is really just the inactive container for each messenger
3431
+				// showing inactive message types for that messenger)
3432
+				$mt_boxes[ $messenger . '_i_box' ] = esc_html__('Inactive Message Types', 'event_espresso');
3433
+				$mt_template_args[ $messenger . '_i_box' ] = array(
3434
+					'active_message_types'   => ! empty($active_mt_tabs) ? $this->_get_mt_tabs($active_mt_tabs) : '',
3435
+					'inactive_message_types' => isset(
3436
+						$this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive']
3437
+					)
3438
+						? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive'])
3439
+						: '',
3440
+					'hidden'                 => $active ? '' : ' hidden',
3441
+					'hide_on_message'        => $hide_on_message,
3442
+					'hide_off_message'       => $hide_off_message,
3443
+					'messenger'              => $messenger,
3444
+					'active'                 => $active,
3445
+				);
3446
+			}
3447
+		}
3448
+
3449
+
3450
+		// register messenger metaboxes
3451
+		$m_template_path = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_messenger_mt_meta_box.template.php';
3452
+		foreach ($m_boxes as $box => $label) {
3453
+			$callback_args = array('template_path' => $m_template_path, 'template_args' => $m_template_args[ $box ]);
3454
+			$msgr = str_replace('_a_box', '', $box);
3455
+			add_meta_box(
3456
+				'espresso_' . $msgr . '_settings',
3457
+				$label,
3458
+				function ($post, $metabox) {
3459
+					echo EEH_Template::display_template(
3460
+						$metabox["args"]["template_path"],
3461
+						$metabox["args"]["template_args"],
3462
+						true
3463
+					);
3464
+				},
3465
+				$this->_current_screen->id,
3466
+				'normal',
3467
+				'high',
3468
+				$callback_args
3469
+			);
3470
+		}
3471
+
3472
+		// register message type metaboxes
3473
+		$mt_template_path = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_messenger_meta_box.template.php';
3474
+		foreach ($mt_boxes as $box => $label) {
3475
+			$callback_args = array(
3476
+				'template_path' => $mt_template_path,
3477
+				'template_args' => $mt_template_args[ $box ],
3478
+			);
3479
+			$mt = str_replace('_i_box', '', $box);
3480
+			add_meta_box(
3481
+				'espresso_' . $mt . '_inactive_mts',
3482
+				$label,
3483
+				function ($post, $metabox) {
3484
+					echo EEH_Template::display_template(
3485
+						$metabox["args"]["template_path"],
3486
+						$metabox["args"]["template_args"],
3487
+						true
3488
+					);
3489
+				},
3490
+				$this->_current_screen->id,
3491
+				'side',
3492
+				'high',
3493
+				$callback_args
3494
+			);
3495
+		}
3496
+
3497
+		// register metabox for global messages settings but only when on the main site.  On single site installs this
3498
+		// will always result in the metabox showing, on multisite installs the metabox will only show on the main site.
3499
+		if (is_main_site()) {
3500
+			add_meta_box(
3501
+				'espresso_global_message_settings',
3502
+				esc_html__('Global Message Settings', 'event_espresso'),
3503
+				array($this, 'global_messages_settings_metabox_content'),
3504
+				$this->_current_screen->id,
3505
+				'normal',
3506
+				'low',
3507
+				array()
3508
+			);
3509
+		}
3510
+	}
3511
+
3512
+
3513
+	/**
3514
+	 *  This generates the content for the global messages settings metabox.
3515
+	 *
3516
+	 * @return string
3517
+	 * @throws EE_Error
3518
+	 * @throws InvalidArgumentException
3519
+	 * @throws ReflectionException
3520
+	 * @throws InvalidDataTypeException
3521
+	 * @throws InvalidInterfaceException
3522
+	 */
3523
+	public function global_messages_settings_metabox_content()
3524
+	{
3525
+		$form = $this->_generate_global_settings_form();
3526
+		echo $form->form_open(
3527
+			$this->add_query_args_and_nonce(array('action' => 'update_global_settings'), EE_MSG_ADMIN_URL),
3528
+			'POST'
3529
+		)
3530
+			 . $form->get_html()
3531
+			 . $form->form_close();
3532
+	}
3533
+
3534
+
3535
+	/**
3536
+	 * This generates and returns the form object for the global messages settings.
3537
+	 *
3538
+	 * @return EE_Form_Section_Proper
3539
+	 * @throws EE_Error
3540
+	 * @throws InvalidArgumentException
3541
+	 * @throws ReflectionException
3542
+	 * @throws InvalidDataTypeException
3543
+	 * @throws InvalidInterfaceException
3544
+	 */
3545
+	protected function _generate_global_settings_form()
3546
+	{
3547
+		EE_Registry::instance()->load_helper('HTML');
3548
+		/** @var EE_Network_Core_Config $network_config */
3549
+		$network_config = EE_Registry::instance()->NET_CFG->core;
3550
+
3551
+		return new EE_Form_Section_Proper(
3552
+			array(
3553
+				'name'            => 'global_messages_settings',
3554
+				'html_id'         => 'global_messages_settings',
3555
+				'html_class'      => 'form-table',
3556
+				'layout_strategy' => new EE_Admin_Two_Column_Layout(),
3557
+				'subsections'     => apply_filters(
3558
+					'FHEE__Messages_Admin_Page__global_messages_settings_metabox_content__form_subsections',
3559
+					array(
3560
+						'do_messages_on_same_request' => new EE_Select_Input(
3561
+							array(
3562
+								true  => esc_html__("On the same request", "event_espresso"),
3563
+								false => esc_html__("On a separate request", "event_espresso"),
3564
+							),
3565
+							array(
3566
+								'default'         => $network_config->do_messages_on_same_request,
3567
+								'html_label_text' => esc_html__(
3568
+									'Generate and send all messages:',
3569
+									'event_espresso'
3570
+								),
3571
+								'html_help_text'  => esc_html__(
3572
+									'By default the messages system uses a more efficient means of processing messages on separate requests and utilizes the wp-cron scheduling system.  This makes things execute faster for people registering for your events.  However, if the wp-cron system is disabled on your site and there is no alternative in place, then you can change this so messages are always executed on the same request.',
3573
+									'event_espresso'
3574
+								),
3575
+							)
3576
+						),
3577
+						'delete_threshold'            => new EE_Select_Input(
3578
+							array(
3579
+								0  => esc_html__('Forever', 'event_espresso'),
3580
+								3  => esc_html__('3 Months', 'event_espresso'),
3581
+								6  => esc_html__('6 Months', 'event_espresso'),
3582
+								9  => esc_html__('9 Months', 'event_espresso'),
3583
+								12 => esc_html__('12 Months', 'event_espresso'),
3584
+								24 => esc_html__('24 Months', 'event_espresso'),
3585
+								36 => esc_html__('36 Months', 'event_espresso'),
3586
+							),
3587
+							array(
3588
+								'default'         => EE_Registry::instance()->CFG->messages->delete_threshold,
3589
+								'html_label_text' => esc_html__('Cleanup of old messages:', 'event_espresso'),
3590
+								'html_help_text'  => esc_html__(
3591
+									'You can control how long a record of processed messages is kept via this option.',
3592
+									'event_espresso'
3593
+								),
3594
+							)
3595
+						),
3596
+						'update_settings'             => new EE_Submit_Input(
3597
+							array(
3598
+								'default'         => esc_html__('Update', 'event_espresso'),
3599
+								'html_label_text' => '&nbsp',
3600
+							)
3601
+						),
3602
+					)
3603
+				),
3604
+			)
3605
+		);
3606
+	}
3607
+
3608
+
3609
+	/**
3610
+	 * This handles updating the global settings set on the admin page.
3611
+	 *
3612
+	 * @throws EE_Error
3613
+	 * @throws InvalidDataTypeException
3614
+	 * @throws InvalidInterfaceException
3615
+	 * @throws InvalidArgumentException
3616
+	 * @throws ReflectionException
3617
+	 */
3618
+	protected function _update_global_settings()
3619
+	{
3620
+		/** @var EE_Network_Core_Config $network_config */
3621
+		$network_config = EE_Registry::instance()->NET_CFG->core;
3622
+		$messages_config = EE_Registry::instance()->CFG->messages;
3623
+		$form = $this->_generate_global_settings_form();
3624
+		if ($form->was_submitted()) {
3625
+			$form->receive_form_submission();
3626
+			if ($form->is_valid()) {
3627
+				$valid_data = $form->valid_data();
3628
+				foreach ($valid_data as $property => $value) {
3629
+					$setter = 'set_' . $property;
3630
+					if (method_exists($network_config, $setter)) {
3631
+						$network_config->{$setter}($value);
3632
+					} elseif (property_exists($network_config, $property)
3633
+						&& $network_config->{$property} !== $value
3634
+					) {
3635
+						$network_config->{$property} = $value;
3636
+					} elseif (property_exists($messages_config, $property)
3637
+						&& $messages_config->{$property} !== $value
3638
+					) {
3639
+						$messages_config->{$property} = $value;
3640
+					}
3641
+				}
3642
+				// only update if the form submission was valid!
3643
+				EE_Registry::instance()->NET_CFG->update_config(true, false);
3644
+				EE_Registry::instance()->CFG->update_espresso_config();
3645
+				EE_Error::overwrite_success();
3646
+				EE_Error::add_success(__('Global message settings were updated', 'event_espresso'));
3647
+			}
3648
+		}
3649
+		$this->_redirect_after_action(0, '', '', array('action' => 'settings'), true);
3650
+	}
3651
+
3652
+
3653
+	/**
3654
+	 * this prepares the messenger tabs that can be dragged in and out of messenger boxes to activate/deactivate
3655
+	 *
3656
+	 * @param  array $tab_array This is an array of message type tab details used to generate the tabs
3657
+	 * @return string html formatted tabs
3658
+	 * @throws DomainException
3659
+	 */
3660
+	protected function _get_mt_tabs($tab_array)
3661
+	{
3662
+		$tab_array = (array) $tab_array;
3663
+		$template = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_mt_settings_tab_item.template.php';
3664
+		$tabs = '';
3665
+
3666
+		foreach ($tab_array as $tab) {
3667
+			$tabs .= EEH_Template::display_template($template, $tab, true);
3668
+		}
3669
+
3670
+		return $tabs;
3671
+	}
3672
+
3673
+
3674
+	/**
3675
+	 * This prepares the content of the messenger meta box admin settings
3676
+	 *
3677
+	 * @param  EE_messenger $messenger The messenger we're setting up content for
3678
+	 * @return string html formatted content
3679
+	 * @throws DomainException
3680
+	 */
3681
+	protected function _get_messenger_box_content(EE_messenger $messenger)
3682
+	{
3683
+
3684
+		$fields = $messenger->get_admin_settings_fields();
3685
+		$settings_template_args['template_form_fields'] = '';
3686
+
3687
+		// is $messenger active?
3688
+		$settings_template_args['active'] = $this->_message_resource_manager->is_messenger_active($messenger->name);
3689
+
3690
+
3691
+		if (! empty($fields)) {
3692
+			$existing_settings = $messenger->get_existing_admin_settings();
3693
+
3694
+			foreach ($fields as $fldname => $fldprops) {
3695
+				$field_id = $messenger->name . '-' . $fldname;
3696
+				$template_form_field[ $field_id ] = array(
3697
+					'name'       => 'messenger_settings[' . $field_id . ']',
3698
+					'label'      => $fldprops['label'],
3699
+					'input'      => $fldprops['field_type'],
3700
+					'type'       => $fldprops['value_type'],
3701
+					'required'   => $fldprops['required'],
3702
+					'validation' => $fldprops['validation'],
3703
+					'value'      => isset($existing_settings[ $field_id ])
3704
+						? $existing_settings[ $field_id ]
3705
+						: $fldprops['default'],
3706
+					'css_class'  => '',
3707
+					'format'     => $fldprops['format'],
3708
+				);
3709
+			}
3710
+
3711
+
3712
+			$settings_template_args['template_form_fields'] = ! empty($template_form_field)
3713
+				? $this->_generate_admin_form_fields($template_form_field, 'string', 'ee_m_activate_form')
3714
+				: '';
3715
+		}
3716
+
3717
+		// we also need some hidden fields
3718
+		$settings_template_args['hidden_fields'] = array(
3719
+			'messenger_settings[messenger]' => array(
3720
+				'type'  => 'hidden',
3721
+				'value' => $messenger->name,
3722
+			),
3723
+			'type'                          => array(
3724
+				'type'  => 'hidden',
3725
+				'value' => 'messenger',
3726
+			),
3727
+		);
3728
+
3729
+		// make sure any active message types that are existing are included in the hidden fields
3730
+		if (isset($this->_m_mt_settings['message_type_tabs'][ $messenger->name ]['active'])) {
3731
+			foreach ($this->_m_mt_settings['message_type_tabs'][ $messenger->name ]['active'] as $mt => $values) {
3732
+				$settings_template_args['hidden_fields'][ 'messenger_settings[message_types][' . $mt . ']' ] = array(
3733
+					'type'  => 'hidden',
3734
+					'value' => $mt,
3735
+				);
3736
+			}
3737
+		}
3738
+		$settings_template_args['hidden_fields'] = $this->_generate_admin_form_fields(
3739
+			$settings_template_args['hidden_fields'],
3740
+			'array'
3741
+		);
3742
+		$active = $this->_message_resource_manager->is_messenger_active($messenger->name);
3743
+
3744
+		$settings_template_args['messenger'] = $messenger->name;
3745
+		$settings_template_args['description'] = $messenger->description;
3746
+		$settings_template_args['show_hide_edit_form'] = $active ? '' : ' hidden';
3747
+
3748
+
3749
+		$settings_template_args['show_hide_edit_form'] = $this->_message_resource_manager->is_messenger_active(
3750
+			$messenger->name
3751
+		)
3752
+			? $settings_template_args['show_hide_edit_form']
3753
+			: ' hidden';
3754
+
3755
+		$settings_template_args['show_hide_edit_form'] = empty($settings_template_args['template_form_fields'])
3756
+			? ' hidden'
3757
+			: $settings_template_args['show_hide_edit_form'];
3758
+
3759
+
3760
+		$settings_template_args['on_off_action'] = $active ? 'messenger-off' : 'messenger-on';
3761
+		$settings_template_args['nonce'] = wp_create_nonce('activate_' . $messenger->name . '_toggle_nonce');
3762
+		$settings_template_args['on_off_status'] = $active ? true : false;
3763
+		$template = EE_MSG_TEMPLATE_PATH . 'ee_msg_m_settings_content.template.php';
3764
+		$content = EEH_Template::display_template(
3765
+			$template,
3766
+			$settings_template_args,
3767
+			true
3768
+		);
3769
+
3770
+		return $content;
3771
+	}
3772
+
3773
+
3774
+	/**
3775
+	 * used by ajax on the messages settings page to activate|deactivate the messenger
3776
+	 *
3777
+	 * @throws DomainException
3778
+	 * @throws EE_Error
3779
+	 * @throws InvalidDataTypeException
3780
+	 * @throws InvalidInterfaceException
3781
+	 * @throws InvalidArgumentException
3782
+	 * @throws ReflectionException
3783
+	 */
3784
+	public function activate_messenger_toggle()
3785
+	{
3786
+		$success = true;
3787
+		$this->_prep_default_response_for_messenger_or_message_type_toggle();
3788
+		// let's check that we have required data
3789
+		if (! isset($this->_req_data['messenger'])) {
3790
+			EE_Error::add_error(
3791
+				esc_html__('Messenger name needed to toggle activation. None given', 'event_espresso'),
3792
+				__FILE__,
3793
+				__FUNCTION__,
3794
+				__LINE__
3795
+			);
3796
+			$success = false;
3797
+		}
3798
+
3799
+		// do a nonce check here since we're not arriving via a normal route
3800
+		$nonce = isset($this->_req_data['activate_nonce'])
3801
+			? sanitize_text_field($this->_req_data['activate_nonce'])
3802
+			: '';
3803
+		$nonce_ref = 'activate_' . $this->_req_data['messenger'] . '_toggle_nonce';
3804
+
3805
+		$this->_verify_nonce($nonce, $nonce_ref);
3806
+
3807
+
3808
+		if (! isset($this->_req_data['status'])) {
3809
+			EE_Error::add_error(
3810
+				esc_html__(
3811
+					'Messenger status needed to know whether activation or deactivation is happening. No status is given',
3812
+					'event_espresso'
3813
+				),
3814
+				__FILE__,
3815
+				__FUNCTION__,
3816
+				__LINE__
3817
+			);
3818
+			$success = false;
3819
+		}
3820
+
3821
+		// do check to verify we have a valid status.
3822
+		$status = $this->_req_data['status'];
3823
+
3824
+		if ($status !== 'off' && $status !== 'on') {
3825
+			EE_Error::add_error(
3826
+				sprintf(
3827
+					esc_html__('The given status (%s) is not valid. Must be "off" or "on"', 'event_espresso'),
3828
+					$this->_req_data['status']
3829
+				),
3830
+				__FILE__,
3831
+				__FUNCTION__,
3832
+				__LINE__
3833
+			);
3834
+			$success = false;
3835
+		}
3836
+
3837
+		if ($success) {
3838
+			// made it here?  Stop dawdling then!!
3839
+			$success = $status === 'off'
3840
+				? $this->_deactivate_messenger($this->_req_data['messenger'])
3841
+				: $this->_activate_messenger($this->_req_data['messenger']);
3842
+		}
3843
+
3844
+		$this->_template_args['success'] = $success;
3845
+
3846
+		// no special instructions so let's just do the json return (which should automatically do all the special stuff).
3847
+		$this->_return_json();
3848
+	}
3849
+
3850
+
3851
+	/**
3852
+	 * used by ajax from the messages settings page to activate|deactivate a message type
3853
+	 *
3854
+	 * @throws DomainException
3855
+	 * @throws EE_Error
3856
+	 * @throws ReflectionException
3857
+	 * @throws InvalidDataTypeException
3858
+	 * @throws InvalidInterfaceException
3859
+	 * @throws InvalidArgumentException
3860
+	 */
3861
+	public function activate_mt_toggle()
3862
+	{
3863
+		$success = true;
3864
+		$this->_prep_default_response_for_messenger_or_message_type_toggle();
3865
+
3866
+		// let's make sure we have the necessary data
3867
+		if (! isset($this->_req_data['message_type'])) {
3868
+			EE_Error::add_error(
3869
+				esc_html__('Message Type name needed to toggle activation. None given', 'event_espresso'),
3870
+				__FILE__,
3871
+				__FUNCTION__,
3872
+				__LINE__
3873
+			);
3874
+			$success = false;
3875
+		}
3876
+
3877
+		if (! isset($this->_req_data['messenger'])) {
3878
+			EE_Error::add_error(
3879
+				esc_html__('Messenger name needed to toggle activation. None given', 'event_espresso'),
3880
+				__FILE__,
3881
+				__FUNCTION__,
3882
+				__LINE__
3883
+			);
3884
+			$success = false;
3885
+		}
3886
+
3887
+		if (! isset($this->_req_data['status'])) {
3888
+			EE_Error::add_error(
3889
+				esc_html__(
3890
+					'Messenger status needed to know whether activation or deactivation is happening. No status is given',
3891
+					'event_espresso'
3892
+				),
3893
+				__FILE__,
3894
+				__FUNCTION__,
3895
+				__LINE__
3896
+			);
3897
+			$success = false;
3898
+		}
3899
+
3900
+
3901
+		// do check to verify we have a valid status.
3902
+		$status = $this->_req_data['status'];
3903
+
3904
+		if ($status !== 'activate' && $status !== 'deactivate') {
3905
+			EE_Error::add_error(
3906
+				sprintf(
3907
+					esc_html__('The given status (%s) is not valid. Must be "active" or "inactive"', 'event_espresso'),
3908
+					$this->_req_data['status']
3909
+				),
3910
+				__FILE__,
3911
+				__FUNCTION__,
3912
+				__LINE__
3913
+			);
3914
+			$success = false;
3915
+		}
3916
+
3917
+
3918
+		// do a nonce check here since we're not arriving via a normal route
3919
+		$nonce = isset($this->_req_data['mt_nonce']) ? sanitize_text_field($this->_req_data['mt_nonce']) : '';
3920
+		$nonce_ref = $this->_req_data['message_type'] . '_nonce';
3921
+
3922
+		$this->_verify_nonce($nonce, $nonce_ref);
3923
+
3924
+		if ($success) {
3925
+			// made it here? um, what are you waiting for then?
3926
+			$success = $status === 'deactivate'
3927
+				? $this->_deactivate_message_type_for_messenger(
3928
+					$this->_req_data['messenger'],
3929
+					$this->_req_data['message_type']
3930
+				)
3931
+				: $this->_activate_message_type_for_messenger(
3932
+					$this->_req_data['messenger'],
3933
+					$this->_req_data['message_type']
3934
+				);
3935
+		}
3936
+
3937
+		$this->_template_args['success'] = $success;
3938
+		$this->_return_json();
3939
+	}
3940
+
3941
+
3942
+	/**
3943
+	 * Takes care of processing activating a messenger and preparing the appropriate response.
3944
+	 *
3945
+	 * @param string $messenger_name The name of the messenger being activated
3946
+	 * @return bool
3947
+	 * @throws DomainException
3948
+	 * @throws EE_Error
3949
+	 * @throws InvalidArgumentException
3950
+	 * @throws ReflectionException
3951
+	 * @throws InvalidDataTypeException
3952
+	 * @throws InvalidInterfaceException
3953
+	 */
3954
+	protected function _activate_messenger($messenger_name)
3955
+	{
3956
+		/** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
3957
+		$active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
3958
+		$message_types_to_activate = $active_messenger instanceof EE_Messenger
3959
+			? $active_messenger->get_default_message_types()
3960
+			: array();
3961
+
3962
+		// ensure is active
3963
+		$this->_message_resource_manager->activate_messenger($messenger_name, $message_types_to_activate);
3964
+
3965
+		// set response_data for reload
3966
+		foreach ($message_types_to_activate as $message_type_name) {
3967
+			/** @var EE_message_type $message_type */
3968
+			$message_type = $this->_message_resource_manager->get_message_type($message_type_name);
3969
+			if ($this->_message_resource_manager->is_message_type_active_for_messenger(
3970
+				$messenger_name,
3971
+				$message_type_name
3972
+			)
3973
+				&& $message_type instanceof EE_message_type
3974
+			) {
3975
+				$this->_template_args['data']['active_mts'][] = $message_type_name;
3976
+				if ($message_type->get_admin_settings_fields()) {
3977
+					$this->_template_args['data']['mt_reload'][] = $message_type_name;
3978
+				}
3979
+			}
3980
+		}
3981
+
3982
+		// add success message for activating messenger
3983
+		return $this->_setup_response_message_for_activating_messenger_with_message_types($active_messenger);
3984
+	}
3985
+
3986
+
3987
+	/**
3988
+	 * Takes care of processing deactivating a messenger and preparing the appropriate response.
3989
+	 *
3990
+	 * @param string $messenger_name The name of the messenger being activated
3991
+	 * @return bool
3992
+	 * @throws DomainException
3993
+	 * @throws EE_Error
3994
+	 * @throws InvalidArgumentException
3995
+	 * @throws ReflectionException
3996
+	 * @throws InvalidDataTypeException
3997
+	 * @throws InvalidInterfaceException
3998
+	 */
3999
+	protected function _deactivate_messenger($messenger_name)
4000
+	{
4001
+		/** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
4002
+		$active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
4003
+		$this->_message_resource_manager->deactivate_messenger($messenger_name);
4004
+
4005
+		return $this->_setup_response_message_for_deactivating_messenger_with_message_types($active_messenger);
4006
+	}
4007
+
4008
+
4009
+	/**
4010
+	 * Takes care of processing activating a message type for a messenger and preparing the appropriate response.
4011
+	 *
4012
+	 * @param string $messenger_name    The name of the messenger the message type is being activated for.
4013
+	 * @param string $message_type_name The name of the message type being activated for the messenger
4014
+	 * @return bool
4015
+	 * @throws DomainException
4016
+	 * @throws EE_Error
4017
+	 * @throws InvalidArgumentException
4018
+	 * @throws ReflectionException
4019
+	 * @throws InvalidDataTypeException
4020
+	 * @throws InvalidInterfaceException
4021
+	 */
4022
+	protected function _activate_message_type_for_messenger($messenger_name, $message_type_name)
4023
+	{
4024
+		/** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
4025
+		$active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
4026
+		/** @var EE_message_type $message_type_to_activate This will be present because it can't be toggled if it isn't */
4027
+		$message_type_to_activate = $this->_message_resource_manager->get_message_type($message_type_name);
4028
+
4029
+		// ensure is active
4030
+		$this->_message_resource_manager->activate_messenger($messenger_name, $message_type_name);
4031
+
4032
+		// set response for load
4033
+		if ($this->_message_resource_manager->is_message_type_active_for_messenger(
4034
+			$messenger_name,
4035
+			$message_type_name
4036
+		)
4037
+		) {
4038
+			$this->_template_args['data']['active_mts'][] = $message_type_name;
4039
+			if ($message_type_to_activate->get_admin_settings_fields()) {
4040
+				$this->_template_args['data']['mt_reload'][] = $message_type_name;
4041
+			}
4042
+		}
4043
+
4044
+		return $this->_setup_response_message_for_activating_messenger_with_message_types(
4045
+			$active_messenger,
4046
+			$message_type_to_activate
4047
+		);
4048
+	}
4049
+
4050
+
4051
+	/**
4052
+	 * Takes care of processing deactivating a message type for a messenger and preparing the appropriate response.
4053
+	 *
4054
+	 * @param string $messenger_name    The name of the messenger the message type is being deactivated for.
4055
+	 * @param string $message_type_name The name of the message type being deactivated for the messenger
4056
+	 * @return bool
4057
+	 * @throws DomainException
4058
+	 * @throws EE_Error
4059
+	 * @throws InvalidArgumentException
4060
+	 * @throws ReflectionException
4061
+	 * @throws InvalidDataTypeException
4062
+	 * @throws InvalidInterfaceException
4063
+	 */
4064
+	protected function _deactivate_message_type_for_messenger($messenger_name, $message_type_name)
4065
+	{
4066
+		/** @var EE_messenger $active_messenger This will be present because it can't be toggled if it isn't */
4067
+		$active_messenger = $this->_message_resource_manager->get_messenger($messenger_name);
4068
+		/** @var EE_message_type $message_type_to_activate This will be present because it can't be toggled if it isn't */
4069
+		$message_type_to_deactivate = $this->_message_resource_manager->get_message_type($message_type_name);
4070
+		$this->_message_resource_manager->deactivate_message_type_for_messenger($message_type_name, $messenger_name);
4071
+
4072
+		return $this->_setup_response_message_for_deactivating_messenger_with_message_types(
4073
+			$active_messenger,
4074
+			$message_type_to_deactivate
4075
+		);
4076
+	}
4077
+
4078
+
4079
+	/**
4080
+	 * This just initializes the defaults for activating messenger and message type responses.
4081
+	 */
4082
+	protected function _prep_default_response_for_messenger_or_message_type_toggle()
4083
+	{
4084
+		$this->_template_args['data']['active_mts'] = array();
4085
+		$this->_template_args['data']['mt_reload'] = array();
4086
+	}
4087
+
4088
+
4089
+	/**
4090
+	 * Setup appropriate response for activating a messenger and/or message types
4091
+	 *
4092
+	 * @param EE_messenger         $messenger
4093
+	 * @param EE_message_type|null $message_type
4094
+	 * @return bool
4095
+	 * @throws DomainException
4096
+	 * @throws EE_Error
4097
+	 * @throws InvalidArgumentException
4098
+	 * @throws ReflectionException
4099
+	 * @throws InvalidDataTypeException
4100
+	 * @throws InvalidInterfaceException
4101
+	 */
4102
+	protected function _setup_response_message_for_activating_messenger_with_message_types(
4103
+		$messenger,
4104
+		EE_Message_Type $message_type = null
4105
+	) {
4106
+		// if $messenger isn't a valid messenger object then get out.
4107
+		if (! $messenger instanceof EE_Messenger) {
4108
+			EE_Error::add_error(
4109
+				esc_html__('The messenger being activated is not a valid messenger', 'event_espresso'),
4110
+				__FILE__,
4111
+				__FUNCTION__,
4112
+				__LINE__
4113
+			);
4114
+
4115
+			return false;
4116
+		}
4117
+		// activated
4118
+		if ($this->_template_args['data']['active_mts']) {
4119
+			EE_Error::overwrite_success();
4120
+			// activated a message type with the messenger
4121
+			if ($message_type instanceof EE_message_type) {
4122
+				EE_Error::add_success(
4123
+					sprintf(
4124
+						esc_html__(
4125
+							'%s message type has been successfully activated with the %s messenger',
4126
+							'event_espresso'
4127
+						),
4128
+						ucwords($message_type->label['singular']),
4129
+						ucwords($messenger->label['singular'])
4130
+					)
4131
+				);
4132
+
4133
+				// if message type was invoice then let's make sure we activate the invoice payment method.
4134
+				if ($message_type->name === 'invoice') {
4135
+					EE_Registry::instance()->load_lib('Payment_Method_Manager');
4136
+					$pm = EE_Payment_Method_Manager::instance()->activate_a_payment_method_of_type('Invoice');
4137
+					if ($pm instanceof EE_Payment_Method) {
4138
+						EE_Error::add_attention(
4139
+							esc_html__(
4140
+								'Activating the invoice message type also automatically activates the invoice payment method.  If you do not wish the invoice payment method to be active, or to change its settings, visit the payment method admin page.',
4141
+								'event_espresso'
4142
+							)
4143
+						);
4144
+					}
4145
+				}
4146
+				// just toggles the entire messenger
4147
+			} else {
4148
+				EE_Error::add_success(
4149
+					sprintf(
4150
+						esc_html__('%s messenger has been successfully activated', 'event_espresso'),
4151
+						ucwords($messenger->label['singular'])
4152
+					)
4153
+				);
4154
+			}
4155
+
4156
+			return true;
4157
+
4158
+			// possible error condition. This will happen when our active_mts data is empty because it is validated for actual active
4159
+			// message types after the activation process.  However its possible some messengers don't HAVE any default_message_types
4160
+			// in which case we just give a success message for the messenger being successfully activated.
4161
+		} else {
4162
+			if (! $messenger->get_default_message_types()) {
4163
+				// messenger doesn't have any default message types so still a success.
4164
+				EE_Error::add_success(
4165
+					sprintf(
4166
+						esc_html__('%s messenger was successfully activated.', 'event_espresso'),
4167
+						ucwords($messenger->label['singular'])
4168
+					)
4169
+				);
4170
+
4171
+				return true;
4172
+			} else {
4173
+				EE_Error::add_error(
4174
+					$message_type instanceof EE_message_type
4175
+						? sprintf(
4176
+							esc_html__(
4177
+								'%s message type was not successfully activated with the %s messenger',
4178
+								'event_espresso'
4179
+							),
4180
+							ucwords($message_type->label['singular']),
4181
+							ucwords($messenger->label['singular'])
4182
+						)
4183
+						: sprintf(
4184
+							esc_html__('%s messenger was not successfully activated', 'event_espresso'),
4185
+							ucwords($messenger->label['singular'])
4186
+						),
4187
+					__FILE__,
4188
+					__FUNCTION__,
4189
+					__LINE__
4190
+				);
4191
+
4192
+				return false;
4193
+			}
4194
+		}
4195
+	}
4196
+
4197
+
4198
+	/**
4199
+	 * This sets up the appropriate response for deactivating a messenger and/or message type.
4200
+	 *
4201
+	 * @param EE_messenger         $messenger
4202
+	 * @param EE_message_type|null $message_type
4203
+	 * @return bool
4204
+	 * @throws DomainException
4205
+	 * @throws EE_Error
4206
+	 * @throws InvalidArgumentException
4207
+	 * @throws ReflectionException
4208
+	 * @throws InvalidDataTypeException
4209
+	 * @throws InvalidInterfaceException
4210
+	 */
4211
+	protected function _setup_response_message_for_deactivating_messenger_with_message_types(
4212
+		$messenger,
4213
+		EE_message_type $message_type = null
4214
+	) {
4215
+		EE_Error::overwrite_success();
4216
+
4217
+		// if $messenger isn't a valid messenger object then get out.
4218
+		if (! $messenger instanceof EE_Messenger) {
4219
+			EE_Error::add_error(
4220
+				esc_html__('The messenger being deactivated is not a valid messenger', 'event_espresso'),
4221
+				__FILE__,
4222
+				__FUNCTION__,
4223
+				__LINE__
4224
+			);
4225
+
4226
+			return false;
4227
+		}
4228
+
4229
+		if ($message_type instanceof EE_message_type) {
4230
+			$message_type_name = $message_type->name;
4231
+			EE_Error::add_success(
4232
+				sprintf(
4233
+					esc_html__(
4234
+						'%s message type has been successfully deactivated for the %s messenger.',
4235
+						'event_espresso'
4236
+					),
4237
+					ucwords($message_type->label['singular']),
4238
+					ucwords($messenger->label['singular'])
4239
+				)
4240
+			);
4241
+		} else {
4242
+			$message_type_name = '';
4243
+			EE_Error::add_success(
4244
+				sprintf(
4245
+					esc_html__('%s messenger has been successfully deactivated.', 'event_espresso'),
4246
+					ucwords($messenger->label['singular'])
4247
+				)
4248
+			);
4249
+		}
4250
+
4251
+		// if messenger was html or message type was invoice then let's make sure we deactivate invoice payment method.
4252
+		if ($messenger->name === 'html' || $message_type_name === 'invoice') {
4253
+			EE_Registry::instance()->load_lib('Payment_Method_Manager');
4254
+			$count_updated = EE_Payment_Method_Manager::instance()->deactivate_payment_method('invoice');
4255
+			if ($count_updated > 0) {
4256
+				$msg = $message_type_name === 'invoice'
4257
+					? esc_html__(
4258
+						'Deactivating the invoice message type also automatically deactivates the invoice payment method. In order for invoices to be generated the invoice message type must be active. If you completed this action by mistake, simply reactivate the invoice message type and then visit the payment methods admin page to reactivate the invoice payment method.',
4259
+						'event_espresso'
4260
+					)
4261
+					: esc_html__(
4262
+						'Deactivating the html messenger also automatically deactivates the invoice payment method.  In order for invoices to be generated the html messenger must be be active.  If you completed this action by mistake, simply reactivate the html messenger, then visit the payment methods admin page to reactivate the invoice payment method.',
4263
+						'event_espresso'
4264
+					);
4265
+				EE_Error::add_attention($msg);
4266
+			}
4267
+		}
4268
+
4269
+		return true;
4270
+	}
4271
+
4272
+
4273
+	/**
4274
+	 * handles updating a message type form on messenger activation IF the message type has settings fields. (via ajax)
4275
+	 *
4276
+	 * @throws DomainException
4277
+	 */
4278
+	public function update_mt_form()
4279
+	{
4280
+		if (! isset($this->_req_data['messenger']) || ! isset($this->_req_data['message_type'])) {
4281
+			EE_Error::add_error(
4282
+				esc_html__('Require message type or messenger to send an updated form', 'event_espresso'),
4283
+				__FILE__,
4284
+				__FUNCTION__,
4285
+				__LINE__
4286
+			);
4287
+			$this->_return_json();
4288
+		}
4289
+
4290
+		$message_types = $this->get_installed_message_types();
4291
+
4292
+		$message_type = $message_types[ $this->_req_data['message_type'] ];
4293
+		$messenger = $this->_message_resource_manager->get_active_messenger($this->_req_data['messenger']);
4294
+
4295
+		$content = $this->_message_type_settings_content(
4296
+			$message_type,
4297
+			$messenger,
4298
+			true
4299
+		);
4300
+		$this->_template_args['success'] = true;
4301
+		$this->_template_args['content'] = $content;
4302
+		$this->_return_json();
4303
+	}
4304
+
4305
+
4306
+	/**
4307
+	 * this handles saving the settings for a messenger or message type
4308
+	 *
4309
+	 */
4310
+	public function save_settings()
4311
+	{
4312
+		if (! isset($this->_req_data['type'])) {
4313
+			EE_Error::add_error(
4314
+				esc_html__(
4315
+					'Cannot save settings because type is unknown (messenger settings or messsage type settings?)',
4316
+					'event_espresso'
4317
+				),
4318
+				__FILE__,
4319
+				__FUNCTION__,
4320
+				__LINE__
4321
+			);
4322
+			$this->_template_args['error'] = true;
4323
+			$this->_return_json();
4324
+		}
4325
+
4326
+
4327
+		if ($this->_req_data['type'] === 'messenger') {
4328
+			// this should be an array.
4329
+			$settings = $this->_req_data['messenger_settings'];
4330
+			$messenger = $settings['messenger'];
4331
+			// let's setup the settings data
4332
+			foreach ($settings as $key => $value) {
4333
+				switch ($key) {
4334
+					case 'messenger':
4335
+						unset($settings['messenger']);
4336
+						break;
4337
+					case 'message_types':
4338
+						unset($settings['message_types']);
4339
+						break;
4340
+					default:
4341
+						$settings[ $key ] = $value;
4342
+						break;
4343
+				}
4344
+			}
4345
+			$this->_message_resource_manager->add_settings_for_messenger($messenger, $settings);
4346
+		} elseif ($this->_req_data['type'] === 'message_type') {
4347
+			$settings = $this->_req_data['message_type_settings'];
4348
+			$messenger = $settings['messenger'];
4349
+			$message_type = $settings['message_type'];
4350
+
4351
+			foreach ($settings as $key => $value) {
4352
+				switch ($key) {
4353
+					case 'messenger':
4354
+						unset($settings['messenger']);
4355
+						break;
4356
+					case 'message_type':
4357
+						unset($settings['message_type']);
4358
+						break;
4359
+					default:
4360
+						$settings[ $key ] = $value;
4361
+						break;
4362
+				}
4363
+			}
4364
+
4365
+			$this->_message_resource_manager->add_settings_for_message_type($messenger, $message_type, $settings);
4366
+		}
4367
+
4368
+		// okay we should have the data all setup.  Now we just update!
4369
+		$success = $this->_message_resource_manager->update_active_messengers_option();
4370
+
4371
+		if ($success) {
4372
+			EE_Error::add_success(__('Settings updated', 'event_espresso'));
4373
+		} else {
4374
+			EE_Error::add_error(
4375
+				esc_html__(
4376
+					'Settings did not get updated',
4377
+					'event_espresso'
4378
+				),
4379
+				__FILE__,
4380
+				__FUNCTION__,
4381
+				__LINE__
4382
+			);
4383
+		}
4384
+
4385
+		$this->_template_args['success'] = $success;
4386
+		$this->_return_json();
4387
+	}
4388
+
4389
+
4390
+
4391
+
4392
+	/**  EE MESSAGE PROCESSING ACTIONS **/
4393
+
4394
+
4395
+	/**
4396
+	 * This immediately generates any EE_Message ID's that are selected that are EEM_Message::status_incomplete
4397
+	 * However, this does not send immediately, it just queues for sending.
4398
+	 *
4399
+	 * @since 4.9.0
4400
+	 * @throws EE_Error
4401
+	 * @throws InvalidDataTypeException
4402
+	 * @throws InvalidInterfaceException
4403
+	 * @throws InvalidArgumentException
4404
+	 * @throws ReflectionException
4405
+	 */
4406
+	protected function _generate_now()
4407
+	{
4408
+		EED_Messages::generate_now($this->_get_msg_ids_from_request());
4409
+		$this->_redirect_after_action(false, '', '', array(), true);
4410
+	}
4411
+
4412
+
4413
+	/**
4414
+	 * This immediately generates AND sends any EE_Message's selected that are EEM_Message::status_incomplete or that
4415
+	 * are EEM_Message::status_resend or EEM_Message::status_idle
4416
+	 *
4417
+	 * @since 4.9.0
4418
+	 * @throws EE_Error
4419
+	 * @throws InvalidDataTypeException
4420
+	 * @throws InvalidInterfaceException
4421
+	 * @throws InvalidArgumentException
4422
+	 * @throws ReflectionException
4423
+	 */
4424
+	protected function _generate_and_send_now()
4425
+	{
4426
+		EED_Messages::generate_and_send_now($this->_get_msg_ids_from_request());
4427
+		$this->_redirect_after_action(false, '', '', array(), true);
4428
+	}
4429
+
4430
+
4431
+	/**
4432
+	 * This queues any EEM_Message::status_sent EE_Message ids in the request for resending.
4433
+	 *
4434
+	 * @since 4.9.0
4435
+	 * @throws EE_Error
4436
+	 * @throws InvalidDataTypeException
4437
+	 * @throws InvalidInterfaceException
4438
+	 * @throws InvalidArgumentException
4439
+	 * @throws ReflectionException
4440
+	 */
4441
+	protected function _queue_for_resending()
4442
+	{
4443
+		EED_Messages::queue_for_resending($this->_get_msg_ids_from_request());
4444
+		$this->_redirect_after_action(false, '', '', array(), true);
4445
+	}
4446
+
4447
+
4448
+	/**
4449
+	 *  This sends immediately any EEM_Message::status_idle or EEM_Message::status_resend messages in the queue
4450
+	 *
4451
+	 * @since 4.9.0
4452
+	 * @throws EE_Error
4453
+	 * @throws InvalidDataTypeException
4454
+	 * @throws InvalidInterfaceException
4455
+	 * @throws InvalidArgumentException
4456
+	 * @throws ReflectionException
4457
+	 */
4458
+	protected function _send_now()
4459
+	{
4460
+		EED_Messages::send_now($this->_get_msg_ids_from_request());
4461
+		$this->_redirect_after_action(false, '', '', array(), true);
4462
+	}
4463
+
4464
+
4465
+	/**
4466
+	 * Deletes EE_messages for IDs in the request.
4467
+	 *
4468
+	 * @since 4.9.0
4469
+	 * @throws EE_Error
4470
+	 * @throws InvalidDataTypeException
4471
+	 * @throws InvalidInterfaceException
4472
+	 * @throws InvalidArgumentException
4473
+	 */
4474
+	protected function _delete_ee_messages()
4475
+	{
4476
+		$msg_ids = $this->_get_msg_ids_from_request();
4477
+		$deleted_count = 0;
4478
+		foreach ($msg_ids as $msg_id) {
4479
+			if (EEM_Message::instance()->delete_by_ID($msg_id)) {
4480
+				$deleted_count++;
4481
+			}
4482
+		}
4483
+		if ($deleted_count) {
4484
+			EE_Error::add_success(
4485
+				esc_html(
4486
+					_n(
4487
+						'Message successfully deleted',
4488
+						'Messages successfully deleted',
4489
+						$deleted_count,
4490
+						'event_espresso'
4491
+					)
4492
+				)
4493
+			);
4494
+			$this->_redirect_after_action(
4495
+				false,
4496
+				'',
4497
+				'',
4498
+				array(),
4499
+				true
4500
+			);
4501
+		} else {
4502
+			EE_Error::add_error(
4503
+				_n('The message was not deleted.', 'The messages were not deleted', count($msg_ids), 'event_espresso'),
4504
+				__FILE__,
4505
+				__FUNCTION__,
4506
+				__LINE__
4507
+			);
4508
+			$this->_redirect_after_action(false, '', '', array(), true);
4509
+		}
4510
+	}
4511
+
4512
+
4513
+	/**
4514
+	 *  This looks for 'MSG_ID' key in the request and returns an array of MSG_ID's if present.
4515
+	 *
4516
+	 * @since 4.9.0
4517
+	 * @return array
4518
+	 */
4519
+	protected function _get_msg_ids_from_request()
4520
+	{
4521
+		if (! isset($this->_req_data['MSG_ID'])) {
4522
+			return array();
4523
+		}
4524
+
4525
+		return is_array($this->_req_data['MSG_ID'])
4526
+			? array_keys($this->_req_data['MSG_ID'])
4527
+			: array($this->_req_data['MSG_ID']);
4528
+	}
4529 4529
 }
Please login to merge, or discard this patch.
Spacing   +210 added lines, -210 removed lines patch added patch discarded remove patch
@@ -141,8 +141,8 @@  discard block
 block discarded – undo
141 141
         $i = 1;
142 142
         foreach ($active_messengers as $active_messenger) {
143 143
             if ($active_messenger instanceof EE_Message) {
144
-                $m_values[ $i ]['id'] = $active_messenger->messenger();
145
-                $m_values[ $i ]['text'] = ucwords($active_messenger->messenger_label());
144
+                $m_values[$i]['id'] = $active_messenger->messenger();
145
+                $m_values[$i]['text'] = ucwords($active_messenger->messenger_label());
146 146
                 $i++;
147 147
             }
148 148
         }
@@ -178,8 +178,8 @@  discard block
 block discarded – undo
178 178
         $i = 1;
179 179
         foreach ($active_messages as $active_message) {
180 180
             if ($active_message instanceof EE_Message) {
181
-                $mt_values[ $i ]['id'] = $active_message->message_type();
182
-                $mt_values[ $i ]['text'] = ucwords($active_message->message_type_label());
181
+                $mt_values[$i]['id'] = $active_message->message_type();
182
+                $mt_values[$i]['text'] = ucwords($active_message->message_type_label());
183 183
                 $i++;
184 184
             }
185 185
         }
@@ -218,7 +218,7 @@  discard block
 block discarded – undo
218 218
                 if ($message_type instanceof EE_message_type) {
219 219
                     $message_type_contexts = $message_type->get_contexts();
220 220
                     foreach ($message_type_contexts as $context => $context_details) {
221
-                        $contexts[ $context ] = $context_details['label'];
221
+                        $contexts[$context] = $context_details['label'];
222 222
                     }
223 223
                 }
224 224
             }
@@ -709,53 +709,53 @@  discard block
 block discarded – undo
709 709
 
710 710
     public function messages_help_tab()
711 711
     {
712
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_help_tab.template.php');
712
+        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH.'ee_msg_messages_help_tab.template.php');
713 713
     }
714 714
 
715 715
 
716 716
     public function messengers_help_tab()
717 717
     {
718
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messenger_help_tab.template.php');
718
+        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH.'ee_msg_messenger_help_tab.template.php');
719 719
     }
720 720
 
721 721
 
722 722
     public function message_types_help_tab()
723 723
     {
724
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_message_type_help_tab.template.php');
724
+        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH.'ee_msg_message_type_help_tab.template.php');
725 725
     }
726 726
 
727 727
 
728 728
     public function messages_overview_help_tab()
729 729
     {
730
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_overview_help_tab.template.php');
730
+        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH.'ee_msg_overview_help_tab.template.php');
731 731
     }
732 732
 
733 733
 
734 734
     public function message_templates_help_tab()
735 735
     {
736
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_message_templates_help_tab.template.php');
736
+        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH.'ee_msg_message_templates_help_tab.template.php');
737 737
     }
738 738
 
739 739
 
740 740
     public function edit_message_template_help_tab()
741 741
     {
742
-        $args['img1'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/editor.png' . '" alt="'
742
+        $args['img1'] = '<img src="'.EE_MSG_ASSETS_URL.'images/editor.png'.'" alt="'
743 743
                         . esc_attr__('Editor Title', 'event_espresso')
744 744
                         . '" />';
745
-        $args['img2'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/switch-context.png' . '" alt="'
745
+        $args['img2'] = '<img src="'.EE_MSG_ASSETS_URL.'images/switch-context.png'.'" alt="'
746 746
                         . esc_attr__('Context Switcher and Preview', 'event_espresso')
747 747
                         . '" />';
748
-        $args['img3'] = '<img class="left" src="' . EE_MSG_ASSETS_URL . 'images/form-fields.png' . '" alt="'
748
+        $args['img3'] = '<img class="left" src="'.EE_MSG_ASSETS_URL.'images/form-fields.png'.'" alt="'
749 749
                         . esc_attr__('Message Template Form Fields', 'event_espresso')
750 750
                         . '" />';
751
-        $args['img4'] = '<img class="right" src="' . EE_MSG_ASSETS_URL . 'images/shortcodes-metabox.png' . '" alt="'
751
+        $args['img4'] = '<img class="right" src="'.EE_MSG_ASSETS_URL.'images/shortcodes-metabox.png'.'" alt="'
752 752
                         . esc_attr__('Shortcodes Metabox', 'event_espresso')
753 753
                         . '" />';
754
-        $args['img5'] = '<img class="right" src="' . EE_MSG_ASSETS_URL . 'images/publish-meta-box.png' . '" alt="'
754
+        $args['img5'] = '<img class="right" src="'.EE_MSG_ASSETS_URL.'images/publish-meta-box.png'.'" alt="'
755 755
                         . esc_attr__('Publish Metabox', 'event_espresso')
756 756
                         . '" />';
757 757
         EEH_Template::display_template(
758
-            EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_templates_editor_help_tab.template.php',
758
+            EE_MSG_TEMPLATE_PATH.'ee_msg_messages_templates_editor_help_tab.template.php',
759 759
             $args
760 760
         );
761 761
     }
@@ -766,7 +766,7 @@  discard block
 block discarded – undo
766 766
         $this->_set_shortcodes();
767 767
         $args['shortcodes'] = $this->_shortcodes;
768 768
         EEH_Template::display_template(
769
-            EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_shortcodes_help_tab.template.php',
769
+            EE_MSG_TEMPLATE_PATH.'ee_msg_messages_shortcodes_help_tab.template.php',
770 770
             $args
771 771
         );
772 772
     }
@@ -774,16 +774,16 @@  discard block
 block discarded – undo
774 774
 
775 775
     public function preview_message_help_tab()
776 776
     {
777
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_preview_help_tab.template.php');
777
+        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH.'ee_msg_preview_help_tab.template.php');
778 778
     }
779 779
 
780 780
 
781 781
     public function settings_help_tab()
782 782
     {
783
-        $args['img1'] = '<img class="inline-text" src="' . EE_MSG_ASSETS_URL . 'images/email-tab-active.png'
784
-                        . '" alt="' . esc_attr__('Active Email Tab', 'event_espresso') . '" />';
785
-        $args['img2'] = '<img class="inline-text" src="' . EE_MSG_ASSETS_URL . 'images/email-tab-inactive.png'
786
-                        . '" alt="' . esc_attr__('Inactive Email Tab', 'event_espresso') . '" />';
783
+        $args['img1'] = '<img class="inline-text" src="'.EE_MSG_ASSETS_URL.'images/email-tab-active.png'
784
+                        . '" alt="'.esc_attr__('Active Email Tab', 'event_espresso').'" />';
785
+        $args['img2'] = '<img class="inline-text" src="'.EE_MSG_ASSETS_URL.'images/email-tab-inactive.png'
786
+                        . '" alt="'.esc_attr__('Inactive Email Tab', 'event_espresso').'" />';
787 787
         $args['img3'] = '<div class="switch">'
788 788
                         . '<input class="ee-on-off-toggle ee-toggle-round-flat"'
789 789
                         . ' type="checkbox" checked="checked">'
@@ -794,25 +794,25 @@  discard block
 block discarded – undo
794 794
                         . ' type="checkbox">'
795 795
                         . '<label for="ee-on-off-toggle-on"></label>'
796 796
                         . '</div>';
797
-        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH . 'ee_msg_messages_settings_help_tab.template.php', $args);
797
+        EEH_Template::display_template(EE_MSG_TEMPLATE_PATH.'ee_msg_messages_settings_help_tab.template.php', $args);
798 798
     }
799 799
 
800 800
 
801 801
     public function load_scripts_styles()
802 802
     {
803
-        wp_register_style('espresso_ee_msg', EE_MSG_ASSETS_URL . 'ee_message_admin.css', EVENT_ESPRESSO_VERSION);
803
+        wp_register_style('espresso_ee_msg', EE_MSG_ASSETS_URL.'ee_message_admin.css', EVENT_ESPRESSO_VERSION);
804 804
         wp_enqueue_style('espresso_ee_msg');
805 805
 
806 806
         wp_register_script(
807 807
             'ee-messages-settings',
808
-            EE_MSG_ASSETS_URL . 'ee-messages-settings.js',
808
+            EE_MSG_ASSETS_URL.'ee-messages-settings.js',
809 809
             array('jquery-ui-droppable', 'ee-serialize-full-array'),
810 810
             EVENT_ESPRESSO_VERSION,
811 811
             true
812 812
         );
813 813
         wp_register_script(
814 814
             'ee-msg-list-table-js',
815
-            EE_MSG_ASSETS_URL . 'ee_message_admin_list_table.js',
815
+            EE_MSG_ASSETS_URL.'ee_message_admin_list_table.js',
816 816
             array('ee-dialog'),
817 817
             EVENT_ESPRESSO_VERSION
818 818
         );
@@ -870,7 +870,7 @@  discard block
 block discarded – undo
870 870
 
871 871
         wp_register_script(
872 872
             'ee_msgs_edit_js',
873
-            EE_MSG_ASSETS_URL . 'ee_message_editor.js',
873
+            EE_MSG_ASSETS_URL.'ee_message_editor.js',
874 874
             array('jquery'),
875 875
             EVENT_ESPRESSO_VERSION
876 876
         );
@@ -914,7 +914,7 @@  discard block
 block discarded – undo
914 914
     {
915 915
         wp_register_style(
916 916
             'ee-message-settings',
917
-            EE_MSG_ASSETS_URL . 'ee_message_settings.css',
917
+            EE_MSG_ASSETS_URL.'ee_message_settings.css',
918 918
             array(),
919 919
             EVENT_ESPRESSO_VERSION
920 920
         );
@@ -1000,7 +1000,7 @@  discard block
 block discarded – undo
1000 1000
             }
1001 1001
             $status_bulk_actions = $common_bulk_actions;
1002 1002
             // unset bulk actions not applying to status
1003
-            if (! empty($status_bulk_actions)) {
1003
+            if ( ! empty($status_bulk_actions)) {
1004 1004
                 switch ($status) {
1005 1005
                     case EEM_Message::status_idle:
1006 1006
                     case EEM_Message::status_resend:
@@ -1029,7 +1029,7 @@  discard block
 block discarded – undo
1029 1029
                 continue;
1030 1030
             }
1031 1031
 
1032
-            $this->_views[ strtolower($status) ] = array(
1032
+            $this->_views[strtolower($status)] = array(
1033 1033
                 'slug'        => strtolower($status),
1034 1034
                 'label'       => EEH_Template::pretty_status($status, false, 'sentence'),
1035 1035
                 'count'       => 0,
@@ -1068,7 +1068,7 @@  discard block
 block discarded – undo
1068 1068
             if ($action_item === 'see_notifications_for') {
1069 1069
                 continue;
1070 1070
             }
1071
-            $action_items[ $action_item ] = array(
1071
+            $action_items[$action_item] = array(
1072 1072
                 'class' => $action_details['css_class'],
1073 1073
                 'desc'  => $action_details['label'],
1074 1074
             );
@@ -1077,37 +1077,37 @@  discard block
 block discarded – undo
1077 1077
         /** @type array $status_items status legend setup */
1078 1078
         $status_items = array(
1079 1079
             'sent_status'                => array(
1080
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_sent,
1080
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_sent,
1081 1081
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_sent, false, 'sentence'),
1082 1082
             ),
1083 1083
             'idle_status'                => array(
1084
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_idle,
1084
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_idle,
1085 1085
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_idle, false, 'sentence'),
1086 1086
             ),
1087 1087
             'failed_status'              => array(
1088
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_failed,
1088
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_failed,
1089 1089
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_failed, false, 'sentence'),
1090 1090
             ),
1091 1091
             'messenger_executing_status' => array(
1092
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_messenger_executing,
1092
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_messenger_executing,
1093 1093
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_messenger_executing, false, 'sentence'),
1094 1094
             ),
1095 1095
             'resend_status'              => array(
1096
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_resend,
1096
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_resend,
1097 1097
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_resend, false, 'sentence'),
1098 1098
             ),
1099 1099
             'incomplete_status'          => array(
1100
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_incomplete,
1100
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_incomplete,
1101 1101
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_incomplete, false, 'sentence'),
1102 1102
             ),
1103 1103
             'retry_status'               => array(
1104
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_retry,
1104
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_retry,
1105 1105
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_retry, false, 'sentence'),
1106 1106
             ),
1107 1107
         );
1108 1108
         if (EEM_Message::debug()) {
1109 1109
             $status_items['debug_only_status'] = array(
1110
-                'class' => 'ee-status-legend ee-status-legend-' . EEM_Message::status_debug_only,
1110
+                'class' => 'ee-status-legend ee-status-legend-'.EEM_Message::status_debug_only,
1111 1111
                 'desc'  => EEH_Template::pretty_status(EEM_Message::status_debug_only, false, 'sentence'),
1112 1112
             );
1113 1113
         }
@@ -1119,11 +1119,11 @@  discard block
 block discarded – undo
1119 1119
     protected function _custom_mtps_preview()
1120 1120
     {
1121 1121
         $this->_admin_page_title = esc_html__('Custom Message Templates (Preview)', 'event_espresso');
1122
-        $this->_template_args['preview_img'] = '<img src="' . EE_MSG_ASSETS_URL . 'images/custom_mtps_preview.png"'
1123
-                                               . ' alt="' . esc_attr__(
1122
+        $this->_template_args['preview_img'] = '<img src="'.EE_MSG_ASSETS_URL.'images/custom_mtps_preview.png"'
1123
+                                               . ' alt="'.esc_attr__(
1124 1124
                                                    'Preview Custom Message Templates screenshot',
1125 1125
                                                    'event_espresso'
1126
-                                               ) . '" />';
1126
+                                               ).'" />';
1127 1127
         $this->_template_args['preview_text'] = '<strong>'
1128 1128
                                                 . esc_html__(
1129 1129
                                                     'Custom Message Templates is a feature that is only available in the premium version of Event Espresso 4 which is available with a support license purchase on EventEspresso.com. With the Custom Message Templates feature, you are able to create custom message templates and assign them on a per-event basis.',
@@ -1203,7 +1203,7 @@  discard block
 block discarded – undo
1203 1203
         $installed = array();
1204 1204
 
1205 1205
         foreach ($installed_message_types as $message_type) {
1206
-            $installed[ $message_type->name ] = $message_type;
1206
+            $installed[$message_type->name] = $message_type;
1207 1207
         }
1208 1208
 
1209 1209
         return $installed;
@@ -1348,7 +1348,7 @@  discard block
 block discarded – undo
1348 1348
         // we need to assemble the title from Various details
1349 1349
         $context_label = sprintf(
1350 1350
             esc_html__('(%s %s)', 'event_espresso'),
1351
-            $c_config[ $context ]['label'],
1351
+            $c_config[$context]['label'],
1352 1352
             ucwords($c_label['label'])
1353 1353
         );
1354 1354
 
@@ -1370,7 +1370,7 @@  discard block
 block discarded – undo
1370 1370
             $message_template_group->message_type()
1371 1371
         );
1372 1372
 
1373
-        if (! $template_field_structure) {
1373
+        if ( ! $template_field_structure) {
1374 1374
             $template_field_structure = false;
1375 1375
             $template_fields = esc_html__(
1376 1376
                 'There was an error in assembling the fields for this display (you should see an error message)',
@@ -1384,21 +1384,21 @@  discard block
 block discarded – undo
1384 1384
 
1385 1385
         // if we have the extra key.. then we need to remove the content index from the template_field_structure as it
1386 1386
         // will get handled in the "extra" array.
1387
-        if (is_array($template_field_structure[ $context ]) && isset($template_field_structure[ $context ]['extra'])) {
1388
-            foreach ($template_field_structure[ $context ]['extra'] as $reference_field => $new_fields) {
1389
-                unset($template_field_structure[ $context ][ $reference_field ]);
1387
+        if (is_array($template_field_structure[$context]) && isset($template_field_structure[$context]['extra'])) {
1388
+            foreach ($template_field_structure[$context]['extra'] as $reference_field => $new_fields) {
1389
+                unset($template_field_structure[$context][$reference_field]);
1390 1390
             }
1391 1391
         }
1392 1392
 
1393 1393
         // let's loop through the template_field_structure and actually assemble the input fields!
1394
-        if (! empty($template_field_structure)) {
1395
-            foreach ($template_field_structure[ $context ] as $template_field => $field_setup_array) {
1394
+        if ( ! empty($template_field_structure)) {
1395
+            foreach ($template_field_structure[$context] as $template_field => $field_setup_array) {
1396 1396
                 // if this is an 'extra' template field then we need to remove any existing fields that are keyed up in
1397 1397
                 // the extra array and reset them.
1398 1398
                 if ($template_field === 'extra') {
1399 1399
                     $this->_template_args['is_extra_fields'] = true;
1400 1400
                     foreach ($field_setup_array as $reference_field => $new_fields_array) {
1401
-                        $message_template = $message_templates[ $context ][ $reference_field ];
1401
+                        $message_template = $message_templates[$context][$reference_field];
1402 1402
                         $content = $message_template instanceof EE_Message_Template
1403 1403
                             ? $message_template->get('MTP_content')
1404 1404
                             : '';
@@ -1407,7 +1407,7 @@  discard block
 block discarded – undo
1407 1407
                             $continue = false;
1408 1408
                             if (isset($extra_array['shortcodes_required'])) {
1409 1409
                                 foreach ((array) $extra_array['shortcodes_required'] as $shortcode) {
1410
-                                    if (! array_key_exists($shortcode, $this->_shortcodes)) {
1410
+                                    if ( ! array_key_exists($shortcode, $this->_shortcodes)) {
1411 1411
                                         $continue = true;
1412 1412
                                     }
1413 1413
                                 }
@@ -1420,57 +1420,57 @@  discard block
 block discarded – undo
1420 1420
                                         . '-'
1421 1421
                                         . $extra_field
1422 1422
                                         . '-content';
1423
-                            $template_form_fields[ $field_id ] = $extra_array;
1424
-                            $template_form_fields[ $field_id ]['name'] = 'MTP_template_fields['
1423
+                            $template_form_fields[$field_id] = $extra_array;
1424
+                            $template_form_fields[$field_id]['name'] = 'MTP_template_fields['
1425 1425
                                                                          . $reference_field
1426 1426
                                                                          . '][content]['
1427
-                                                                         . $extra_field . ']';
1427
+                                                                         . $extra_field.']';
1428 1428
                             $css_class = isset($extra_array['css_class'])
1429 1429
                                 ? $extra_array['css_class']
1430 1430
                                 : '';
1431 1431
 
1432
-                            $template_form_fields[ $field_id ]['css_class'] = ! empty($v_fields)
1432
+                            $template_form_fields[$field_id]['css_class'] = ! empty($v_fields)
1433 1433
                                                                               && in_array($extra_field, $v_fields, true)
1434 1434
                                                                               &&
1435 1435
                                                                               (
1436
-                                                                                  is_array($validators[ $extra_field ])
1437
-                                                                                  && isset($validators[ $extra_field ]['msg'])
1436
+                                                                                  is_array($validators[$extra_field])
1437
+                                                                                  && isset($validators[$extra_field]['msg'])
1438 1438
                                                                               )
1439
-                                ? 'validate-error ' . $css_class
1439
+                                ? 'validate-error '.$css_class
1440 1440
                                 : $css_class;
1441 1441
 
1442
-                            $template_form_fields[ $field_id ]['value'] = ! empty($message_templates)
1443
-                                                                          && isset($content[ $extra_field ])
1444
-                                ? $content[ $extra_field ]
1442
+                            $template_form_fields[$field_id]['value'] = ! empty($message_templates)
1443
+                                                                          && isset($content[$extra_field])
1444
+                                ? $content[$extra_field]
1445 1445
                                 : '';
1446 1446
 
1447 1447
                             // do we have a validation error?  if we do then let's use that value instead
1448
-                            $template_form_fields[ $field_id ]['value'] = isset($validators[ $extra_field ])
1449
-                                ? $validators[ $extra_field ]['value']
1450
-                                : $template_form_fields[ $field_id ]['value'];
1448
+                            $template_form_fields[$field_id]['value'] = isset($validators[$extra_field])
1449
+                                ? $validators[$extra_field]['value']
1450
+                                : $template_form_fields[$field_id]['value'];
1451 1451
 
1452 1452
 
1453
-                            $template_form_fields[ $field_id ]['db-col'] = 'MTP_content';
1453
+                            $template_form_fields[$field_id]['db-col'] = 'MTP_content';
1454 1454
 
1455 1455
                             // shortcode selector
1456 1456
                             $field_name_to_use = $extra_field === 'main'
1457 1457
                                 ? 'content'
1458 1458
                                 : $extra_field;
1459
-                            $template_form_fields[ $field_id ]['append_content'] = $this->_get_shortcode_selector(
1459
+                            $template_form_fields[$field_id]['append_content'] = $this->_get_shortcode_selector(
1460 1460
                                 $field_name_to_use,
1461 1461
                                 $field_id
1462 1462
                             );
1463 1463
 
1464 1464
                             if (isset($extra_array['input']) && $extra_array['input'] === 'wp_editor') {
1465 1465
                                 // we want to decode the entities
1466
-                                $template_form_fields[ $field_id ]['value'] = $template_form_fields[ $field_id ]['value'];
1466
+                                $template_form_fields[$field_id]['value'] = $template_form_fields[$field_id]['value'];
1467 1467
                             }/**/
1468 1468
                         }
1469
-                        $templatefield_MTP_id = $reference_field . '-MTP_ID';
1470
-                        $templatefield_templatename_id = $reference_field . '-name';
1469
+                        $templatefield_MTP_id = $reference_field.'-MTP_ID';
1470
+                        $templatefield_templatename_id = $reference_field.'-name';
1471 1471
 
1472
-                        $template_form_fields[ $templatefield_MTP_id ] = array(
1473
-                            'name'       => 'MTP_template_fields[' . $reference_field . '][MTP_ID]',
1472
+                        $template_form_fields[$templatefield_MTP_id] = array(
1473
+                            'name'       => 'MTP_template_fields['.$reference_field.'][MTP_ID]',
1474 1474
                             'label'      => null,
1475 1475
                             'input'      => 'hidden',
1476 1476
                             'type'       => 'int',
@@ -1482,8 +1482,8 @@  discard block
 block discarded – undo
1482 1482
                             'db-col'     => 'MTP_ID',
1483 1483
                         );
1484 1484
 
1485
-                        $template_form_fields[ $templatefield_templatename_id ] = array(
1486
-                            'name'       => 'MTP_template_fields[' . $reference_field . '][name]',
1485
+                        $template_form_fields[$templatefield_templatename_id] = array(
1486
+                            'name'       => 'MTP_template_fields['.$reference_field.'][name]',
1487 1487
                             'label'      => null,
1488 1488
                             'input'      => 'hidden',
1489 1489
                             'type'       => 'string',
@@ -1497,36 +1497,36 @@  discard block
 block discarded – undo
1497 1497
                     }
1498 1498
                     continue; // skip the next stuff, we got the necessary fields here for this dataset.
1499 1499
                 } else {
1500
-                    $field_id = $template_field . '-content';
1501
-                    $template_form_fields[ $field_id ] = $field_setup_array;
1502
-                    $template_form_fields[ $field_id ]['name'] = 'MTP_template_fields[' . $template_field . '][content]';
1503
-                    $message_template = isset($message_templates[ $context ][ $template_field ])
1504
-                        ? $message_templates[ $context ][ $template_field ]
1500
+                    $field_id = $template_field.'-content';
1501
+                    $template_form_fields[$field_id] = $field_setup_array;
1502
+                    $template_form_fields[$field_id]['name'] = 'MTP_template_fields['.$template_field.'][content]';
1503
+                    $message_template = isset($message_templates[$context][$template_field])
1504
+                        ? $message_templates[$context][$template_field]
1505 1505
                         : null;
1506
-                    $template_form_fields[ $field_id ]['value'] = ! empty($message_templates)
1507
-                                                                  && is_array($message_templates[ $context ])
1506
+                    $template_form_fields[$field_id]['value'] = ! empty($message_templates)
1507
+                                                                  && is_array($message_templates[$context])
1508 1508
                                                                   && $message_template instanceof EE_Message_Template
1509 1509
                         ? $message_template->get('MTP_content')
1510 1510
                         : '';
1511 1511
 
1512 1512
                     // do we have a validator error for this field?  if we do then we'll use that value instead
1513
-                    $template_form_fields[ $field_id ]['value'] = isset($validators[ $template_field ])
1514
-                        ? $validators[ $template_field ]['value']
1515
-                        : $template_form_fields[ $field_id ]['value'];
1513
+                    $template_form_fields[$field_id]['value'] = isset($validators[$template_field])
1514
+                        ? $validators[$template_field]['value']
1515
+                        : $template_form_fields[$field_id]['value'];
1516 1516
 
1517 1517
 
1518
-                    $template_form_fields[ $field_id ]['db-col'] = 'MTP_content';
1518
+                    $template_form_fields[$field_id]['db-col'] = 'MTP_content';
1519 1519
                     $css_class = isset($field_setup_array['css_class'])
1520 1520
                         ? $field_setup_array['css_class']
1521 1521
                         : '';
1522
-                    $template_form_fields[ $field_id ]['css_class'] = ! empty($v_fields)
1522
+                    $template_form_fields[$field_id]['css_class'] = ! empty($v_fields)
1523 1523
                                                                       && in_array($template_field, $v_fields, true)
1524
-                                                                      && isset($validators[ $template_field ]['msg'])
1525
-                        ? 'validate-error ' . $css_class
1524
+                                                                      && isset($validators[$template_field]['msg'])
1525
+                        ? 'validate-error '.$css_class
1526 1526
                         : $css_class;
1527 1527
 
1528 1528
                     // shortcode selector
1529
-                    $template_form_fields[ $field_id ]['append_content'] = $this->_get_shortcode_selector(
1529
+                    $template_form_fields[$field_id]['append_content'] = $this->_get_shortcode_selector(
1530 1530
                         $template_field,
1531 1531
                         $field_id
1532 1532
                     );
@@ -1534,12 +1534,12 @@  discard block
 block discarded – undo
1534 1534
 
1535 1535
                 // k took care of content field(s) now let's take care of others.
1536 1536
 
1537
-                $templatefield_MTP_id = $template_field . '-MTP_ID';
1538
-                $templatefield_field_templatename_id = $template_field . '-name';
1537
+                $templatefield_MTP_id = $template_field.'-MTP_ID';
1538
+                $templatefield_field_templatename_id = $template_field.'-name';
1539 1539
 
1540 1540
                 // foreach template field there are actually two form fields created
1541
-                $template_form_fields[ $templatefield_MTP_id ] = array(
1542
-                    'name'       => 'MTP_template_fields[' . $template_field . '][MTP_ID]',
1541
+                $template_form_fields[$templatefield_MTP_id] = array(
1542
+                    'name'       => 'MTP_template_fields['.$template_field.'][MTP_ID]',
1543 1543
                     'label'      => null,
1544 1544
                     'input'      => 'hidden',
1545 1545
                     'type'       => 'int',
@@ -1551,8 +1551,8 @@  discard block
 block discarded – undo
1551 1551
                     'db-col'     => 'MTP_ID',
1552 1552
                 );
1553 1553
 
1554
-                $template_form_fields[ $templatefield_field_templatename_id ] = array(
1555
-                    'name'       => 'MTP_template_fields[' . $template_field . '][name]',
1554
+                $template_form_fields[$templatefield_field_templatename_id] = array(
1555
+                    'name'       => 'MTP_template_fields['.$template_field.'][name]',
1556 1556
                     'label'      => null,
1557 1557
                     'input'      => 'hidden',
1558 1558
                     'type'       => 'string',
@@ -1695,10 +1695,10 @@  discard block
 block discarded – undo
1695 1695
                 'value' => $GRP_ID,
1696 1696
             );
1697 1697
             $sidebar_form_fields['ee-msg-evt-nonce'] = array(
1698
-                'name'  => $action . '_nonce',
1698
+                'name'  => $action.'_nonce',
1699 1699
                 'input' => 'hidden',
1700 1700
                 'type'  => 'string',
1701
-                'value' => wp_create_nonce($action . '_nonce'),
1701
+                'value' => wp_create_nonce($action.'_nonce'),
1702 1702
             );
1703 1703
 
1704 1704
             if (isset($this->_req_data['template_switch']) && $this->_req_data['template_switch']) {
@@ -1738,7 +1738,7 @@  discard block
 block discarded – undo
1738 1738
             ),
1739 1739
             $this->_admin_base_url
1740 1740
         );
1741
-        $preview_button = '<a href="' . $preview_url . '" class="button-secondary messages-preview-button">'
1741
+        $preview_button = '<a href="'.$preview_url.'" class="button-secondary messages-preview-button">'
1742 1742
                           . esc_html__('Preview', 'event_espresso')
1743 1743
                           . '</a>';
1744 1744
 
@@ -1775,7 +1775,7 @@  discard block
 block discarded – undo
1775 1775
 
1776 1776
         $this->_template_path = $this->_template_args['GRP_ID']
1777 1777
             ? EE_MSG_TEMPLATE_PATH . 'ee_msg_details_main_edit_meta_box.template.php'
1778
-            : EE_MSG_TEMPLATE_PATH . 'ee_msg_details_main_add_meta_box.template.php';
1778
+            : EE_MSG_TEMPLATE_PATH.'ee_msg_details_main_add_meta_box.template.php';
1779 1779
 
1780 1780
         // send along EE_Message_Template_Group object for further template use.
1781 1781
         $this->_template_args['MTP'] = $message_template_group;
@@ -1830,7 +1830,7 @@  discard block
 block discarded – undo
1830 1830
     ) {
1831 1831
         $template_args = array(
1832 1832
             'context'                   => $context,
1833
-            'nonce'                     => wp_create_nonce('activate_' . $context . '_toggle_nonce'),
1833
+            'nonce'                     => wp_create_nonce('activate_'.$context.'_toggle_nonce'),
1834 1834
             'is_active'                 => $message_template_group->is_context_active($context),
1835 1835
             'on_off_action'             => $message_template_group->is_context_active($context)
1836 1836
                 ? 'context-off'
@@ -1839,7 +1839,7 @@  discard block
 block discarded – undo
1839 1839
             'message_template_group_id' => $message_template_group->ID(),
1840 1840
         );
1841 1841
         return EEH_Template::display_template(
1842
-            EE_MSG_TEMPLATE_PATH . 'ee_msg_editor_active_context_element.template.php',
1842
+            EE_MSG_TEMPLATE_PATH.'ee_msg_editor_active_context_element.template.php',
1843 1843
             $template_args,
1844 1844
             true
1845 1845
         );
@@ -1860,7 +1860,7 @@  discard block
 block discarded – undo
1860 1860
     {
1861 1861
         $success = true;
1862 1862
         // check for required data
1863
-        if (! isset(
1863
+        if ( ! isset(
1864 1864
             $this->_req_data['message_template_group_id'],
1865 1865
             $this->_req_data['context'],
1866 1866
             $this->_req_data['status']
@@ -1877,7 +1877,7 @@  discard block
 block discarded – undo
1877 1877
         $nonce = isset($this->_req_data['toggle_context_nonce'])
1878 1878
             ? sanitize_text_field($this->_req_data['toggle_context_nonce'])
1879 1879
             : '';
1880
-        $nonce_ref = 'activate_' . $this->_req_data['context'] . '_toggle_nonce';
1880
+        $nonce_ref = 'activate_'.$this->_req_data['context'].'_toggle_nonce';
1881 1881
         $this->_verify_nonce($nonce, $nonce_ref);
1882 1882
         $status = $this->_req_data['status'];
1883 1883
         if ($status !== 'off' && $status !== 'on') {
@@ -1895,7 +1895,7 @@  discard block
 block discarded – undo
1895 1895
         $message_template_group = EEM_Message_Template_Group::instance()->get_one_by_ID(
1896 1896
             $this->_req_data['message_template_group_id']
1897 1897
         );
1898
-        if (! $message_template_group instanceof EE_Message_Template_Group) {
1898
+        if ( ! $message_template_group instanceof EE_Message_Template_Group) {
1899 1899
             EE_Error::add_error(
1900 1900
                 sprintf(
1901 1901
                     esc_html__(
@@ -2024,7 +2024,7 @@  discard block
 block discarded – undo
2024 2024
         $templates = array();
2025 2025
         $GRP_ID = ! empty($this->_req_data['GRP_ID']) ? $this->_req_data['GRP_ID'] : 0;
2026 2026
         // we need to make sure we've got the info we need.
2027
-        if (! isset($this->_req_data['msgr'], $this->_req_data['mt'], $this->_req_data['GRP_ID'])) {
2027
+        if ( ! isset($this->_req_data['msgr'], $this->_req_data['mt'], $this->_req_data['GRP_ID'])) {
2028 2028
             EE_Error::add_error(
2029 2029
                 esc_html__(
2030 2030
                     'In order to reset the template to its default we require the messenger, message type, and message template GRP_ID to know what is being reset.  At least one of these is missing.',
@@ -2066,7 +2066,7 @@  discard block
 block discarded – undo
2066 2066
         }
2067 2067
 
2068 2068
         // any error messages?
2069
-        if (! $success) {
2069
+        if ( ! $success) {
2070 2070
             EE_Error::add_error(
2071 2071
                 esc_html__(
2072 2072
                     'Something went wrong with deleting existing templates. Unable to reset to default',
@@ -2118,7 +2118,7 @@  discard block
 block discarded – undo
2118 2118
     public function _preview_message($send = false)
2119 2119
     {
2120 2120
         // first make sure we've got the necessary parameters
2121
-        if (! isset(
2121
+        if ( ! isset(
2122 2122
             $this->_req_data['message_type'],
2123 2123
             $this->_req_data['messenger'],
2124 2124
             $this->_req_data['messenger'],
@@ -2170,11 +2170,11 @@  discard block
 block discarded – undo
2170 2170
         $preview_title = sprintf(
2171 2171
             esc_html__('Viewing Preview for %s %s Message Template', 'event_espresso'),
2172 2172
             $active_messenger_label,
2173
-            ucwords($message_types[ $this->_req_data['message_type'] ]->label['singular'])
2173
+            ucwords($message_types[$this->_req_data['message_type']]->label['singular'])
2174 2174
         );
2175 2175
         // setup display of preview.
2176 2176
         $this->_admin_page_title = $preview_title;
2177
-        $this->_template_args['admin_page_content'] = $preview_button . '<br />' . $preview;
2177
+        $this->_template_args['admin_page_content'] = $preview_button.'<br />'.$preview;
2178 2178
         $this->_template_args['data']['force_json'] = true;
2179 2179
 
2180 2180
         return '';
@@ -2253,10 +2253,10 @@  discard block
 block discarded – undo
2253 2253
         foreach ($tp_collection as $tp) {
2254 2254
             // only include template packs that support this messenger and message type!
2255 2255
             $supports = $tp->get_supports();
2256
-            if (! isset($supports[ $this->_message_template_group->messenger() ])
2256
+            if ( ! isset($supports[$this->_message_template_group->messenger()])
2257 2257
                 || ! in_array(
2258 2258
                     $this->_message_template_group->message_type(),
2259
-                    $supports[ $this->_message_template_group->messenger() ],
2259
+                    $supports[$this->_message_template_group->messenger()],
2260 2260
                     true
2261 2261
                 )
2262 2262
             ) {
@@ -2309,7 +2309,7 @@  discard block
 block discarded – undo
2309 2309
         $template_args['template_pack_description'] = $template_pack_labels->template_pack_description;
2310 2310
         $template_args['template_variation_description'] = $template_pack_labels->template_variation_description;
2311 2311
 
2312
-        $template = EE_MSG_TEMPLATE_PATH . 'template_pack_and_variations_metabox.template.php';
2312
+        $template = EE_MSG_TEMPLATE_PATH.'template_pack_and_variations_metabox.template.php';
2313 2313
 
2314 2314
         EEH_Template::display_template($template, $template_args);
2315 2315
     }
@@ -2335,10 +2335,10 @@  discard block
 block discarded – undo
2335 2335
         // first we need to see if there are any fields
2336 2336
         $fields = $this->_message_template_group->messenger_obj()->get_test_settings_fields();
2337 2337
 
2338
-        if (! empty($fields)) {
2338
+        if ( ! empty($fields)) {
2339 2339
             // yup there be fields
2340 2340
             foreach ($fields as $field => $config) {
2341
-                $field_id = $this->_message_template_group->messenger() . '_' . $field;
2341
+                $field_id = $this->_message_template_group->messenger().'_'.$field;
2342 2342
                 $existing = $this->_message_template_group->messenger_obj()->get_existing_test_settings();
2343 2343
                 $default = isset($config['default']) ? $config['default'] : '';
2344 2344
                 $default = isset($config['value']) ? $config['value'] : $default;
@@ -2346,22 +2346,22 @@  discard block
 block discarded – undo
2346 2346
                 // if type is hidden and the value is empty
2347 2347
                 // something may have gone wrong so let's correct with the defaults
2348 2348
                 $fix = $config['input'] === 'hidden'
2349
-                       && isset($existing[ $field ])
2350
-                       && empty($existing[ $field ])
2349
+                       && isset($existing[$field])
2350
+                       && empty($existing[$field])
2351 2351
                     ? $default
2352 2352
                     : '';
2353
-                $existing[ $field ] = isset($existing[ $field ]) && empty($fix)
2354
-                    ? $existing[ $field ]
2353
+                $existing[$field] = isset($existing[$field]) && empty($fix)
2354
+                    ? $existing[$field]
2355 2355
                     : $fix;
2356 2356
 
2357
-                $template_form_fields[ $field_id ] = array(
2358
-                    'name'       => 'test_settings_fld[' . $field . ']',
2357
+                $template_form_fields[$field_id] = array(
2358
+                    'name'       => 'test_settings_fld['.$field.']',
2359 2359
                     'label'      => $config['label'],
2360 2360
                     'input'      => $config['input'],
2361 2361
                     'type'       => $config['type'],
2362 2362
                     'required'   => $config['required'],
2363 2363
                     'validation' => $config['validation'],
2364
-                    'value'      => isset($existing[ $field ]) ? $existing[ $field ] : $default,
2364
+                    'value'      => isset($existing[$field]) ? $existing[$field] : $default,
2365 2365
                     'css_class'  => $config['css_class'],
2366 2366
                     'options'    => isset($config['options']) ? $config['options'] : array(),
2367 2367
                     'default'    => $default,
@@ -2376,7 +2376,7 @@  discard block
 block discarded – undo
2376 2376
 
2377 2377
         $test_settings_html = '';
2378 2378
         // print out $test_settings_fields
2379
-        if (! empty($test_settings_fields)) {
2379
+        if ( ! empty($test_settings_fields)) {
2380 2380
             echo $test_settings_fields;
2381 2381
             $test_settings_html = '<input type="submit" class="button-primary mtp-test-button alignright" ';
2382 2382
             $test_settings_html .= 'name="test_button" value="';
@@ -2423,7 +2423,7 @@  discard block
 block discarded – undo
2423 2423
         );
2424 2424
 
2425 2425
         return EEH_Template::display_template(
2426
-            EE_MSG_TEMPLATE_PATH . 'shortcode_selector_skeleton.template.php',
2426
+            EE_MSG_TEMPLATE_PATH.'shortcode_selector_skeleton.template.php',
2427 2427
             $template_args,
2428 2428
             true
2429 2429
         );
@@ -2448,7 +2448,7 @@  discard block
 block discarded – undo
2448 2448
         // $messenger = $this->_message_template_group->messenger_obj();
2449 2449
         // now let's set the content depending on the status of the shortcodes array
2450 2450
         if (empty($shortcodes)) {
2451
-            $content = '<p>' . esc_html__('There are no valid shortcodes available', 'event_espresso') . '</p>';
2451
+            $content = '<p>'.esc_html__('There are no valid shortcodes available', 'event_espresso').'</p>';
2452 2452
             echo $content;
2453 2453
         } else {
2454 2454
             // $alt = 0;
@@ -2486,7 +2486,7 @@  discard block
 block discarded – undo
2486 2486
     {
2487 2487
 
2488 2488
         // no need to run this if the property is already set
2489
-        if (! empty($this->_shortcodes)) {
2489
+        if ( ! empty($this->_shortcodes)) {
2490 2490
             return;
2491 2491
         }
2492 2492
 
@@ -2540,7 +2540,7 @@  discard block
 block discarded – undo
2540 2540
     protected function _set_message_template_group()
2541 2541
     {
2542 2542
 
2543
-        if (! empty($this->_message_template_group)) {
2543
+        if ( ! empty($this->_message_template_group)) {
2544 2544
             return;
2545 2545
         } //get out if this is already set.
2546 2546
 
@@ -2587,7 +2587,7 @@  discard block
 block discarded – undo
2587 2587
                     <?php
2588 2588
                 }
2589 2589
                 // setup nonce_url
2590
-                wp_nonce_field($args['action'] . '_nonce', $args['action'] . '_nonce', false);
2590
+                wp_nonce_field($args['action'].'_nonce', $args['action'].'_nonce', false);
2591 2591
                 ?>
2592 2592
                 <select name="context">
2593 2593
                     <?php
@@ -2597,7 +2597,7 @@  discard block
 block discarded – undo
2597 2597
                             $checked = ($context === $args['context']) ? 'selected="selected"' : '';
2598 2598
                             ?>
2599 2599
                             <option value="<?php echo $context; ?>" <?php echo $checked; ?>>
2600
-                                <?php echo $context_details[ $context ]['label']; ?>
2600
+                                <?php echo $context_details[$context]['label']; ?>
2601 2601
                             </option>
2602 2602
                         <?php endforeach;
2603 2603
                     endif; ?>
@@ -2627,22 +2627,22 @@  discard block
 block discarded – undo
2627 2627
      */
2628 2628
     protected function _set_message_template_column_values($index)
2629 2629
     {
2630
-        if (is_array($this->_req_data['MTP_template_fields'][ $index ]['content'])) {
2631
-            foreach ($this->_req_data['MTP_template_fields'][ $index ]['content'] as $field => $value) {
2632
-                $this->_req_data['MTP_template_fields'][ $index ]['content'][ $field ] = $value;
2630
+        if (is_array($this->_req_data['MTP_template_fields'][$index]['content'])) {
2631
+            foreach ($this->_req_data['MTP_template_fields'][$index]['content'] as $field => $value) {
2632
+                $this->_req_data['MTP_template_fields'][$index]['content'][$field] = $value;
2633 2633
             }
2634 2634
         }
2635 2635
 
2636 2636
 
2637 2637
         $set_column_values = array(
2638
-            'MTP_ID'             => absint($this->_req_data['MTP_template_fields'][ $index ]['MTP_ID']),
2638
+            'MTP_ID'             => absint($this->_req_data['MTP_template_fields'][$index]['MTP_ID']),
2639 2639
             'GRP_ID'             => absint($this->_req_data['GRP_ID']),
2640 2640
             'MTP_user_id'        => absint($this->_req_data['MTP_user_id']),
2641 2641
             'MTP_messenger'      => strtolower($this->_req_data['MTP_messenger']),
2642 2642
             'MTP_message_type'   => strtolower($this->_req_data['MTP_message_type']),
2643
-            'MTP_template_field' => strtolower($this->_req_data['MTP_template_fields'][ $index ]['name']),
2643
+            'MTP_template_field' => strtolower($this->_req_data['MTP_template_fields'][$index]['name']),
2644 2644
             'MTP_context'        => strtolower($this->_req_data['MTP_context']),
2645
-            'MTP_content'        => $this->_req_data['MTP_template_fields'][ $index ]['content'],
2645
+            'MTP_content'        => $this->_req_data['MTP_template_fields'][$index]['content'],
2646 2646
             'MTP_is_global'      => isset($this->_req_data['MTP_is_global'])
2647 2647
                 ? absint($this->_req_data['MTP_is_global'])
2648 2648
                 : 0,
@@ -2689,7 +2689,7 @@  discard block
 block discarded – undo
2689 2689
         $context = ucwords(str_replace('_', ' ', $context_slug));
2690 2690
 
2691 2691
         $item_desc = $messenger_label && $message_type_label
2692
-            ? $messenger_label . ' ' . $message_type_label . ' ' . $context . ' '
2692
+            ? $messenger_label.' '.$message_type_label.' '.$context.' '
2693 2693
             : '';
2694 2694
         $item_desc .= 'Message Template';
2695 2695
         $query_args = array();
@@ -2720,7 +2720,7 @@  discard block
 block discarded – undo
2720 2720
 
2721 2721
 
2722 2722
             // run update for each template field in displayed context
2723
-            if (! isset($this->_req_data['MTP_template_fields']) && empty($this->_req_data['MTP_template_fields'])) {
2723
+            if ( ! isset($this->_req_data['MTP_template_fields']) && empty($this->_req_data['MTP_template_fields'])) {
2724 2724
                 EE_Error::add_error(
2725 2725
                     esc_html__(
2726 2726
                         'There was a problem saving the template fields from the form because I didn\'t receive any actual template field data.',
@@ -2782,10 +2782,10 @@  discard block
 block discarded – undo
2782 2782
                         $set_column_values = $this->_set_message_template_column_values($template_field);
2783 2783
 
2784 2784
                         $where_cols_n_values = array(
2785
-                            'MTP_ID' => $this->_req_data['MTP_template_fields'][ $template_field ]['MTP_ID'],
2785
+                            'MTP_ID' => $this->_req_data['MTP_template_fields'][$template_field]['MTP_ID'],
2786 2786
                         );
2787 2787
                         // if they aren't allowed to use all JS, restrict them to just posty-y tags
2788
-                        if (! current_user_can('unfiltered_html')) {
2788
+                        if ( ! current_user_can('unfiltered_html')) {
2789 2789
                             if (is_array($set_column_values['MTP_content'])) {
2790 2790
                                 foreach ($set_column_values['MTP_content'] as $key => $value) {
2791 2791
                                     // remove slashes so wp_kses works properly (its wp_kses_stripslashes() function
@@ -2793,7 +2793,7 @@  discard block
 block discarded – undo
2793 2793
                                     // appear invalid.) But currently the models expect slashed data, so after wp_kses
2794 2794
                                     // runs we need to re-slash the data. Sheesh. See
2795 2795
                                     // https://events.codebasehq.com/projects/event-espresso/tickets/11211#update-47321587
2796
-                                    $set_column_values['MTP_content'][ $key ] = addslashes(
2796
+                                    $set_column_values['MTP_content'][$key] = addslashes(
2797 2797
                                         wp_kses(
2798 2798
                                             stripslashes($value),
2799 2799
                                             wp_kses_allowed_html('post')
@@ -2829,14 +2829,14 @@  discard block
 block discarded – undo
2829 2829
                             }
2830 2830
                         } else {
2831 2831
                             // only do this logic if we don't have a MTP_ID for this field
2832
-                            if (empty($this->_req_data['MTP_template_fields'][ $template_field ]['MTP_ID'])) {
2832
+                            if (empty($this->_req_data['MTP_template_fields'][$template_field]['MTP_ID'])) {
2833 2833
                                 // this has already been through the template field validator and sanitized, so it will be
2834 2834
                                 // safe to insert this field.  Why insert?  This typically happens when we introduce a new
2835 2835
                                 // message template field in a messenger/message type and existing users don't have the
2836 2836
                                 // default setup for it.
2837 2837
                                 // @link https://events.codebasehq.com/projects/event-espresso/tickets/9465
2838 2838
                                 $updated = $MTP->insert($message_template_fields);
2839
-                                if (! $updated || is_wp_error($updated)) {
2839
+                                if ( ! $updated || is_wp_error($updated)) {
2840 2840
                                     EE_Error::add_error(
2841 2841
                                         sprintf(
2842 2842
                                             esc_html__('%s field could not be updated.', 'event_espresso'),
@@ -3063,7 +3063,7 @@  discard block
 block discarded – undo
3063 3063
         // incoming GRP_IDs
3064 3064
         if ($all) {
3065 3065
             // Checkboxes
3066
-            if (! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3066
+            if ( ! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3067 3067
                 // if array has more than one element then success message should be plural.
3068 3068
                 // todo: what about nonce?
3069 3069
                 $success = count($this->_req_data['checkbox']) > 1 ? 2 : 1;
@@ -3071,16 +3071,16 @@  discard block
 block discarded – undo
3071 3071
                 // cycle through checkboxes
3072 3072
                 while (list($GRP_ID, $value) = each($this->_req_data['checkbox'])) {
3073 3073
                     $trashed_or_restored = $trash ? $MTP->delete_by_ID($GRP_ID) : $MTP->restore_by_ID($GRP_ID);
3074
-                    if (! $trashed_or_restored) {
3074
+                    if ( ! $trashed_or_restored) {
3075 3075
                         $success = 0;
3076 3076
                     }
3077 3077
                 }
3078 3078
             } else {
3079 3079
                 // grab single GRP_ID and handle
3080 3080
                 $GRP_ID = isset($this->_req_data['id']) ? absint($this->_req_data['id']) : 0;
3081
-                if (! empty($GRP_ID)) {
3081
+                if ( ! empty($GRP_ID)) {
3082 3082
                     $trashed_or_restored = $trash ? $MTP->delete_by_ID($GRP_ID) : $MTP->restore_by_ID($GRP_ID);
3083
-                    if (! $trashed_or_restored) {
3083
+                    if ( ! $trashed_or_restored) {
3084 3084
                         $success = 0;
3085 3085
                     }
3086 3086
                 } else {
@@ -3128,7 +3128,7 @@  discard block
 block discarded – undo
3128 3128
         do_action('AHEE_log', __FILE__, __FUNCTION__, '');
3129 3129
 
3130 3130
         // checkboxes
3131
-        if (! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3131
+        if ( ! empty($this->_req_data['checkbox']) && is_array($this->_req_data['checkbox'])) {
3132 3132
             // if array has more than one element then success message should be plural
3133 3133
             $success = count($this->_req_data['checkbox']) > 1 ? 2 : 1;
3134 3134
 
@@ -3235,7 +3235,7 @@  discard block
 block discarded – undo
3235 3235
     protected function _set_m_mt_settings()
3236 3236
     {
3237 3237
         // first if this is already set then lets get out no need to regenerate data.
3238
-        if (! empty($this->_m_mt_settings)) {
3238
+        if ( ! empty($this->_m_mt_settings)) {
3239 3239
             return;
3240 3240
         }
3241 3241
 
@@ -3249,7 +3249,7 @@  discard block
 block discarded – undo
3249 3249
         // assemble the array for the _tab_text_links helper
3250 3250
 
3251 3251
         foreach ($messengers as $messenger) {
3252
-            $this->_m_mt_settings['messenger_tabs'][ $messenger->name ] = array(
3252
+            $this->_m_mt_settings['messenger_tabs'][$messenger->name] = array(
3253 3253
                 'label' => ucwords($messenger->label['singular']),
3254 3254
                 'class' => $this->_message_resource_manager->is_messenger_active($messenger->name)
3255 3255
                     ? 'messenger-active'
@@ -3266,7 +3266,7 @@  discard block
 block discarded – undo
3266 3266
             foreach ($message_types as $message_type) {
3267 3267
                 // first we need to verify that this message type is valid with this messenger. Cause if it isn't then
3268 3268
                 // it shouldn't show in either the inactive OR active metabox.
3269
-                if (! in_array($message_type->name, $message_types_for_messenger, true)) {
3269
+                if ( ! in_array($message_type->name, $message_types_for_messenger, true)) {
3270 3270
                     continue;
3271 3271
                 }
3272 3272
 
@@ -3277,12 +3277,12 @@  discard block
 block discarded – undo
3277 3277
                     ? 'active'
3278 3278
                     : 'inactive';
3279 3279
 
3280
-                $this->_m_mt_settings['message_type_tabs'][ $messenger->name ][ $a_or_i ][ $message_type->name ] = array(
3280
+                $this->_m_mt_settings['message_type_tabs'][$messenger->name][$a_or_i][$message_type->name] = array(
3281 3281
                     'label'    => ucwords($message_type->label['singular']),
3282
-                    'class'    => 'message-type-' . $a_or_i,
3283
-                    'slug_id'  => $message_type->name . '-messagetype-' . $messenger->name,
3284
-                    'mt_nonce' => wp_create_nonce($message_type->name . '_nonce'),
3285
-                    'href'     => 'espresso_' . $message_type->name . '_message_type_settings',
3282
+                    'class'    => 'message-type-'.$a_or_i,
3283
+                    'slug_id'  => $message_type->name.'-messagetype-'.$messenger->name,
3284
+                    'mt_nonce' => wp_create_nonce($message_type->name.'_nonce'),
3285
+                    'href'     => 'espresso_'.$message_type->name.'_message_type_settings',
3286 3286
                     'title'    => $a_or_i === 'active'
3287 3287
                         ? esc_html__('Drag this message type to the Inactive window to deactivate', 'event_espresso')
3288 3288
                         : esc_html__('Drag this message type to the messenger to activate', 'event_espresso'),
@@ -3313,25 +3313,25 @@  discard block
 block discarded – undo
3313 3313
         $fields = $message_type->get_admin_settings_fields();
3314 3314
         $settings_template_args['template_form_fields'] = '';
3315 3315
 
3316
-        if (! empty($fields) && $active) {
3316
+        if ( ! empty($fields) && $active) {
3317 3317
             $existing_settings = $message_type->get_existing_admin_settings($messenger->name);
3318 3318
             foreach ($fields as $fldname => $fldprops) {
3319
-                $field_id = $messenger->name . '-' . $message_type->name . '-' . $fldname;
3320
-                $template_form_field[ $field_id ] = array(
3321
-                    'name'       => 'message_type_settings[' . $fldname . ']',
3319
+                $field_id = $messenger->name.'-'.$message_type->name.'-'.$fldname;
3320
+                $template_form_field[$field_id] = array(
3321
+                    'name'       => 'message_type_settings['.$fldname.']',
3322 3322
                     'label'      => $fldprops['label'],
3323 3323
                     'input'      => $fldprops['field_type'],
3324 3324
                     'type'       => $fldprops['value_type'],
3325 3325
                     'required'   => $fldprops['required'],
3326 3326
                     'validation' => $fldprops['validation'],
3327
-                    'value'      => isset($existing_settings[ $fldname ])
3328
-                        ? $existing_settings[ $fldname ]
3327
+                    'value'      => isset($existing_settings[$fldname])
3328
+                        ? $existing_settings[$fldname]
3329 3329
                         : $fldprops['default'],
3330 3330
                     'options'    => isset($fldprops['options'])
3331 3331
                         ? $fldprops['options']
3332 3332
                         : array(),
3333
-                    'default'    => isset($existing_settings[ $fldname ])
3334
-                        ? $existing_settings[ $fldname ]
3333
+                    'default'    => isset($existing_settings[$fldname])
3334
+                        ? $existing_settings[$fldname]
3335 3335
                         : $fldprops['default'],
3336 3336
                     'css_class'  => 'no-drag',
3337 3337
                     'format'     => $fldprops['format'],
@@ -3374,7 +3374,7 @@  discard block
 block discarded – undo
3374 3374
             : '';
3375 3375
 
3376 3376
 
3377
-        $template = EE_MSG_TEMPLATE_PATH . 'ee_msg_mt_settings_content.template.php';
3377
+        $template = EE_MSG_TEMPLATE_PATH.'ee_msg_mt_settings_content.template.php';
3378 3378
         $content = EEH_Template::display_template($template, $settings_template_args, true);
3379 3379
 
3380 3380
         return $content;
@@ -3405,20 +3405,20 @@  discard block
 block discarded – undo
3405 3405
                 // messenger meta boxes
3406 3406
                 $active = $selected_messenger === $messenger;
3407 3407
                 $active_mt_tabs = isset(
3408
-                    $this->_m_mt_settings['message_type_tabs'][ $messenger ]['active']
3408
+                    $this->_m_mt_settings['message_type_tabs'][$messenger]['active']
3409 3409
                 )
3410
-                    ? $this->_m_mt_settings['message_type_tabs'][ $messenger ]['active']
3410
+                    ? $this->_m_mt_settings['message_type_tabs'][$messenger]['active']
3411 3411
                     : '';
3412
-                $m_boxes[ $messenger . '_a_box' ] = sprintf(
3412
+                $m_boxes[$messenger.'_a_box'] = sprintf(
3413 3413
                     esc_html__('%s Settings', 'event_espresso'),
3414 3414
                     $tab_array['label']
3415 3415
                 );
3416
-                $m_template_args[ $messenger . '_a_box' ] = array(
3416
+                $m_template_args[$messenger.'_a_box'] = array(
3417 3417
                     'active_message_types'   => ! empty($active_mt_tabs) ? $this->_get_mt_tabs($active_mt_tabs) : '',
3418 3418
                     'inactive_message_types' => isset(
3419
-                        $this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive']
3419
+                        $this->_m_mt_settings['message_type_tabs'][$messenger]['inactive']
3420 3420
                     )
3421
-                        ? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive'])
3421
+                        ? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][$messenger]['inactive'])
3422 3422
                         : '',
3423 3423
                     'content'                => $this->_get_messenger_box_content($tab_array['obj']),
3424 3424
                     'hidden'                 => $active ? '' : ' hidden',
@@ -3429,13 +3429,13 @@  discard block
 block discarded – undo
3429 3429
                 // message type meta boxes
3430 3430
                 // (which is really just the inactive container for each messenger
3431 3431
                 // showing inactive message types for that messenger)
3432
-                $mt_boxes[ $messenger . '_i_box' ] = esc_html__('Inactive Message Types', 'event_espresso');
3433
-                $mt_template_args[ $messenger . '_i_box' ] = array(
3432
+                $mt_boxes[$messenger.'_i_box'] = esc_html__('Inactive Message Types', 'event_espresso');
3433
+                $mt_template_args[$messenger.'_i_box'] = array(
3434 3434
                     'active_message_types'   => ! empty($active_mt_tabs) ? $this->_get_mt_tabs($active_mt_tabs) : '',
3435 3435
                     'inactive_message_types' => isset(
3436
-                        $this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive']
3436
+                        $this->_m_mt_settings['message_type_tabs'][$messenger]['inactive']
3437 3437
                     )
3438
-                        ? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][ $messenger ]['inactive'])
3438
+                        ? $this->_get_mt_tabs($this->_m_mt_settings['message_type_tabs'][$messenger]['inactive'])
3439 3439
                         : '',
3440 3440
                     'hidden'                 => $active ? '' : ' hidden',
3441 3441
                     'hide_on_message'        => $hide_on_message,
@@ -3448,14 +3448,14 @@  discard block
 block discarded – undo
3448 3448
 
3449 3449
 
3450 3450
         // register messenger metaboxes
3451
-        $m_template_path = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_messenger_mt_meta_box.template.php';
3451
+        $m_template_path = EE_MSG_TEMPLATE_PATH.'ee_msg_details_messenger_mt_meta_box.template.php';
3452 3452
         foreach ($m_boxes as $box => $label) {
3453
-            $callback_args = array('template_path' => $m_template_path, 'template_args' => $m_template_args[ $box ]);
3453
+            $callback_args = array('template_path' => $m_template_path, 'template_args' => $m_template_args[$box]);
3454 3454
             $msgr = str_replace('_a_box', '', $box);
3455 3455
             add_meta_box(
3456
-                'espresso_' . $msgr . '_settings',
3456
+                'espresso_'.$msgr.'_settings',
3457 3457
                 $label,
3458
-                function ($post, $metabox) {
3458
+                function($post, $metabox) {
3459 3459
                     echo EEH_Template::display_template(
3460 3460
                         $metabox["args"]["template_path"],
3461 3461
                         $metabox["args"]["template_args"],
@@ -3470,17 +3470,17 @@  discard block
 block discarded – undo
3470 3470
         }
3471 3471
 
3472 3472
         // register message type metaboxes
3473
-        $mt_template_path = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_messenger_meta_box.template.php';
3473
+        $mt_template_path = EE_MSG_TEMPLATE_PATH.'ee_msg_details_messenger_meta_box.template.php';
3474 3474
         foreach ($mt_boxes as $box => $label) {
3475 3475
             $callback_args = array(
3476 3476
                 'template_path' => $mt_template_path,
3477
-                'template_args' => $mt_template_args[ $box ],
3477
+                'template_args' => $mt_template_args[$box],
3478 3478
             );
3479 3479
             $mt = str_replace('_i_box', '', $box);
3480 3480
             add_meta_box(
3481
-                'espresso_' . $mt . '_inactive_mts',
3481
+                'espresso_'.$mt.'_inactive_mts',
3482 3482
                 $label,
3483
-                function ($post, $metabox) {
3483
+                function($post, $metabox) {
3484 3484
                     echo EEH_Template::display_template(
3485 3485
                         $metabox["args"]["template_path"],
3486 3486
                         $metabox["args"]["template_args"],
@@ -3626,7 +3626,7 @@  discard block
 block discarded – undo
3626 3626
             if ($form->is_valid()) {
3627 3627
                 $valid_data = $form->valid_data();
3628 3628
                 foreach ($valid_data as $property => $value) {
3629
-                    $setter = 'set_' . $property;
3629
+                    $setter = 'set_'.$property;
3630 3630
                     if (method_exists($network_config, $setter)) {
3631 3631
                         $network_config->{$setter}($value);
3632 3632
                     } elseif (property_exists($network_config, $property)
@@ -3660,7 +3660,7 @@  discard block
 block discarded – undo
3660 3660
     protected function _get_mt_tabs($tab_array)
3661 3661
     {
3662 3662
         $tab_array = (array) $tab_array;
3663
-        $template = EE_MSG_TEMPLATE_PATH . 'ee_msg_details_mt_settings_tab_item.template.php';
3663
+        $template = EE_MSG_TEMPLATE_PATH.'ee_msg_details_mt_settings_tab_item.template.php';
3664 3664
         $tabs = '';
3665 3665
 
3666 3666
         foreach ($tab_array as $tab) {
@@ -3688,20 +3688,20 @@  discard block
 block discarded – undo
3688 3688
         $settings_template_args['active'] = $this->_message_resource_manager->is_messenger_active($messenger->name);
3689 3689
 
3690 3690
 
3691
-        if (! empty($fields)) {
3691
+        if ( ! empty($fields)) {
3692 3692
             $existing_settings = $messenger->get_existing_admin_settings();
3693 3693
 
3694 3694
             foreach ($fields as $fldname => $fldprops) {
3695
-                $field_id = $messenger->name . '-' . $fldname;
3696
-                $template_form_field[ $field_id ] = array(
3697
-                    'name'       => 'messenger_settings[' . $field_id . ']',
3695
+                $field_id = $messenger->name.'-'.$fldname;
3696
+                $template_form_field[$field_id] = array(
3697
+                    'name'       => 'messenger_settings['.$field_id.']',
3698 3698
                     'label'      => $fldprops['label'],
3699 3699
                     'input'      => $fldprops['field_type'],
3700 3700
                     'type'       => $fldprops['value_type'],
3701 3701
                     'required'   => $fldprops['required'],
3702 3702
                     'validation' => $fldprops['validation'],
3703
-                    'value'      => isset($existing_settings[ $field_id ])
3704
-                        ? $existing_settings[ $field_id ]
3703
+                    'value'      => isset($existing_settings[$field_id])
3704
+                        ? $existing_settings[$field_id]
3705 3705
                         : $fldprops['default'],
3706 3706
                     'css_class'  => '',
3707 3707
                     'format'     => $fldprops['format'],
@@ -3727,9 +3727,9 @@  discard block
 block discarded – undo
3727 3727
         );
3728 3728
 
3729 3729
         // make sure any active message types that are existing are included in the hidden fields
3730
-        if (isset($this->_m_mt_settings['message_type_tabs'][ $messenger->name ]['active'])) {
3731
-            foreach ($this->_m_mt_settings['message_type_tabs'][ $messenger->name ]['active'] as $mt => $values) {
3732
-                $settings_template_args['hidden_fields'][ 'messenger_settings[message_types][' . $mt . ']' ] = array(
3730
+        if (isset($this->_m_mt_settings['message_type_tabs'][$messenger->name]['active'])) {
3731
+            foreach ($this->_m_mt_settings['message_type_tabs'][$messenger->name]['active'] as $mt => $values) {
3732
+                $settings_template_args['hidden_fields']['messenger_settings[message_types]['.$mt.']'] = array(
3733 3733
                     'type'  => 'hidden',
3734 3734
                     'value' => $mt,
3735 3735
                 );
@@ -3758,9 +3758,9 @@  discard block
 block discarded – undo
3758 3758
 
3759 3759
 
3760 3760
         $settings_template_args['on_off_action'] = $active ? 'messenger-off' : 'messenger-on';
3761
-        $settings_template_args['nonce'] = wp_create_nonce('activate_' . $messenger->name . '_toggle_nonce');
3761
+        $settings_template_args['nonce'] = wp_create_nonce('activate_'.$messenger->name.'_toggle_nonce');
3762 3762
         $settings_template_args['on_off_status'] = $active ? true : false;
3763
-        $template = EE_MSG_TEMPLATE_PATH . 'ee_msg_m_settings_content.template.php';
3763
+        $template = EE_MSG_TEMPLATE_PATH.'ee_msg_m_settings_content.template.php';
3764 3764
         $content = EEH_Template::display_template(
3765 3765
             $template,
3766 3766
             $settings_template_args,
@@ -3786,7 +3786,7 @@  discard block
 block discarded – undo
3786 3786
         $success = true;
3787 3787
         $this->_prep_default_response_for_messenger_or_message_type_toggle();
3788 3788
         // let's check that we have required data
3789
-        if (! isset($this->_req_data['messenger'])) {
3789
+        if ( ! isset($this->_req_data['messenger'])) {
3790 3790
             EE_Error::add_error(
3791 3791
                 esc_html__('Messenger name needed to toggle activation. None given', 'event_espresso'),
3792 3792
                 __FILE__,
@@ -3800,12 +3800,12 @@  discard block
 block discarded – undo
3800 3800
         $nonce = isset($this->_req_data['activate_nonce'])
3801 3801
             ? sanitize_text_field($this->_req_data['activate_nonce'])
3802 3802
             : '';
3803
-        $nonce_ref = 'activate_' . $this->_req_data['messenger'] . '_toggle_nonce';
3803
+        $nonce_ref = 'activate_'.$this->_req_data['messenger'].'_toggle_nonce';
3804 3804
 
3805 3805
         $this->_verify_nonce($nonce, $nonce_ref);
3806 3806
 
3807 3807
 
3808
-        if (! isset($this->_req_data['status'])) {
3808
+        if ( ! isset($this->_req_data['status'])) {
3809 3809
             EE_Error::add_error(
3810 3810
                 esc_html__(
3811 3811
                     'Messenger status needed to know whether activation or deactivation is happening. No status is given',
@@ -3864,7 +3864,7 @@  discard block
 block discarded – undo
3864 3864
         $this->_prep_default_response_for_messenger_or_message_type_toggle();
3865 3865
 
3866 3866
         // let's make sure we have the necessary data
3867
-        if (! isset($this->_req_data['message_type'])) {
3867
+        if ( ! isset($this->_req_data['message_type'])) {
3868 3868
             EE_Error::add_error(
3869 3869
                 esc_html__('Message Type name needed to toggle activation. None given', 'event_espresso'),
3870 3870
                 __FILE__,
@@ -3874,7 +3874,7 @@  discard block
 block discarded – undo
3874 3874
             $success = false;
3875 3875
         }
3876 3876
 
3877
-        if (! isset($this->_req_data['messenger'])) {
3877
+        if ( ! isset($this->_req_data['messenger'])) {
3878 3878
             EE_Error::add_error(
3879 3879
                 esc_html__('Messenger name needed to toggle activation. None given', 'event_espresso'),
3880 3880
                 __FILE__,
@@ -3884,7 +3884,7 @@  discard block
 block discarded – undo
3884 3884
             $success = false;
3885 3885
         }
3886 3886
 
3887
-        if (! isset($this->_req_data['status'])) {
3887
+        if ( ! isset($this->_req_data['status'])) {
3888 3888
             EE_Error::add_error(
3889 3889
                 esc_html__(
3890 3890
                     'Messenger status needed to know whether activation or deactivation is happening. No status is given',
@@ -3917,7 +3917,7 @@  discard block
 block discarded – undo
3917 3917
 
3918 3918
         // do a nonce check here since we're not arriving via a normal route
3919 3919
         $nonce = isset($this->_req_data['mt_nonce']) ? sanitize_text_field($this->_req_data['mt_nonce']) : '';
3920
-        $nonce_ref = $this->_req_data['message_type'] . '_nonce';
3920
+        $nonce_ref = $this->_req_data['message_type'].'_nonce';
3921 3921
 
3922 3922
         $this->_verify_nonce($nonce, $nonce_ref);
3923 3923
 
@@ -4104,7 +4104,7 @@  discard block
 block discarded – undo
4104 4104
         EE_Message_Type $message_type = null
4105 4105
     ) {
4106 4106
         // if $messenger isn't a valid messenger object then get out.
4107
-        if (! $messenger instanceof EE_Messenger) {
4107
+        if ( ! $messenger instanceof EE_Messenger) {
4108 4108
             EE_Error::add_error(
4109 4109
                 esc_html__('The messenger being activated is not a valid messenger', 'event_espresso'),
4110 4110
                 __FILE__,
@@ -4159,7 +4159,7 @@  discard block
 block discarded – undo
4159 4159
             // message types after the activation process.  However its possible some messengers don't HAVE any default_message_types
4160 4160
             // in which case we just give a success message for the messenger being successfully activated.
4161 4161
         } else {
4162
-            if (! $messenger->get_default_message_types()) {
4162
+            if ( ! $messenger->get_default_message_types()) {
4163 4163
                 // messenger doesn't have any default message types so still a success.
4164 4164
                 EE_Error::add_success(
4165 4165
                     sprintf(
@@ -4215,7 +4215,7 @@  discard block
 block discarded – undo
4215 4215
         EE_Error::overwrite_success();
4216 4216
 
4217 4217
         // if $messenger isn't a valid messenger object then get out.
4218
-        if (! $messenger instanceof EE_Messenger) {
4218
+        if ( ! $messenger instanceof EE_Messenger) {
4219 4219
             EE_Error::add_error(
4220 4220
                 esc_html__('The messenger being deactivated is not a valid messenger', 'event_espresso'),
4221 4221
                 __FILE__,
@@ -4277,7 +4277,7 @@  discard block
 block discarded – undo
4277 4277
      */
4278 4278
     public function update_mt_form()
4279 4279
     {
4280
-        if (! isset($this->_req_data['messenger']) || ! isset($this->_req_data['message_type'])) {
4280
+        if ( ! isset($this->_req_data['messenger']) || ! isset($this->_req_data['message_type'])) {
4281 4281
             EE_Error::add_error(
4282 4282
                 esc_html__('Require message type or messenger to send an updated form', 'event_espresso'),
4283 4283
                 __FILE__,
@@ -4289,7 +4289,7 @@  discard block
 block discarded – undo
4289 4289
 
4290 4290
         $message_types = $this->get_installed_message_types();
4291 4291
 
4292
-        $message_type = $message_types[ $this->_req_data['message_type'] ];
4292
+        $message_type = $message_types[$this->_req_data['message_type']];
4293 4293
         $messenger = $this->_message_resource_manager->get_active_messenger($this->_req_data['messenger']);
4294 4294
 
4295 4295
         $content = $this->_message_type_settings_content(
@@ -4309,7 +4309,7 @@  discard block
 block discarded – undo
4309 4309
      */
4310 4310
     public function save_settings()
4311 4311
     {
4312
-        if (! isset($this->_req_data['type'])) {
4312
+        if ( ! isset($this->_req_data['type'])) {
4313 4313
             EE_Error::add_error(
4314 4314
                 esc_html__(
4315 4315
                     'Cannot save settings because type is unknown (messenger settings or messsage type settings?)',
@@ -4338,7 +4338,7 @@  discard block
 block discarded – undo
4338 4338
                         unset($settings['message_types']);
4339 4339
                         break;
4340 4340
                     default:
4341
-                        $settings[ $key ] = $value;
4341
+                        $settings[$key] = $value;
4342 4342
                         break;
4343 4343
                 }
4344 4344
             }
@@ -4357,7 +4357,7 @@  discard block
 block discarded – undo
4357 4357
                         unset($settings['message_type']);
4358 4358
                         break;
4359 4359
                     default:
4360
-                        $settings[ $key ] = $value;
4360
+                        $settings[$key] = $value;
4361 4361
                         break;
4362 4362
                 }
4363 4363
             }
@@ -4518,7 +4518,7 @@  discard block
 block discarded – undo
4518 4518
      */
4519 4519
     protected function _get_msg_ids_from_request()
4520 4520
     {
4521
-        if (! isset($this->_req_data['MSG_ID'])) {
4521
+        if ( ! isset($this->_req_data['MSG_ID'])) {
4522 4522
             return array();
4523 4523
         }
4524 4524
 
Please login to merge, or discard this patch.
core/helpers/EEH_Line_Item.helper.php 3 patches
Doc Comments   +2 added lines, -1 removed lines patch added patch discarded remove patch
@@ -874,7 +874,7 @@  discard block
 block discarded – undo
874 874
      * Deletes ALL children of the passed line item
875 875
      *
876 876
      * @param EE_Line_Item $parent_line_item
877
-     * @return bool
877
+     * @return integer
878 878
      * @throws \EE_Error
879 879
      */
880 880
     public static function delete_all_child_items(EE_Line_Item $parent_line_item)
@@ -1138,6 +1138,7 @@  discard block
 block discarded – undo
1138 1138
      * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1139 1139
      * @param string $line_item_type one of the EEM_Line_Item constants
1140 1140
      * @param string | NULL $obj_type object model class name (minus prefix) or NULL to ignore object type when searching
1141
+     * @param string $obj_type
1141 1142
      * @return EE_Line_Item[]
1142 1143
      */
1143 1144
     protected static function _get_descendants_by_type_and_object_type(
Please login to merge, or discard this patch.
Indentation   +1685 added lines, -1685 removed lines patch added patch discarded remove patch
@@ -20,1689 +20,1689 @@
 block discarded – undo
20 20
 class EEH_Line_Item
21 21
 {
22 22
 
23
-    // other functions: cancel ticket purchase
24
-    // delete ticket purchase
25
-    // add promotion
26
-
27
-
28
-    /**
29
-     * Adds a simple item (unrelated to any other model object) to the provided PARENT line item.
30
-     * Does NOT automatically re-calculate the line item totals or update the related transaction.
31
-     * You should call recalculate_total_including_taxes() on the grant total line item after this
32
-     * to update the subtotals, and EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
33
-     * to keep the registration final prices in-sync with the transaction's total.
34
-     *
35
-     * @param EE_Line_Item $parent_line_item
36
-     * @param string $name
37
-     * @param float $unit_price
38
-     * @param string $description
39
-     * @param int $quantity
40
-     * @param boolean $taxable
41
-     * @param boolean $code if set to a value, ensures there is only one line item with that code
42
-     * @return boolean success
43
-     * @throws \EE_Error
44
-     */
45
-    public static function add_unrelated_item(EE_Line_Item $parent_line_item, $name, $unit_price, $description = '', $quantity = 1, $taxable = false, $code = null)
46
-    {
47
-        $items_subtotal = self::get_pre_tax_subtotal($parent_line_item);
48
-        $line_item = EE_Line_Item::new_instance(array(
49
-            'LIN_name' => $name,
50
-            'LIN_desc' => $description,
51
-            'LIN_unit_price' => $unit_price,
52
-            'LIN_quantity' => $quantity,
53
-            'LIN_percent' => null,
54
-            'LIN_is_taxable' => $taxable,
55
-            'LIN_order' => $items_subtotal instanceof EE_Line_Item ? count($items_subtotal->children()) : 0,
56
-            'LIN_total' => (float) $unit_price * (int) $quantity,
57
-            'LIN_type' => EEM_Line_Item::type_line_item,
58
-            'LIN_code' => $code,
59
-        ));
60
-        $line_item = apply_filters(
61
-            'FHEE__EEH_Line_Item__add_unrelated_item__line_item',
62
-            $line_item,
63
-            $parent_line_item
64
-        );
65
-        return self::add_item($parent_line_item, $line_item);
66
-    }
67
-
68
-
69
-    /**
70
-     * Adds a simple item ( unrelated to any other model object) to the total line item,
71
-     * in the correct spot in the line item tree. Automatically
72
-     * re-calculates the line item totals and updates the related transaction. But
73
-     * DOES NOT automatically upgrade the transaction's registrations' final prices (which
74
-     * should probably change because of this).
75
-     * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
76
-     * after using this, to keep the registration final prices in-sync with the transaction's total.
77
-     *
78
-     * @param EE_Line_Item $parent_line_item
79
-     * @param string $name
80
-     * @param float $percentage_amount
81
-     * @param string $description
82
-     * @param boolean $taxable
83
-     * @return boolean success
84
-     * @throws \EE_Error
85
-     */
86
-    public static function add_percentage_based_item(EE_Line_Item $parent_line_item, $name, $percentage_amount, $description = '', $taxable = false)
87
-    {
88
-        $line_item = EE_Line_Item::new_instance(array(
89
-            'LIN_name' => $name,
90
-            'LIN_desc' => $description,
91
-            'LIN_unit_price' => 0,
92
-            'LIN_percent' => $percentage_amount,
93
-            'LIN_quantity' => 1,
94
-            'LIN_is_taxable' => $taxable,
95
-            'LIN_total' => (float) ($percentage_amount * ($parent_line_item->total() / 100)),
96
-            'LIN_type' => EEM_Line_Item::type_line_item,
97
-            'LIN_parent' => $parent_line_item->ID()
98
-        ));
99
-        $line_item = apply_filters(
100
-            'FHEE__EEH_Line_Item__add_percentage_based_item__line_item',
101
-            $line_item
102
-        );
103
-        return $parent_line_item->add_child_line_item($line_item, false);
104
-    }
105
-
106
-
107
-    /**
108
-     * Returns the new line item created by adding a purchase of the ticket
109
-     * ensures that ticket line item is saved, and that cart total has been recalculated.
110
-     * If this ticket has already been purchased, just increments its count.
111
-     * Automatically re-calculates the line item totals and updates the related transaction. But
112
-     * DOES NOT automatically upgrade the transaction's registrations' final prices (which
113
-     * should probably change because of this).
114
-     * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
115
-     * after using this, to keep the registration final prices in-sync with the transaction's total.
116
-     *
117
-     * @param EE_Line_Item $total_line_item grand total line item of type EEM_Line_Item::type_total
118
-     * @param EE_Ticket $ticket
119
-     * @param int $qty
120
-     * @return \EE_Line_Item
121
-     * @throws \EE_Error
122
-     */
123
-    public static function add_ticket_purchase(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1)
124
-    {
125
-        if (!$total_line_item instanceof EE_Line_Item || !$total_line_item->is_total()) {
126
-            throw new EE_Error(sprintf(__('A valid line item total is required in order to add tickets. A line item of type "%s" was passed.', 'event_espresso'), $ticket->ID(), $total_line_item->ID()));
127
-        }
128
-        // either increment the qty for an existing ticket
129
-        $line_item = self::increment_ticket_qty_if_already_in_cart($total_line_item, $ticket, $qty);
130
-        // or add a new one
131
-        if (!$line_item instanceof EE_Line_Item) {
132
-            $line_item = self::create_ticket_line_item($total_line_item, $ticket, $qty);
133
-        }
134
-        $total_line_item->recalculate_total_including_taxes();
135
-        return $line_item;
136
-    }
137
-
138
-
139
-    /**
140
-     * Returns the new line item created by adding a purchase of the ticket
141
-     * @param \EE_Line_Item $total_line_item
142
-     * @param EE_Ticket $ticket
143
-     * @param int $qty
144
-     * @return \EE_Line_Item
145
-     * @throws \EE_Error
146
-     */
147
-    public static function increment_ticket_qty_if_already_in_cart(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1)
148
-    {
149
-        $line_item = null;
150
-        if ($total_line_item instanceof EE_Line_Item && $total_line_item->is_total()) {
151
-            $ticket_line_items = EEH_Line_Item::get_ticket_line_items($total_line_item);
152
-            foreach ((array) $ticket_line_items as $ticket_line_item) {
153
-                if ($ticket_line_item instanceof EE_Line_Item
154
-                    && (int) $ticket_line_item->OBJ_ID() === (int) $ticket->ID()
155
-                ) {
156
-                    $line_item = $ticket_line_item;
157
-                    break;
158
-                }
159
-            }
160
-        }
161
-        if ($line_item instanceof EE_Line_Item) {
162
-            EEH_Line_Item::increment_quantity($line_item, $qty);
163
-            return $line_item;
164
-        }
165
-        return null;
166
-    }
167
-
168
-
169
-    /**
170
-     * Increments the line item and all its children's quantity by $qty (but percent line items are unaffected).
171
-     * Does NOT save or recalculate other line items totals
172
-     *
173
-     * @param EE_Line_Item $line_item
174
-     * @param int $qty
175
-     * @return void
176
-     * @throws \EE_Error
177
-     */
178
-    public static function increment_quantity(EE_Line_Item $line_item, $qty = 1)
179
-    {
180
-        if (!$line_item->is_percent()) {
181
-            $qty += $line_item->quantity();
182
-            $line_item->set_quantity($qty);
183
-            $line_item->set_total($line_item->unit_price() * $qty);
184
-            $line_item->save();
185
-        }
186
-        foreach ($line_item->children() as $child) {
187
-            if ($child->is_sub_line_item()) {
188
-                EEH_Line_Item::update_quantity($child, $qty);
189
-            }
190
-        }
191
-    }
192
-
193
-
194
-    /**
195
-     * Decrements the line item and all its children's quantity by $qty (but percent line items are unaffected).
196
-     * Does NOT save or recalculate other line items totals
197
-     *
198
-     * @param EE_Line_Item $line_item
199
-     * @param int $qty
200
-     * @return void
201
-     * @throws \EE_Error
202
-     */
203
-    public static function decrement_quantity(EE_Line_Item $line_item, $qty = 1)
204
-    {
205
-        if (!$line_item->is_percent()) {
206
-            $qty = $line_item->quantity() - $qty;
207
-            $qty = max($qty, 0);
208
-            $line_item->set_quantity($qty);
209
-            $line_item->set_total($line_item->unit_price() * $qty);
210
-            $line_item->save();
211
-        }
212
-        foreach ($line_item->children() as $child) {
213
-            if ($child->is_sub_line_item()) {
214
-                EEH_Line_Item::update_quantity($child, $qty);
215
-            }
216
-        }
217
-    }
218
-
219
-
220
-    /**
221
-     * Updates the line item and its children's quantities to the specified number.
222
-     * Does NOT save them or recalculate totals.
223
-     *
224
-     * @param EE_Line_Item $line_item
225
-     * @param int $new_quantity
226
-     * @throws \EE_Error
227
-     */
228
-    public static function update_quantity(EE_Line_Item $line_item, $new_quantity)
229
-    {
230
-        if (!$line_item->is_percent()) {
231
-            $line_item->set_quantity($new_quantity);
232
-            $line_item->set_total($line_item->unit_price() * $new_quantity);
233
-            $line_item->save();
234
-        }
235
-        foreach ($line_item->children() as $child) {
236
-            if ($child->is_sub_line_item()) {
237
-                EEH_Line_Item::update_quantity($child, $new_quantity);
238
-            }
239
-        }
240
-    }
241
-
242
-
243
-    /**
244
-     * Returns the new line item created by adding a purchase of the ticket
245
-     * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
246
-     * @param EE_Ticket $ticket
247
-     * @param int $qty
248
-     * @return \EE_Line_Item
249
-     * @throws \EE_Error
250
-     */
251
-    public static function create_ticket_line_item(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1)
252
-    {
253
-        $datetimes = $ticket->datetimes();
254
-        $first_datetime = reset($datetimes);
255
-        if ($first_datetime instanceof EE_Datetime && $first_datetime->event() instanceof EE_Event) {
256
-            $first_datetime_name = $first_datetime->event()->name();
257
-        } else {
258
-            $first_datetime_name = __('Event', 'event_espresso');
259
-        }
260
-        $event = sprintf(_x('(For %1$s)', '(For Event Name)', 'event_espresso'), $first_datetime_name);
261
-        // get event subtotal line
262
-        $events_sub_total = self::get_event_line_item_for_ticket($total_line_item, $ticket);
263
-        // add $ticket to cart
264
-        $line_item = EE_Line_Item::new_instance(array(
265
-            'LIN_name' => $ticket->name(),
266
-            'LIN_desc' => $ticket->description() !== '' ? $ticket->description() . ' ' . $event : $event,
267
-            'LIN_unit_price' => $ticket->price(),
268
-            'LIN_quantity' => $qty,
269
-            'LIN_is_taxable' => $ticket->taxable(),
270
-            'LIN_order' => count($events_sub_total->children()),
271
-            'LIN_total' => $ticket->price() * $qty,
272
-            'LIN_type' => EEM_Line_Item::type_line_item,
273
-            'OBJ_ID' => $ticket->ID(),
274
-            'OBJ_type' => 'Ticket'
275
-        ));
276
-        $line_item = apply_filters(
277
-            'FHEE__EEH_Line_Item__create_ticket_line_item__line_item',
278
-            $line_item
279
-        );
280
-        $events_sub_total->add_child_line_item($line_item);
281
-        // now add the sub-line items
282
-        $running_total_for_ticket = 0;
283
-        foreach ($ticket->prices(array('order_by' => array('PRC_order' => 'ASC'))) as $price) {
284
-            $sign = $price->is_discount() ? -1 : 1;
285
-            $price_total = $price->is_percent()
286
-                ? $running_total_for_ticket * $price->amount() / 100
287
-                : $price->amount() * $qty;
288
-            $sub_line_item = EE_Line_Item::new_instance(array(
289
-                'LIN_name' => $price->name(),
290
-                'LIN_desc' => $price->desc(),
291
-                'LIN_quantity' => $price->is_percent() ? null : $qty,
292
-                'LIN_is_taxable' => false,
293
-                'LIN_order' => $price->order(),
294
-                'LIN_total' => $sign * $price_total,
295
-                'LIN_type' => EEM_Line_Item::type_sub_line_item,
296
-                'OBJ_ID' => $price->ID(),
297
-                'OBJ_type' => 'Price'
298
-            ));
299
-            $sub_line_item = apply_filters(
300
-                'FHEE__EEH_Line_Item__create_ticket_line_item__sub_line_item',
301
-                $sub_line_item
302
-            );
303
-            if ($price->is_percent()) {
304
-                $sub_line_item->set_percent($sign * $price->amount());
305
-            } else {
306
-                $sub_line_item->set_unit_price($sign * $price->amount());
307
-            }
308
-            $running_total_for_ticket += $price_total;
309
-            $line_item->add_child_line_item($sub_line_item);
310
-        }
311
-        return $line_item;
312
-    }
313
-
314
-
315
-    /**
316
-     * Adds the specified item under the pre-tax-sub-total line item. Automatically
317
-     * re-calculates the line item totals and updates the related transaction. But
318
-     * DOES NOT automatically upgrade the transaction's registrations' final prices (which
319
-     * should probably change because of this).
320
-     * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
321
-     * after using this, to keep the registration final prices in-sync with the transaction's total.
322
-     *
323
-     * @param EE_Line_Item $total_line_item
324
-     * @param EE_Line_Item $item to be added
325
-     * @return boolean
326
-     * @throws \EE_Error
327
-     */
328
-    public static function add_item(EE_Line_Item $total_line_item, EE_Line_Item $item)
329
-    {
330
-        $pre_tax_subtotal = self::get_pre_tax_subtotal($total_line_item);
331
-        if ($pre_tax_subtotal instanceof EE_Line_Item) {
332
-            $success = $pre_tax_subtotal->add_child_line_item($item);
333
-        } else {
334
-            return false;
335
-        }
336
-        $total_line_item->recalculate_total_including_taxes();
337
-        return $success;
338
-    }
339
-
340
-
341
-    /**
342
-     * cancels an existing ticket line item,
343
-     * by decrementing it's quantity by 1 and adding a new "type_cancellation" sub-line-item.
344
-     * ALL totals and subtotals will NEED TO BE UPDATED after performing this action
345
-     *
346
-     * @param EE_Line_Item $ticket_line_item
347
-     * @param int $qty
348
-     * @return bool success
349
-     * @throws \EE_Error
350
-     */
351
-    public static function cancel_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1)
352
-    {
353
-        // validate incoming line_item
354
-        if ($ticket_line_item->OBJ_type() !== 'Ticket') {
355
-            throw new EE_Error(
356
-                sprintf(
357
-                    __('The supplied line item must have an Object Type of "Ticket", not %1$s.', 'event_espresso'),
358
-                    $ticket_line_item->type()
359
-                )
360
-            );
361
-        }
362
-        if ($ticket_line_item->quantity() < $qty) {
363
-            throw new EE_Error(
364
-                sprintf(
365
-                    __('Can not cancel %1$d ticket(s) because the supplied line item has a quantity of %2$d.', 'event_espresso'),
366
-                    $qty,
367
-                    $ticket_line_item->quantity()
368
-                )
369
-            );
370
-        }
371
-        // decrement ticket quantity; don't rely on auto-fixing when recalculating totals to do this
372
-        $ticket_line_item->set_quantity($ticket_line_item->quantity() - $qty);
373
-        foreach ($ticket_line_item->children() as $child_line_item) {
374
-            if ($child_line_item->is_sub_line_item()
375
-                && !$child_line_item->is_percent()
376
-                && !$child_line_item->is_cancellation()
377
-            ) {
378
-                $child_line_item->set_quantity($child_line_item->quantity() - $qty);
379
-            }
380
-        }
381
-        // get cancellation sub line item
382
-        $cancellation_line_item = EEH_Line_Item::get_descendants_of_type(
383
-            $ticket_line_item,
384
-            EEM_Line_Item::type_cancellation
385
-        );
386
-        $cancellation_line_item = reset($cancellation_line_item);
387
-        // verify that this ticket was indeed previously cancelled
388
-        if ($cancellation_line_item instanceof EE_Line_Item) {
389
-            // increment cancelled quantity
390
-            $cancellation_line_item->set_quantity($cancellation_line_item->quantity() + $qty);
391
-        } else {
392
-            // create cancellation sub line item
393
-            $cancellation_line_item = EE_Line_Item::new_instance(array(
394
-                'LIN_name' => __('Cancellation', 'event_espresso'),
395
-                'LIN_desc' => sprintf(
396
-                    _x('Cancelled %1$s : %2$s', 'Cancelled Ticket Name : 2015-01-01 11:11', 'event_espresso'),
397
-                    $ticket_line_item->name(),
398
-                    current_time(get_option('date_format') . ' ' . get_option('time_format'))
399
-                ),
400
-                'LIN_unit_price' => 0, // $ticket_line_item->unit_price()
401
-                'LIN_quantity' => $qty,
402
-                'LIN_is_taxable' => $ticket_line_item->is_taxable(),
403
-                'LIN_order' => count($ticket_line_item->children()),
404
-                'LIN_total' => 0, // $ticket_line_item->unit_price()
405
-                'LIN_type' => EEM_Line_Item::type_cancellation,
406
-            ));
407
-            $ticket_line_item->add_child_line_item($cancellation_line_item);
408
-        }
409
-        if ($ticket_line_item->save_this_and_descendants() > 0) {
410
-            // decrement parent line item quantity
411
-            $event_line_item = $ticket_line_item->parent();
412
-            if ($event_line_item instanceof EE_Line_Item && $event_line_item->OBJ_type() === 'Event') {
413
-                $event_line_item->set_quantity($event_line_item->quantity() - $qty);
414
-                $event_line_item->save();
415
-            }
416
-            EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item);
417
-            return true;
418
-        }
419
-        return false;
420
-    }
421
-
422
-
423
-    /**
424
-     * reinstates (un-cancels?) a previously canceled ticket line item,
425
-     * by incrementing it's quantity by 1, and decrementing it's "type_cancellation" sub-line-item.
426
-     * ALL totals and subtotals will NEED TO BE UPDATED after performing this action
427
-     *
428
-     * @param EE_Line_Item $ticket_line_item
429
-     * @param int $qty
430
-     * @return bool success
431
-     * @throws \EE_Error
432
-     */
433
-    public static function reinstate_canceled_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1)
434
-    {
435
-        // validate incoming line_item
436
-        if ($ticket_line_item->OBJ_type() !== 'Ticket') {
437
-            throw new EE_Error(
438
-                sprintf(
439
-                    __('The supplied line item must have an Object Type of "Ticket", not %1$s.', 'event_espresso'),
440
-                    $ticket_line_item->type()
441
-                )
442
-            );
443
-        }
444
-        // get cancellation sub line item
445
-        $cancellation_line_item = EEH_Line_Item::get_descendants_of_type(
446
-            $ticket_line_item,
447
-            EEM_Line_Item::type_cancellation
448
-        );
449
-        $cancellation_line_item = reset($cancellation_line_item);
450
-        // verify that this ticket was indeed previously cancelled
451
-        if (!$cancellation_line_item instanceof EE_Line_Item) {
452
-            return false;
453
-        }
454
-        if ($cancellation_line_item->quantity() > $qty) {
455
-            // decrement cancelled quantity
456
-            $cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty);
457
-        } elseif ($cancellation_line_item->quantity() == $qty) {
458
-            // decrement cancelled quantity in case anyone still has the object kicking around
459
-            $cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty);
460
-            // delete because quantity will end up as 0
461
-            $cancellation_line_item->delete();
462
-            // and attempt to destroy the object,
463
-            // even though PHP won't actually destroy it until it needs the memory
464
-            unset($cancellation_line_item);
465
-        } else {
466
-            // what ?!?! negative quantity ?!?!
467
-            throw new EE_Error(
468
-                sprintf(
469
-                    __(
470
-                        'Can not reinstate %1$d cancelled ticket(s) because the cancelled ticket quantity is only %2$d.',
471
-                        'event_espresso'
472
-                    ),
473
-                    $qty,
474
-                    $cancellation_line_item->quantity()
475
-                )
476
-            );
477
-        }
478
-        // increment ticket quantity
479
-        $ticket_line_item->set_quantity($ticket_line_item->quantity() + $qty);
480
-        if ($ticket_line_item->save_this_and_descendants() > 0) {
481
-            // increment parent line item quantity
482
-            $event_line_item = $ticket_line_item->parent();
483
-            if ($event_line_item instanceof EE_Line_Item && $event_line_item->OBJ_type() === 'Event') {
484
-                $event_line_item->set_quantity($event_line_item->quantity() + $qty);
485
-            }
486
-            EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item);
487
-            return true;
488
-        }
489
-        return false;
490
-    }
491
-
492
-
493
-    /**
494
-     * calls EEH_Line_Item::find_transaction_grand_total_for_line_item()
495
-     * then EE_Line_Item::recalculate_total_including_taxes() on the result
496
-     *
497
-     * @param EE_Line_Item $line_item
498
-     * @return \EE_Line_Item
499
-     */
500
-    public static function get_grand_total_and_recalculate_everything(EE_Line_Item $line_item)
501
-    {
502
-        $grand_total_line_item = EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item);
503
-        return $grand_total_line_item->recalculate_total_including_taxes();
504
-    }
505
-
506
-
507
-    /**
508
-     * Gets the line item which contains the subtotal of all the items
509
-     *
510
-     * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
511
-     * @return \EE_Line_Item
512
-     * @throws \EE_Error
513
-     */
514
-    public static function get_pre_tax_subtotal(EE_Line_Item $total_line_item)
515
-    {
516
-        $pre_tax_subtotal = $total_line_item->get_child_line_item('pre-tax-subtotal');
517
-        return $pre_tax_subtotal instanceof EE_Line_Item
518
-            ? $pre_tax_subtotal
519
-            : self::create_pre_tax_subtotal($total_line_item);
520
-    }
521
-
522
-
523
-    /**
524
-     * Gets the line item for the taxes subtotal
525
-     *
526
-     * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
527
-     * @return \EE_Line_Item
528
-     * @throws \EE_Error
529
-     */
530
-    public static function get_taxes_subtotal(EE_Line_Item $total_line_item)
531
-    {
532
-        $taxes = $total_line_item->get_child_line_item('taxes');
533
-        return $taxes ? $taxes : self::create_taxes_subtotal($total_line_item);
534
-    }
535
-
536
-
537
-    /**
538
-     * sets the TXN ID on an EE_Line_Item if passed a valid EE_Transaction object
539
-     *
540
-     * @param EE_Line_Item $line_item
541
-     * @param EE_Transaction $transaction
542
-     * @return void
543
-     * @throws \EE_Error
544
-     */
545
-    public static function set_TXN_ID(EE_Line_Item $line_item, $transaction = null)
546
-    {
547
-        if ($transaction) {
548
-            /** @type EEM_Transaction $EEM_Transaction */
549
-            $EEM_Transaction = EE_Registry::instance()->load_model('Transaction');
550
-            $TXN_ID = $EEM_Transaction->ensure_is_ID($transaction);
551
-            $line_item->set_TXN_ID($TXN_ID);
552
-        }
553
-    }
554
-
555
-
556
-    /**
557
-     * Creates a new default total line item for the transaction,
558
-     * and its tickets subtotal and taxes subtotal line items (and adds the
559
-     * existing taxes as children of the taxes subtotal line item)
560
-     *
561
-     * @param EE_Transaction $transaction
562
-     * @return \EE_Line_Item of type total
563
-     * @throws \EE_Error
564
-     */
565
-    public static function create_total_line_item($transaction = null)
566
-    {
567
-        $total_line_item = EE_Line_Item::new_instance(array(
568
-            'LIN_code' => 'total',
569
-            'LIN_name' => __('Grand Total', 'event_espresso'),
570
-            'LIN_type' => EEM_Line_Item::type_total,
571
-            'OBJ_type' => 'Transaction'
572
-        ));
573
-        $total_line_item = apply_filters(
574
-            'FHEE__EEH_Line_Item__create_total_line_item__total_line_item',
575
-            $total_line_item
576
-        );
577
-        self::set_TXN_ID($total_line_item, $transaction);
578
-        self::create_pre_tax_subtotal($total_line_item, $transaction);
579
-        self::create_taxes_subtotal($total_line_item, $transaction);
580
-        return $total_line_item;
581
-    }
582
-
583
-
584
-    /**
585
-     * Creates a default items subtotal line item
586
-     *
587
-     * @param EE_Line_Item $total_line_item
588
-     * @param EE_Transaction $transaction
589
-     * @return EE_Line_Item
590
-     * @throws \EE_Error
591
-     */
592
-    protected static function create_pre_tax_subtotal(EE_Line_Item $total_line_item, $transaction = null)
593
-    {
594
-        $pre_tax_line_item = EE_Line_Item::new_instance(array(
595
-            'LIN_code' => 'pre-tax-subtotal',
596
-            'LIN_name' => __('Pre-Tax Subtotal', 'event_espresso'),
597
-            'LIN_type' => EEM_Line_Item::type_sub_total
598
-        ));
599
-        $pre_tax_line_item = apply_filters(
600
-            'FHEE__EEH_Line_Item__create_pre_tax_subtotal__pre_tax_line_item',
601
-            $pre_tax_line_item
602
-        );
603
-        self::set_TXN_ID($pre_tax_line_item, $transaction);
604
-        $total_line_item->add_child_line_item($pre_tax_line_item);
605
-        self::create_event_subtotal($pre_tax_line_item, $transaction);
606
-        return $pre_tax_line_item;
607
-    }
608
-
609
-
610
-    /**
611
-     * Creates a line item for the taxes subtotal and finds all the tax prices
612
-     * and applies taxes to it
613
-     *
614
-     * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
615
-     * @param EE_Transaction $transaction
616
-     * @return EE_Line_Item
617
-     * @throws \EE_Error
618
-     */
619
-    protected static function create_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null)
620
-    {
621
-        $tax_line_item = EE_Line_Item::new_instance(array(
622
-            'LIN_code' => 'taxes',
623
-            'LIN_name' => __('Taxes', 'event_espresso'),
624
-            'LIN_type' => EEM_Line_Item::type_tax_sub_total,
625
-            'LIN_order' => 1000,// this should always come last
626
-        ));
627
-        $tax_line_item = apply_filters(
628
-            'FHEE__EEH_Line_Item__create_taxes_subtotal__tax_line_item',
629
-            $tax_line_item
630
-        );
631
-        self::set_TXN_ID($tax_line_item, $transaction);
632
-        $total_line_item->add_child_line_item($tax_line_item);
633
-        // and lastly, add the actual taxes
634
-        self::apply_taxes($total_line_item);
635
-        return $tax_line_item;
636
-    }
637
-
638
-
639
-    /**
640
-     * Creates a default items subtotal line item
641
-     *
642
-     * @param EE_Line_Item $pre_tax_line_item
643
-     * @param EE_Transaction $transaction
644
-     * @param EE_Event $event
645
-     * @return EE_Line_Item
646
-     * @throws \EE_Error
647
-     */
648
-    public static function create_event_subtotal(EE_Line_Item $pre_tax_line_item, $transaction = null, $event = null)
649
-    {
650
-        $event_line_item = EE_Line_Item::new_instance(array(
651
-            'LIN_code' => self::get_event_code($event),
652
-            'LIN_name' => self::get_event_name($event),
653
-            'LIN_desc' => self::get_event_desc($event),
654
-            'LIN_type' => EEM_Line_Item::type_sub_total,
655
-            'OBJ_type' => 'Event',
656
-            'OBJ_ID' => $event instanceof EE_Event ? $event->ID() : 0
657
-        ));
658
-        $event_line_item = apply_filters(
659
-            'FHEE__EEH_Line_Item__create_event_subtotal__event_line_item',
660
-            $event_line_item
661
-        );
662
-        self::set_TXN_ID($event_line_item, $transaction);
663
-        $pre_tax_line_item->add_child_line_item($event_line_item);
664
-        return $event_line_item;
665
-    }
666
-
667
-
668
-    /**
669
-     * Gets what the event ticket's code SHOULD be
670
-     *
671
-     * @param EE_Event $event
672
-     * @return string
673
-     * @throws \EE_Error
674
-     */
675
-    public static function get_event_code($event)
676
-    {
677
-        return 'event-' . ($event instanceof EE_Event ? $event->ID() : '0');
678
-    }
679
-
680
-    /**
681
-     * Gets the event name
682
-     * @param EE_Event $event
683
-     * @return string
684
-     */
685
-    public static function get_event_name($event)
686
-    {
687
-        return $event instanceof EE_Event ? $event->name() : __('Event', 'event_espresso');
688
-    }
689
-
690
-    /**
691
-     * Gets the event excerpt
692
-     * @param EE_Event $event
693
-     * @return string
694
-     */
695
-    public static function get_event_desc($event)
696
-    {
697
-        return $event instanceof EE_Event ? $event->short_description() : '';
698
-    }
699
-
700
-    /**
701
-     * Given the grand total line item and a ticket, finds the event sub-total
702
-     * line item the ticket's purchase should be added onto
703
-     *
704
-     * @access public
705
-     * @param EE_Line_Item $grand_total the grand total line item
706
-     * @param EE_Ticket $ticket
707
-     * @throws \EE_Error
708
-     * @return EE_Line_Item
709
-     */
710
-    public static function get_event_line_item_for_ticket(EE_Line_Item $grand_total, EE_Ticket $ticket)
711
-    {
712
-        $first_datetime = $ticket->first_datetime();
713
-        if (!$first_datetime instanceof EE_Datetime) {
714
-            throw new EE_Error(
715
-                sprintf(__('The supplied ticket (ID %d) has no datetimes', 'event_espresso'), $ticket->ID())
716
-            );
717
-        }
718
-        $event = $first_datetime->event();
719
-        if (!$event instanceof EE_Event) {
720
-            throw new EE_Error(
721
-                sprintf(
722
-                    __('The supplied ticket (ID %d) has no event data associated with it.', 'event_espresso'),
723
-                    $ticket->ID()
724
-                )
725
-            );
726
-        }
727
-        $events_sub_total = EEH_Line_Item::get_event_line_item($grand_total, $event);
728
-        if (!$events_sub_total instanceof EE_Line_Item) {
729
-            throw new EE_Error(
730
-                sprintf(
731
-                    __('There is no events sub-total for ticket %s on total line item %d', 'event_espresso'),
732
-                    $ticket->ID(),
733
-                    $grand_total->ID()
734
-                )
735
-            );
736
-        }
737
-        return $events_sub_total;
738
-    }
739
-
740
-
741
-    /**
742
-     * Gets the event line item
743
-     *
744
-     * @param EE_Line_Item $grand_total
745
-     * @param EE_Event $event
746
-     * @return EE_Line_Item for the event subtotal which is a child of $grand_total
747
-     * @throws \EE_Error
748
-     */
749
-    public static function get_event_line_item(EE_Line_Item $grand_total, $event)
750
-    {
751
-        /** @type EE_Event $event */
752
-        $event = EEM_Event::instance()->ensure_is_obj($event, true);
753
-        $event_line_item = null;
754
-        $found = false;
755
-        foreach (EEH_Line_Item::get_event_subtotals($grand_total) as $event_line_item) {
756
-            // default event subtotal, we should only ever find this the first time this method is called
757
-            if (!$event_line_item->OBJ_ID()) {
758
-                // let's use this! but first... set the event details
759
-                EEH_Line_Item::set_event_subtotal_details($event_line_item, $event);
760
-                $found = true;
761
-                break;
762
-            } elseif ($event_line_item->OBJ_ID() === $event->ID()) {
763
-                // found existing line item for this event in the cart, so break out of loop and use this one
764
-                $found = true;
765
-                break;
766
-            }
767
-        }
768
-        if (!$found) {
769
-            // there is no event sub-total yet, so add it
770
-            $pre_tax_subtotal = EEH_Line_Item::get_pre_tax_subtotal($grand_total);
771
-            // create a new "event" subtotal below that
772
-            $event_line_item = EEH_Line_Item::create_event_subtotal($pre_tax_subtotal, null, $event);
773
-            // and set the event details
774
-            EEH_Line_Item::set_event_subtotal_details($event_line_item, $event);
775
-        }
776
-        return $event_line_item;
777
-    }
778
-
779
-
780
-    /**
781
-     * Creates a default items subtotal line item
782
-     *
783
-     * @param EE_Line_Item $event_line_item
784
-     * @param EE_Event $event
785
-     * @param EE_Transaction $transaction
786
-     * @return EE_Line_Item
787
-     * @throws \EE_Error
788
-     */
789
-    public static function set_event_subtotal_details(
790
-        EE_Line_Item $event_line_item,
791
-        EE_Event $event,
792
-        $transaction = null
793
-    ) {
794
-        if ($event instanceof EE_Event) {
795
-            $event_line_item->set_code(self::get_event_code($event));
796
-            $event_line_item->set_name(self::get_event_name($event));
797
-            $event_line_item->set_desc(self::get_event_desc($event));
798
-            $event_line_item->set_OBJ_ID($event->ID());
799
-        }
800
-        self::set_TXN_ID($event_line_item, $transaction);
801
-    }
802
-
803
-
804
-    /**
805
-     * Finds what taxes should apply, adds them as tax line items under the taxes sub-total,
806
-     * and recalculates the taxes sub-total and the grand total. Resets the taxes, so
807
-     * any old taxes are removed
808
-     *
809
-     * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
810
-     * @throws \EE_Error
811
-     */
812
-    public static function apply_taxes(EE_Line_Item $total_line_item)
813
-    {
814
-        /** @type EEM_Price $EEM_Price */
815
-        $EEM_Price = EE_Registry::instance()->load_model('Price');
816
-        // get array of taxes via Price Model
817
-        $ordered_taxes = $EEM_Price->get_all_prices_that_are_taxes();
818
-        ksort($ordered_taxes);
819
-        $taxes_line_item = self::get_taxes_subtotal($total_line_item);
820
-        // just to be safe, remove its old tax line items
821
-        $taxes_line_item->delete_children_line_items();
822
-        // loop thru taxes
823
-        foreach ($ordered_taxes as $order => $taxes) {
824
-            foreach ($taxes as $tax) {
825
-                if ($tax instanceof EE_Price) {
826
-                    $tax_line_item = EE_Line_Item::new_instance(
827
-                        array(
828
-                            'LIN_name' => $tax->name(),
829
-                            'LIN_desc' => $tax->desc(),
830
-                            'LIN_percent' => $tax->amount(),
831
-                            'LIN_is_taxable' => false,
832
-                            'LIN_order' => $order,
833
-                            'LIN_total' => 0,
834
-                            'LIN_type' => EEM_Line_Item::type_tax,
835
-                            'OBJ_type' => 'Price',
836
-                            'OBJ_ID' => $tax->ID()
837
-                        )
838
-                    );
839
-                    $tax_line_item = apply_filters(
840
-                        'FHEE__EEH_Line_Item__apply_taxes__tax_line_item',
841
-                        $tax_line_item
842
-                    );
843
-                    $taxes_line_item->add_child_line_item($tax_line_item);
844
-                }
845
-            }
846
-        }
847
-        $total_line_item->recalculate_total_including_taxes();
848
-    }
849
-
850
-
851
-    /**
852
-     * Ensures that taxes have been applied to the order, if not applies them.
853
-     * Returns the total amount of tax
854
-     *
855
-     * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
856
-     * @return float
857
-     * @throws \EE_Error
858
-     */
859
-    public static function ensure_taxes_applied($total_line_item)
860
-    {
861
-        $taxes_subtotal = self::get_taxes_subtotal($total_line_item);
862
-        if (!$taxes_subtotal->children()) {
863
-            self::apply_taxes($total_line_item);
864
-        }
865
-        return $taxes_subtotal->total();
866
-    }
867
-
868
-
869
-    /**
870
-     * Deletes ALL children of the passed line item
871
-     *
872
-     * @param EE_Line_Item $parent_line_item
873
-     * @return bool
874
-     * @throws \EE_Error
875
-     */
876
-    public static function delete_all_child_items(EE_Line_Item $parent_line_item)
877
-    {
878
-        $deleted = 0;
879
-        foreach ($parent_line_item->children() as $child_line_item) {
880
-            if ($child_line_item instanceof EE_Line_Item) {
881
-                $deleted += EEH_Line_Item::delete_all_child_items($child_line_item);
882
-                if ($child_line_item->ID()) {
883
-                    $child_line_item->delete();
884
-                    unset($child_line_item);
885
-                } else {
886
-                    $parent_line_item->delete_child_line_item($child_line_item->code());
887
-                }
888
-                $deleted++;
889
-            }
890
-        }
891
-        return $deleted;
892
-    }
893
-
894
-
895
-    /**
896
-     * Deletes the line items as indicated by the line item code(s) provided,
897
-     * regardless of where they're found in the line item tree. Automatically
898
-     * re-calculates the line item totals and updates the related transaction. But
899
-     * DOES NOT automatically upgrade the transaction's registrations' final prices (which
900
-     * should probably change because of this).
901
-     * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
902
-     * after using this, to keep the registration final prices in-sync with the transaction's total.
903
-     * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
904
-     * @param array|bool|string $line_item_codes
905
-     * @return int number of items successfully removed
906
-     */
907
-    public static function delete_items(EE_Line_Item $total_line_item, $line_item_codes = false)
908
-    {
909
-
910
-        if ($total_line_item->type() !== EEM_Line_Item::type_total) {
911
-            EE_Error::doing_it_wrong(
912
-                'EEH_Line_Item::delete_items',
913
-                __(
914
-                    'This static method should only be called with a TOTAL line item, otherwise we won\'t recalculate the totals correctly',
915
-                    'event_espresso'
916
-                ),
917
-                '4.6.18'
918
-            );
919
-        }
920
-        do_action('AHEE_log', __FILE__, __FUNCTION__, '');
921
-
922
-        // check if only a single line_item_id was passed
923
-        if (!empty($line_item_codes) && !is_array($line_item_codes)) {
924
-            // place single line_item_id in an array to appear as multiple line_item_ids
925
-            $line_item_codes = array($line_item_codes);
926
-        }
927
-        $removals = 0;
928
-        // cycle thru line_item_ids
929
-        foreach ($line_item_codes as $line_item_id) {
930
-            $removals += $total_line_item->delete_child_line_item($line_item_id);
931
-        }
932
-
933
-        if ($removals > 0) {
934
-            $total_line_item->recalculate_taxes_and_tax_total();
935
-            return $removals;
936
-        } else {
937
-            return false;
938
-        }
939
-    }
940
-
941
-
942
-    /**
943
-     * Overwrites the previous tax by clearing out the old taxes, and creates a new
944
-     * tax and updates the total line item accordingly
945
-     *
946
-     * @param EE_Line_Item $total_line_item
947
-     * @param float $amount
948
-     * @param string $name
949
-     * @param string $description
950
-     * @param string $code
951
-     * @param boolean $add_to_existing_line_item
952
-     *                          if true, and a duplicate line item with the same code is found,
953
-     *                          $amount will be added onto it; otherwise will simply set the taxes to match $amount
954
-     * @return EE_Line_Item the new tax line item created
955
-     * @throws \EE_Error
956
-     */
957
-    public static function set_total_tax_to(
958
-        EE_Line_Item $total_line_item,
959
-        $amount,
960
-        $name = null,
961
-        $description = null,
962
-        $code = null,
963
-        $add_to_existing_line_item = false
964
-    ) {
965
-        $tax_subtotal = self::get_taxes_subtotal($total_line_item);
966
-        $taxable_total = $total_line_item->taxable_total();
967
-
968
-        if ($add_to_existing_line_item) {
969
-            $new_tax = $tax_subtotal->get_child_line_item($code);
970
-            EEM_Line_Item::instance()->delete(
971
-                array(array('LIN_code' => array('!=', $code), 'LIN_parent' => $tax_subtotal->ID()))
972
-            );
973
-        } else {
974
-            $new_tax = null;
975
-            $tax_subtotal->delete_children_line_items();
976
-        }
977
-        if ($new_tax) {
978
-            $new_tax->set_total($new_tax->total() + $amount);
979
-            $new_tax->set_percent($taxable_total ? $new_tax->total() / $taxable_total * 100 : 0);
980
-        } else {
981
-            // no existing tax item. Create it
982
-            $new_tax = EE_Line_Item::new_instance(array(
983
-                'TXN_ID' => $total_line_item->TXN_ID(),
984
-                'LIN_name' => $name ? $name : __('Tax', 'event_espresso'),
985
-                'LIN_desc' => $description ? $description : '',
986
-                'LIN_percent' => $taxable_total ? ($amount / $taxable_total * 100) : 0,
987
-                'LIN_total' => $amount,
988
-                'LIN_parent' => $tax_subtotal->ID(),
989
-                'LIN_type' => EEM_Line_Item::type_tax,
990
-                'LIN_code' => $code
991
-            ));
992
-        }
993
-
994
-        $new_tax = apply_filters(
995
-            'FHEE__EEH_Line_Item__set_total_tax_to__new_tax_subtotal',
996
-            $new_tax,
997
-            $total_line_item
998
-        );
999
-        $new_tax->save();
1000
-        $tax_subtotal->set_total($new_tax->total());
1001
-        $tax_subtotal->save();
1002
-        $total_line_item->recalculate_total_including_taxes();
1003
-        return $new_tax;
1004
-    }
1005
-
1006
-
1007
-    /**
1008
-     * Makes all the line items which are children of $line_item taxable (or not).
1009
-     * Does NOT save the line items
1010
-     * @param EE_Line_Item $line_item
1011
-     * @param string $code_substring_for_whitelist if this string is part of the line item's code
1012
-     *  it will be whitelisted (ie, except from becoming taxable)
1013
-     * @param boolean $taxable
1014
-     */
1015
-    public static function set_line_items_taxable(
1016
-        EE_Line_Item $line_item,
1017
-        $taxable = true,
1018
-        $code_substring_for_whitelist = null
1019
-    ) {
1020
-        $whitelisted = false;
1021
-        if ($code_substring_for_whitelist !== null) {
1022
-            $whitelisted = strpos($line_item->code(), $code_substring_for_whitelist) !== false ? true : false;
1023
-        }
1024
-        if (!$whitelisted && $line_item->is_line_item()) {
1025
-            $line_item->set_is_taxable($taxable);
1026
-        }
1027
-        foreach ($line_item->children() as $child_line_item) {
1028
-            EEH_Line_Item::set_line_items_taxable($child_line_item, $taxable, $code_substring_for_whitelist);
1029
-        }
1030
-    }
1031
-
1032
-
1033
-    /**
1034
-     * Gets all descendants that are event subtotals
1035
-     *
1036
-     * @uses  EEH_Line_Item::get_subtotals_of_object_type()
1037
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1038
-     * @return EE_Line_Item[]
1039
-     */
1040
-    public static function get_event_subtotals(EE_Line_Item $parent_line_item)
1041
-    {
1042
-        return self::get_subtotals_of_object_type($parent_line_item, 'Event');
1043
-    }
1044
-
1045
-
1046
-    /**
1047
-     * Gets all descendants subtotals that match the supplied object type
1048
-     *
1049
-     * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1050
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1051
-     * @param string $obj_type
1052
-     * @return EE_Line_Item[]
1053
-     */
1054
-    public static function get_subtotals_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '')
1055
-    {
1056
-        return self::_get_descendants_by_type_and_object_type(
1057
-            $parent_line_item,
1058
-            EEM_Line_Item::type_sub_total,
1059
-            $obj_type
1060
-        );
1061
-    }
1062
-
1063
-
1064
-    /**
1065
-     * Gets all descendants that are tickets
1066
-     *
1067
-     * @uses  EEH_Line_Item::get_line_items_of_object_type()
1068
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1069
-     * @return EE_Line_Item[]
1070
-     */
1071
-    public static function get_ticket_line_items(EE_Line_Item $parent_line_item)
1072
-    {
1073
-        return self::get_line_items_of_object_type($parent_line_item, 'Ticket');
1074
-    }
1075
-
1076
-
1077
-    /**
1078
-     * Gets all descendants subtotals that match the supplied object type
1079
-     *
1080
-     * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1081
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1082
-     * @param string $obj_type
1083
-     * @return EE_Line_Item[]
1084
-     */
1085
-    public static function get_line_items_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '')
1086
-    {
1087
-        return self::_get_descendants_by_type_and_object_type($parent_line_item, EEM_Line_Item::type_line_item, $obj_type);
1088
-    }
1089
-
1090
-
1091
-    /**
1092
-     * Gets all the descendants (ie, children or children of children etc) that are of the type 'tax'
1093
-     * @uses  EEH_Line_Item::get_descendants_of_type()
1094
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1095
-     * @return EE_Line_Item[]
1096
-     */
1097
-    public static function get_tax_descendants(EE_Line_Item $parent_line_item)
1098
-    {
1099
-        return EEH_Line_Item::get_descendants_of_type($parent_line_item, EEM_Line_Item::type_tax);
1100
-    }
1101
-
1102
-
1103
-    /**
1104
-     * Gets all the real items purchased which are children of this item
1105
-     * @uses  EEH_Line_Item::get_descendants_of_type()
1106
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1107
-     * @return EE_Line_Item[]
1108
-     */
1109
-    public static function get_line_item_descendants(EE_Line_Item $parent_line_item)
1110
-    {
1111
-        return EEH_Line_Item::get_descendants_of_type($parent_line_item, EEM_Line_Item::type_line_item);
1112
-    }
1113
-
1114
-
1115
-    /**
1116
-     * Gets all descendants of supplied line item that match the supplied line item type
1117
-     *
1118
-     * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1119
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1120
-     * @param string $line_item_type one of the EEM_Line_Item constants
1121
-     * @return EE_Line_Item[]
1122
-     */
1123
-    public static function get_descendants_of_type(EE_Line_Item $parent_line_item, $line_item_type)
1124
-    {
1125
-        return self::_get_descendants_by_type_and_object_type($parent_line_item, $line_item_type, null);
1126
-    }
1127
-
1128
-
1129
-    /**
1130
-     * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type as well
1131
-     *
1132
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1133
-     * @param string $line_item_type one of the EEM_Line_Item constants
1134
-     * @param string | NULL $obj_type object model class name (minus prefix) or NULL to ignore object type when searching
1135
-     * @return EE_Line_Item[]
1136
-     */
1137
-    protected static function _get_descendants_by_type_and_object_type(
1138
-        EE_Line_Item $parent_line_item,
1139
-        $line_item_type,
1140
-        $obj_type = null
1141
-    ) {
1142
-        $objects = array();
1143
-        foreach ($parent_line_item->children() as $child_line_item) {
1144
-            if ($child_line_item instanceof EE_Line_Item) {
1145
-                if ($child_line_item->type() === $line_item_type
1146
-                    && (
1147
-                        $child_line_item->OBJ_type() === $obj_type || $obj_type === null
1148
-                    )
1149
-                ) {
1150
-                    $objects[] = $child_line_item;
1151
-                } else {
1152
-                    // go-through-all-its children looking for more matches
1153
-                    $objects = array_merge(
1154
-                        $objects,
1155
-                        self::_get_descendants_by_type_and_object_type(
1156
-                            $child_line_item,
1157
-                            $line_item_type,
1158
-                            $obj_type
1159
-                        )
1160
-                    );
1161
-                }
1162
-            }
1163
-        }
1164
-        return $objects;
1165
-    }
1166
-
1167
-
1168
-    /**
1169
-     * Gets all descendants subtotals that match the supplied object type
1170
-     *
1171
-     * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1172
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1173
-     * @param string $OBJ_type object type (like Event)
1174
-     * @param array $OBJ_IDs array of OBJ_IDs
1175
-     * @return EE_Line_Item[]
1176
-     */
1177
-    public static function get_line_items_by_object_type_and_IDs(
1178
-        EE_Line_Item $parent_line_item,
1179
-        $OBJ_type = '',
1180
-        $OBJ_IDs = array()
1181
-    ) {
1182
-        return self::_get_descendants_by_object_type_and_object_ID($parent_line_item, $OBJ_type, $OBJ_IDs);
1183
-    }
1184
-
1185
-
1186
-    /**
1187
-     * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type as well
1188
-     *
1189
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1190
-     * @param string $OBJ_type object type (like Event)
1191
-     * @param array $OBJ_IDs array of OBJ_IDs
1192
-     * @return EE_Line_Item[]
1193
-     */
1194
-    protected static function _get_descendants_by_object_type_and_object_ID(
1195
-        EE_Line_Item $parent_line_item,
1196
-        $OBJ_type,
1197
-        $OBJ_IDs
1198
-    ) {
1199
-        $objects = array();
1200
-        foreach ($parent_line_item->children() as $child_line_item) {
1201
-            if ($child_line_item instanceof EE_Line_Item) {
1202
-                if ($child_line_item->OBJ_type() === $OBJ_type
1203
-                    && is_array($OBJ_IDs)
1204
-                    && in_array($child_line_item->OBJ_ID(), $OBJ_IDs)
1205
-                ) {
1206
-                    $objects[] = $child_line_item;
1207
-                } else {
1208
-                    // go-through-all-its children looking for more matches
1209
-                    $objects = array_merge(
1210
-                        $objects,
1211
-                        self::_get_descendants_by_object_type_and_object_ID(
1212
-                            $child_line_item,
1213
-                            $OBJ_type,
1214
-                            $OBJ_IDs
1215
-                        )
1216
-                    );
1217
-                }
1218
-            }
1219
-        }
1220
-        return $objects;
1221
-    }
1222
-
1223
-
1224
-    /**
1225
-     * Uses a breadth-first-search in order to find the nearest descendant of
1226
-     * the specified type and returns it, else NULL
1227
-     *
1228
-     * @uses  EEH_Line_Item::_get_nearest_descendant()
1229
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1230
-     * @param string $type like one of the EEM_Line_Item::type_*
1231
-     * @return EE_Line_Item
1232
-     */
1233
-    public static function get_nearest_descendant_of_type(EE_Line_Item $parent_line_item, $type)
1234
-    {
1235
-        return self::_get_nearest_descendant($parent_line_item, 'LIN_type', $type);
1236
-    }
1237
-
1238
-
1239
-    /**
1240
-     * Uses a breadth-first-search in order to find the nearest descendant
1241
-     * having the specified LIN_code and returns it, else NULL
1242
-     *
1243
-     * @uses  EEH_Line_Item::_get_nearest_descendant()
1244
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1245
-     * @param string $code any value used for LIN_code
1246
-     * @return EE_Line_Item
1247
-     */
1248
-    public static function get_nearest_descendant_having_code(EE_Line_Item $parent_line_item, $code)
1249
-    {
1250
-        return self::_get_nearest_descendant($parent_line_item, 'LIN_code', $code);
1251
-    }
1252
-
1253
-
1254
-    /**
1255
-     * Uses a breadth-first-search in order to find the nearest descendant
1256
-     * having the specified LIN_code and returns it, else NULL
1257
-     *
1258
-     * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1259
-     * @param string $search_field name of EE_Line_Item property
1260
-     * @param string $value any value stored in $search_field
1261
-     * @return EE_Line_Item
1262
-     */
1263
-    protected static function _get_nearest_descendant(EE_Line_Item $parent_line_item, $search_field, $value)
1264
-    {
1265
-        foreach ($parent_line_item->children() as $child) {
1266
-            if ($child->get($search_field) == $value) {
1267
-                return $child;
1268
-            }
1269
-        }
1270
-        foreach ($parent_line_item->children() as $child) {
1271
-            $descendant_found = self::_get_nearest_descendant($child, $search_field, $value);
1272
-            if ($descendant_found) {
1273
-                return $descendant_found;
1274
-            }
1275
-        }
1276
-        return null;
1277
-    }
1278
-
1279
-
1280
-    /**
1281
-     * if passed line item has a TXN ID, uses that to jump directly to the grand total line item for the transaction,
1282
-     * else recursively walks up the line item tree until a parent of type total is found,
1283
-     *
1284
-     * @param EE_Line_Item $line_item
1285
-     * @return \EE_Line_Item
1286
-     * @throws \EE_Error
1287
-     */
1288
-    public static function find_transaction_grand_total_for_line_item(EE_Line_Item $line_item)
1289
-    {
1290
-        if ($line_item->TXN_ID()) {
1291
-            $total_line_item = $line_item->transaction()->total_line_item(false);
1292
-            if ($total_line_item instanceof EE_Line_Item) {
1293
-                return $total_line_item;
1294
-            }
1295
-        } else {
1296
-            $line_item_parent = $line_item->parent();
1297
-            if ($line_item_parent instanceof EE_Line_Item) {
1298
-                if ($line_item_parent->is_total()) {
1299
-                    return $line_item_parent;
1300
-                }
1301
-                return EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item_parent);
1302
-            }
1303
-        }
1304
-        throw new EE_Error(
1305
-            sprintf(
1306
-                __('A valid grand total for line item %1$d was not found.', 'event_espresso'),
1307
-                $line_item->ID()
1308
-            )
1309
-        );
1310
-    }
1311
-
1312
-
1313
-    /**
1314
-     * Prints out a representation of the line item tree
1315
-     *
1316
-     * @param EE_Line_Item $line_item
1317
-     * @param int $indentation
1318
-     * @return void
1319
-     * @throws \EE_Error
1320
-     */
1321
-    public static function visualize(EE_Line_Item $line_item, $indentation = 0)
1322
-    {
1323
-        echo defined('EE_TESTS_DIR') ? "\n" : '<br />';
1324
-        if (!$indentation) {
1325
-            echo defined('EE_TESTS_DIR') ? "\n" : '<br />';
1326
-        }
1327
-        for ($i = 0; $i < $indentation; $i++) {
1328
-            echo ". ";
1329
-        }
1330
-        $breakdown = '';
1331
-        if ($line_item->is_line_item()) {
1332
-            if ($line_item->is_percent()) {
1333
-                $breakdown = "{$line_item->percent()}%";
1334
-            } else {
1335
-                $breakdown = '$' . "{$line_item->unit_price()} x {$line_item->quantity()}";
1336
-            }
1337
-        }
1338
-        echo $line_item->name() . " [ ID:{$line_item->ID()} | qty:{$line_item->quantity()} ] {$line_item->type()} : " . '$' . "{$line_item->total()}";
1339
-        if ($breakdown) {
1340
-            echo " ( {$breakdown} )";
1341
-        }
1342
-        if ($line_item->is_taxable()) {
1343
-            echo "  * taxable";
1344
-        }
1345
-        if ($line_item->children()) {
1346
-            foreach ($line_item->children() as $child) {
1347
-                self::visualize($child, $indentation + 1);
1348
-            }
1349
-        }
1350
-    }
1351
-
1352
-
1353
-    /**
1354
-     * Calculates the registration's final price, taking into account that they
1355
-     * need to not only help pay for their OWN ticket, but also any transaction-wide surcharges and taxes,
1356
-     * and receive a portion of any transaction-wide discounts.
1357
-     * eg1, if I buy a $1 ticket and brent buys a $9 ticket, and we receive a $5 discount
1358
-     * then I'll get 1/10 of that $5 discount, which is $0.50, and brent will get
1359
-     * 9/10ths of that $5 discount, which is $4.50. So my final price should be $0.50
1360
-     * and brent's final price should be $5.50.
1361
-     *
1362
-     * In order to do this, we basically need to traverse the line item tree calculating
1363
-     * the running totals (just as if we were recalculating the total), but when we identify
1364
-     * regular line items, we need to keep track of their share of the grand total.
1365
-     * Also, we need to keep track of the TAXABLE total for each ticket purchase, so
1366
-     * we can know how to apply taxes to it. (Note: "taxable total" does not equal the "pretax total"
1367
-     * when there are non-taxable items; otherwise they would be the same)
1368
-     *
1369
-     * @param EE_Line_Item $line_item
1370
-     * @param array $billable_ticket_quantities array of EE_Ticket IDs and their corresponding quantity that
1371
-     *                                                                            can be included in price calculations at this moment
1372
-     * @return array        keys are line items for tickets IDs and values are their share of the running total,
1373
-     *                                          plus the key 'total', and 'taxable' which also has keys of all the ticket IDs. Eg
1374
-     *                                          array(
1375
-     *                                          12 => 4.3
1376
-     *                                          23 => 8.0
1377
-     *                                          'total' => 16.6,
1378
-     *                                          'taxable' => array(
1379
-     *                                          12 => 10,
1380
-     *                                          23 => 4
1381
-     *                                          ).
1382
-     *                                          So to find which registrations have which final price, we need to find which line item
1383
-     *                                          is theirs, which can be done with
1384
-     *                                          `EEM_Line_Item::instance()->get_line_item_for_registration( $registration );`
1385
-     */
1386
-    public static function calculate_reg_final_prices_per_line_item(EE_Line_Item $line_item, $billable_ticket_quantities = array())
1387
-    {
1388
-        // init running grand total if not already
1389
-        if (!isset($running_totals['total'])) {
1390
-            $running_totals['total'] = 0;
1391
-        }
1392
-        if (!isset($running_totals['taxable'])) {
1393
-            $running_totals['taxable'] = array('total' => 0);
1394
-        }
1395
-        foreach ($line_item->children() as $child_line_item) {
1396
-            switch ($child_line_item->type()) {
1397
-                case EEM_Line_Item::type_sub_total:
1398
-                    $running_totals_from_subtotal = EEH_Line_Item::calculate_reg_final_prices_per_line_item($child_line_item, $billable_ticket_quantities);
1399
-                    // combine arrays but preserve numeric keys
1400
-                    $running_totals = array_replace_recursive($running_totals_from_subtotal, $running_totals);
1401
-                    $running_totals['total'] += $running_totals_from_subtotal['total'];
1402
-                    $running_totals['taxable']['total'] += $running_totals_from_subtotal['taxable']['total'];
1403
-                    break;
1404
-
1405
-                case EEM_Line_Item::type_tax_sub_total:
1406
-                    // find how much the taxes percentage is
1407
-                    if ($child_line_item->percent() !== 0) {
1408
-                        $tax_percent_decimal = $child_line_item->percent() / 100;
1409
-                    } else {
1410
-                        $tax_percent_decimal = EE_Taxes::get_total_taxes_percentage() / 100;
1411
-                    }
1412
-                    // and apply to all the taxable totals, and add to the pretax totals
1413
-                    foreach ($running_totals as $line_item_id => $this_running_total) {
1414
-                        // "total" and "taxable" array key is an exception
1415
-                        if ($line_item_id === 'taxable') {
1416
-                            continue;
1417
-                        }
1418
-                        $taxable_total = $running_totals['taxable'][ $line_item_id ];
1419
-                        $running_totals[ $line_item_id ] += ($taxable_total * $tax_percent_decimal);
1420
-                    }
1421
-                    break;
1422
-
1423
-                case EEM_Line_Item::type_line_item:
1424
-                    // ticket line items or ????
1425
-                    if ($child_line_item->OBJ_type() === 'Ticket') {
1426
-                        // kk it's a ticket
1427
-                        if (isset($running_totals[ $child_line_item->ID() ])) {
1428
-                            // huh? that shouldn't happen.
1429
-                            $running_totals['total'] += $child_line_item->total();
1430
-                        } else {
1431
-                            // its not in our running totals yet. great.
1432
-                            if ($child_line_item->is_taxable()) {
1433
-                                $taxable_amount = $child_line_item->unit_price();
1434
-                            } else {
1435
-                                $taxable_amount = 0;
1436
-                            }
1437
-                            // are we only calculating totals for some tickets?
1438
-                            if (isset($billable_ticket_quantities[ $child_line_item->OBJ_ID() ])) {
1439
-                                $quantity = $billable_ticket_quantities[ $child_line_item->OBJ_ID() ];
1440
-                                $running_totals[ $child_line_item->ID() ] = $quantity
1441
-                                    ? $child_line_item->unit_price()
1442
-                                    : 0;
1443
-                                $running_totals['taxable'][ $child_line_item->ID() ] = $quantity
1444
-                                    ? $taxable_amount
1445
-                                    : 0;
1446
-                            } else {
1447
-                                $quantity = $child_line_item->quantity();
1448
-                                $running_totals[ $child_line_item->ID() ] = $child_line_item->unit_price();
1449
-                                $running_totals['taxable'][ $child_line_item->ID() ] = $taxable_amount;
1450
-                            }
1451
-                            $running_totals['taxable']['total'] += $taxable_amount * $quantity;
1452
-                            $running_totals['total'] += $child_line_item->unit_price() * $quantity;
1453
-                        }
1454
-                    } else {
1455
-                        // it's some other type of item added to the cart
1456
-                        // it should affect the running totals
1457
-                        // basically we want to convert it into a PERCENT modifier. Because
1458
-                        // more clearly affect all registration's final price equally
1459
-                        $line_items_percent_of_running_total = $running_totals['total'] > 0
1460
-                            ? ($child_line_item->total() / $running_totals['total']) + 1
1461
-                            : 1;
1462
-                        foreach ($running_totals as $line_item_id => $this_running_total) {
1463
-                            // the "taxable" array key is an exception
1464
-                            if ($line_item_id === 'taxable') {
1465
-                                continue;
1466
-                            }
1467
-                            // update the running totals
1468
-                            // yes this actually even works for the running grand total!
1469
-                            $running_totals[ $line_item_id ] =
1470
-                                $line_items_percent_of_running_total * $this_running_total;
1471
-
1472
-                            if ($child_line_item->is_taxable()) {
1473
-                                $running_totals['taxable'][ $line_item_id ] =
1474
-                                    $line_items_percent_of_running_total * $running_totals['taxable'][ $line_item_id ];
1475
-                            }
1476
-                        }
1477
-                    }
1478
-                    break;
1479
-            }
1480
-        }
1481
-        return $running_totals;
1482
-    }
1483
-
1484
-
1485
-    /**
1486
-     * @param \EE_Line_Item $total_line_item
1487
-     * @param \EE_Line_Item $ticket_line_item
1488
-     * @return float | null
1489
-     * @throws \OutOfRangeException
1490
-     */
1491
-    public static function calculate_final_price_for_ticket_line_item(\EE_Line_Item $total_line_item, \EE_Line_Item $ticket_line_item)
1492
-    {
1493
-        static $final_prices_per_ticket_line_item = array();
1494
-        if (empty($final_prices_per_ticket_line_item)) {
1495
-            $final_prices_per_ticket_line_item = \EEH_Line_Item::calculate_reg_final_prices_per_line_item(
1496
-                $total_line_item
1497
-            );
1498
-        }
1499
-        // ok now find this new registration's final price
1500
-        if (isset($final_prices_per_ticket_line_item[ $ticket_line_item->ID() ])) {
1501
-            return $final_prices_per_ticket_line_item[ $ticket_line_item->ID() ];
1502
-        }
1503
-        $message = sprintf(
1504
-            __(
1505
-                'The final price for the ticket line item (ID:%1$d) could not be calculated.',
1506
-                'event_espresso'
1507
-            ),
1508
-            $ticket_line_item->ID()
1509
-        );
1510
-        if (WP_DEBUG) {
1511
-            $message .= '<br>' . print_r($final_prices_per_ticket_line_item, true);
1512
-            throw new \OutOfRangeException($message);
1513
-        } else {
1514
-            EE_Log::instance()->log(__CLASS__, __FUNCTION__, $message);
1515
-        }
1516
-        return null;
1517
-    }
1518
-
1519
-
1520
-    /**
1521
-     * Creates a duplicate of the line item tree, except only includes billable items
1522
-     * and the portion of line items attributed to billable things
1523
-     *
1524
-     * @param EE_Line_Item $line_item
1525
-     * @param EE_Registration[] $registrations
1526
-     * @return \EE_Line_Item
1527
-     * @throws \EE_Error
1528
-     */
1529
-    public static function billable_line_item_tree(EE_Line_Item $line_item, $registrations)
1530
-    {
1531
-        $copy_li = EEH_Line_Item::billable_line_item($line_item, $registrations);
1532
-        foreach ($line_item->children() as $child_li) {
1533
-            $copy_li->add_child_line_item(EEH_Line_Item::billable_line_item_tree($child_li, $registrations));
1534
-        }
1535
-        // if this is the grand total line item, make sure the totals all add up
1536
-        // (we could have duplicated this logic AS we copied the line items, but
1537
-        // it seems DRYer this way)
1538
-        if ($copy_li->type() === EEM_Line_Item::type_total) {
1539
-            $copy_li->recalculate_total_including_taxes();
1540
-        }
1541
-        return $copy_li;
1542
-    }
1543
-
1544
-
1545
-    /**
1546
-     * Creates a new, unsaved line item from $line_item that factors in the
1547
-     * number of billable registrations on $registrations.
1548
-     *
1549
-     * @param EE_Line_Item $line_item
1550
-     * @return EE_Line_Item
1551
-     * @throws \EE_Error
1552
-     * @param EE_Registration[] $registrations
1553
-     */
1554
-    public static function billable_line_item(EE_Line_Item $line_item, $registrations)
1555
-    {
1556
-        $new_li_fields = $line_item->model_field_array();
1557
-        if ($line_item->type() === EEM_Line_Item::type_line_item &&
1558
-            $line_item->OBJ_type() === 'Ticket'
1559
-        ) {
1560
-            $count = 0;
1561
-            foreach ($registrations as $registration) {
1562
-                if ($line_item->OBJ_ID() === $registration->ticket_ID() &&
1563
-                    in_array($registration->status_ID(), EEM_Registration::reg_statuses_that_allow_payment())
1564
-                ) {
1565
-                    $count++;
1566
-                }
1567
-            }
1568
-            $new_li_fields['LIN_quantity'] = $count;
1569
-        }
1570
-        // don't set the total. We'll leave that up to the code that calculates it
1571
-        unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent'], $new_li_fields['LIN_total']);
1572
-        return EE_Line_Item::new_instance($new_li_fields);
1573
-    }
1574
-
1575
-
1576
-    /**
1577
-     * Returns a modified line item tree where all the subtotals which have a total of 0
1578
-     * are removed, and line items with a quantity of 0
1579
-     *
1580
-     * @param EE_Line_Item $line_item |null
1581
-     * @return \EE_Line_Item|null
1582
-     * @throws \EE_Error
1583
-     */
1584
-    public static function non_empty_line_items(EE_Line_Item $line_item)
1585
-    {
1586
-        $copied_li = EEH_Line_Item::non_empty_line_item($line_item);
1587
-        if ($copied_li === null) {
1588
-            return null;
1589
-        }
1590
-        // if this is an event subtotal, we want to only include it if it
1591
-        // has a non-zero total and at least one ticket line item child
1592
-        $ticket_children = 0;
1593
-        foreach ($line_item->children() as $child_li) {
1594
-            $child_li_copy = EEH_Line_Item::non_empty_line_items($child_li);
1595
-            if ($child_li_copy !== null) {
1596
-                $copied_li->add_child_line_item($child_li_copy);
1597
-                if ($child_li_copy->type() === EEM_Line_Item::type_line_item &&
1598
-                    $child_li_copy->OBJ_type() === 'Ticket'
1599
-                ) {
1600
-                    $ticket_children++;
1601
-                }
1602
-            }
1603
-        }
1604
-        // if this is an event subtotal with NO ticket children
1605
-        // we basically want to ignore it
1606
-        if ($ticket_children === 0
1607
-            && $line_item->type() === EEM_Line_Item::type_sub_total
1608
-            && $line_item->OBJ_type() === 'Event'
1609
-            && $line_item->total() === 0
1610
-        ) {
1611
-            return null;
1612
-        }
1613
-        return $copied_li;
1614
-    }
1615
-
1616
-
1617
-    /**
1618
-     * Creates a new, unsaved line item, but if it's a ticket line item
1619
-     * with a total of 0, or a subtotal of 0, returns null instead
1620
-     *
1621
-     * @param EE_Line_Item $line_item
1622
-     * @return EE_Line_Item
1623
-     * @throws \EE_Error
1624
-     */
1625
-    public static function non_empty_line_item(EE_Line_Item $line_item)
1626
-    {
1627
-        if ($line_item->type() === EEM_Line_Item::type_line_item &&
1628
-            $line_item->OBJ_type() === 'Ticket' &&
1629
-            $line_item->quantity() === 0
1630
-        ) {
1631
-            return null;
1632
-        }
1633
-        $new_li_fields = $line_item->model_field_array();
1634
-        // don't set the total. We'll leave that up to the code that calculates it
1635
-        unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent']);
1636
-        return EE_Line_Item::new_instance($new_li_fields);
1637
-    }
1638
-
1639
-
1640
-
1641
-    /**************************************** @DEPRECATED METHODS *************************************** */
1642
-    /**
1643
-     * @deprecated
1644
-     * @param EE_Line_Item $total_line_item
1645
-     * @return \EE_Line_Item
1646
-     * @throws \EE_Error
1647
-     */
1648
-    public static function get_items_subtotal(EE_Line_Item $total_line_item)
1649
-    {
1650
-        EE_Error::doing_it_wrong('EEH_Line_Item::get_items_subtotal()', __('Method replaced with EEH_Line_Item::get_pre_tax_subtotal()', 'event_espresso'), '4.6.0');
1651
-        return self::get_pre_tax_subtotal($total_line_item);
1652
-    }
1653
-
1654
-
1655
-    /**
1656
-     * @deprecated
1657
-     * @param EE_Transaction $transaction
1658
-     * @return \EE_Line_Item
1659
-     * @throws \EE_Error
1660
-     */
1661
-    public static function create_default_total_line_item($transaction = null)
1662
-    {
1663
-        EE_Error::doing_it_wrong('EEH_Line_Item::create_default_total_line_item()', __('Method replaced with EEH_Line_Item::create_total_line_item()', 'event_espresso'), '4.6.0');
1664
-        return self::create_total_line_item($transaction);
1665
-    }
1666
-
1667
-
1668
-    /**
1669
-     * @deprecated
1670
-     * @param EE_Line_Item $total_line_item
1671
-     * @param EE_Transaction $transaction
1672
-     * @return \EE_Line_Item
1673
-     * @throws \EE_Error
1674
-     */
1675
-    public static function create_default_tickets_subtotal(EE_Line_Item $total_line_item, $transaction = null)
1676
-    {
1677
-        EE_Error::doing_it_wrong('EEH_Line_Item::create_default_tickets_subtotal()', __('Method replaced with EEH_Line_Item::create_pre_tax_subtotal()', 'event_espresso'), '4.6.0');
1678
-        return self::create_pre_tax_subtotal($total_line_item, $transaction);
1679
-    }
1680
-
1681
-
1682
-    /**
1683
-     * @deprecated
1684
-     * @param EE_Line_Item $total_line_item
1685
-     * @param EE_Transaction $transaction
1686
-     * @return \EE_Line_Item
1687
-     * @throws \EE_Error
1688
-     */
1689
-    public static function create_default_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null)
1690
-    {
1691
-        EE_Error::doing_it_wrong('EEH_Line_Item::create_default_taxes_subtotal()', __('Method replaced with EEH_Line_Item::create_taxes_subtotal()', 'event_espresso'), '4.6.0');
1692
-        return self::create_taxes_subtotal($total_line_item, $transaction);
1693
-    }
1694
-
1695
-
1696
-    /**
1697
-     * @deprecated
1698
-     * @param EE_Line_Item $total_line_item
1699
-     * @param EE_Transaction $transaction
1700
-     * @return \EE_Line_Item
1701
-     * @throws \EE_Error
1702
-     */
1703
-    public static function create_default_event_subtotal(EE_Line_Item $total_line_item, $transaction = null)
1704
-    {
1705
-        EE_Error::doing_it_wrong('EEH_Line_Item::create_default_event_subtotal()', __('Method replaced with EEH_Line_Item::create_event_subtotal()', 'event_espresso'), '4.6.0');
1706
-        return self::create_event_subtotal($total_line_item, $transaction);
1707
-    }
23
+	// other functions: cancel ticket purchase
24
+	// delete ticket purchase
25
+	// add promotion
26
+
27
+
28
+	/**
29
+	 * Adds a simple item (unrelated to any other model object) to the provided PARENT line item.
30
+	 * Does NOT automatically re-calculate the line item totals or update the related transaction.
31
+	 * You should call recalculate_total_including_taxes() on the grant total line item after this
32
+	 * to update the subtotals, and EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
33
+	 * to keep the registration final prices in-sync with the transaction's total.
34
+	 *
35
+	 * @param EE_Line_Item $parent_line_item
36
+	 * @param string $name
37
+	 * @param float $unit_price
38
+	 * @param string $description
39
+	 * @param int $quantity
40
+	 * @param boolean $taxable
41
+	 * @param boolean $code if set to a value, ensures there is only one line item with that code
42
+	 * @return boolean success
43
+	 * @throws \EE_Error
44
+	 */
45
+	public static function add_unrelated_item(EE_Line_Item $parent_line_item, $name, $unit_price, $description = '', $quantity = 1, $taxable = false, $code = null)
46
+	{
47
+		$items_subtotal = self::get_pre_tax_subtotal($parent_line_item);
48
+		$line_item = EE_Line_Item::new_instance(array(
49
+			'LIN_name' => $name,
50
+			'LIN_desc' => $description,
51
+			'LIN_unit_price' => $unit_price,
52
+			'LIN_quantity' => $quantity,
53
+			'LIN_percent' => null,
54
+			'LIN_is_taxable' => $taxable,
55
+			'LIN_order' => $items_subtotal instanceof EE_Line_Item ? count($items_subtotal->children()) : 0,
56
+			'LIN_total' => (float) $unit_price * (int) $quantity,
57
+			'LIN_type' => EEM_Line_Item::type_line_item,
58
+			'LIN_code' => $code,
59
+		));
60
+		$line_item = apply_filters(
61
+			'FHEE__EEH_Line_Item__add_unrelated_item__line_item',
62
+			$line_item,
63
+			$parent_line_item
64
+		);
65
+		return self::add_item($parent_line_item, $line_item);
66
+	}
67
+
68
+
69
+	/**
70
+	 * Adds a simple item ( unrelated to any other model object) to the total line item,
71
+	 * in the correct spot in the line item tree. Automatically
72
+	 * re-calculates the line item totals and updates the related transaction. But
73
+	 * DOES NOT automatically upgrade the transaction's registrations' final prices (which
74
+	 * should probably change because of this).
75
+	 * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
76
+	 * after using this, to keep the registration final prices in-sync with the transaction's total.
77
+	 *
78
+	 * @param EE_Line_Item $parent_line_item
79
+	 * @param string $name
80
+	 * @param float $percentage_amount
81
+	 * @param string $description
82
+	 * @param boolean $taxable
83
+	 * @return boolean success
84
+	 * @throws \EE_Error
85
+	 */
86
+	public static function add_percentage_based_item(EE_Line_Item $parent_line_item, $name, $percentage_amount, $description = '', $taxable = false)
87
+	{
88
+		$line_item = EE_Line_Item::new_instance(array(
89
+			'LIN_name' => $name,
90
+			'LIN_desc' => $description,
91
+			'LIN_unit_price' => 0,
92
+			'LIN_percent' => $percentage_amount,
93
+			'LIN_quantity' => 1,
94
+			'LIN_is_taxable' => $taxable,
95
+			'LIN_total' => (float) ($percentage_amount * ($parent_line_item->total() / 100)),
96
+			'LIN_type' => EEM_Line_Item::type_line_item,
97
+			'LIN_parent' => $parent_line_item->ID()
98
+		));
99
+		$line_item = apply_filters(
100
+			'FHEE__EEH_Line_Item__add_percentage_based_item__line_item',
101
+			$line_item
102
+		);
103
+		return $parent_line_item->add_child_line_item($line_item, false);
104
+	}
105
+
106
+
107
+	/**
108
+	 * Returns the new line item created by adding a purchase of the ticket
109
+	 * ensures that ticket line item is saved, and that cart total has been recalculated.
110
+	 * If this ticket has already been purchased, just increments its count.
111
+	 * Automatically re-calculates the line item totals and updates the related transaction. But
112
+	 * DOES NOT automatically upgrade the transaction's registrations' final prices (which
113
+	 * should probably change because of this).
114
+	 * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
115
+	 * after using this, to keep the registration final prices in-sync with the transaction's total.
116
+	 *
117
+	 * @param EE_Line_Item $total_line_item grand total line item of type EEM_Line_Item::type_total
118
+	 * @param EE_Ticket $ticket
119
+	 * @param int $qty
120
+	 * @return \EE_Line_Item
121
+	 * @throws \EE_Error
122
+	 */
123
+	public static function add_ticket_purchase(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1)
124
+	{
125
+		if (!$total_line_item instanceof EE_Line_Item || !$total_line_item->is_total()) {
126
+			throw new EE_Error(sprintf(__('A valid line item total is required in order to add tickets. A line item of type "%s" was passed.', 'event_espresso'), $ticket->ID(), $total_line_item->ID()));
127
+		}
128
+		// either increment the qty for an existing ticket
129
+		$line_item = self::increment_ticket_qty_if_already_in_cart($total_line_item, $ticket, $qty);
130
+		// or add a new one
131
+		if (!$line_item instanceof EE_Line_Item) {
132
+			$line_item = self::create_ticket_line_item($total_line_item, $ticket, $qty);
133
+		}
134
+		$total_line_item->recalculate_total_including_taxes();
135
+		return $line_item;
136
+	}
137
+
138
+
139
+	/**
140
+	 * Returns the new line item created by adding a purchase of the ticket
141
+	 * @param \EE_Line_Item $total_line_item
142
+	 * @param EE_Ticket $ticket
143
+	 * @param int $qty
144
+	 * @return \EE_Line_Item
145
+	 * @throws \EE_Error
146
+	 */
147
+	public static function increment_ticket_qty_if_already_in_cart(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1)
148
+	{
149
+		$line_item = null;
150
+		if ($total_line_item instanceof EE_Line_Item && $total_line_item->is_total()) {
151
+			$ticket_line_items = EEH_Line_Item::get_ticket_line_items($total_line_item);
152
+			foreach ((array) $ticket_line_items as $ticket_line_item) {
153
+				if ($ticket_line_item instanceof EE_Line_Item
154
+					&& (int) $ticket_line_item->OBJ_ID() === (int) $ticket->ID()
155
+				) {
156
+					$line_item = $ticket_line_item;
157
+					break;
158
+				}
159
+			}
160
+		}
161
+		if ($line_item instanceof EE_Line_Item) {
162
+			EEH_Line_Item::increment_quantity($line_item, $qty);
163
+			return $line_item;
164
+		}
165
+		return null;
166
+	}
167
+
168
+
169
+	/**
170
+	 * Increments the line item and all its children's quantity by $qty (but percent line items are unaffected).
171
+	 * Does NOT save or recalculate other line items totals
172
+	 *
173
+	 * @param EE_Line_Item $line_item
174
+	 * @param int $qty
175
+	 * @return void
176
+	 * @throws \EE_Error
177
+	 */
178
+	public static function increment_quantity(EE_Line_Item $line_item, $qty = 1)
179
+	{
180
+		if (!$line_item->is_percent()) {
181
+			$qty += $line_item->quantity();
182
+			$line_item->set_quantity($qty);
183
+			$line_item->set_total($line_item->unit_price() * $qty);
184
+			$line_item->save();
185
+		}
186
+		foreach ($line_item->children() as $child) {
187
+			if ($child->is_sub_line_item()) {
188
+				EEH_Line_Item::update_quantity($child, $qty);
189
+			}
190
+		}
191
+	}
192
+
193
+
194
+	/**
195
+	 * Decrements the line item and all its children's quantity by $qty (but percent line items are unaffected).
196
+	 * Does NOT save or recalculate other line items totals
197
+	 *
198
+	 * @param EE_Line_Item $line_item
199
+	 * @param int $qty
200
+	 * @return void
201
+	 * @throws \EE_Error
202
+	 */
203
+	public static function decrement_quantity(EE_Line_Item $line_item, $qty = 1)
204
+	{
205
+		if (!$line_item->is_percent()) {
206
+			$qty = $line_item->quantity() - $qty;
207
+			$qty = max($qty, 0);
208
+			$line_item->set_quantity($qty);
209
+			$line_item->set_total($line_item->unit_price() * $qty);
210
+			$line_item->save();
211
+		}
212
+		foreach ($line_item->children() as $child) {
213
+			if ($child->is_sub_line_item()) {
214
+				EEH_Line_Item::update_quantity($child, $qty);
215
+			}
216
+		}
217
+	}
218
+
219
+
220
+	/**
221
+	 * Updates the line item and its children's quantities to the specified number.
222
+	 * Does NOT save them or recalculate totals.
223
+	 *
224
+	 * @param EE_Line_Item $line_item
225
+	 * @param int $new_quantity
226
+	 * @throws \EE_Error
227
+	 */
228
+	public static function update_quantity(EE_Line_Item $line_item, $new_quantity)
229
+	{
230
+		if (!$line_item->is_percent()) {
231
+			$line_item->set_quantity($new_quantity);
232
+			$line_item->set_total($line_item->unit_price() * $new_quantity);
233
+			$line_item->save();
234
+		}
235
+		foreach ($line_item->children() as $child) {
236
+			if ($child->is_sub_line_item()) {
237
+				EEH_Line_Item::update_quantity($child, $new_quantity);
238
+			}
239
+		}
240
+	}
241
+
242
+
243
+	/**
244
+	 * Returns the new line item created by adding a purchase of the ticket
245
+	 * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
246
+	 * @param EE_Ticket $ticket
247
+	 * @param int $qty
248
+	 * @return \EE_Line_Item
249
+	 * @throws \EE_Error
250
+	 */
251
+	public static function create_ticket_line_item(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1)
252
+	{
253
+		$datetimes = $ticket->datetimes();
254
+		$first_datetime = reset($datetimes);
255
+		if ($first_datetime instanceof EE_Datetime && $first_datetime->event() instanceof EE_Event) {
256
+			$first_datetime_name = $first_datetime->event()->name();
257
+		} else {
258
+			$first_datetime_name = __('Event', 'event_espresso');
259
+		}
260
+		$event = sprintf(_x('(For %1$s)', '(For Event Name)', 'event_espresso'), $first_datetime_name);
261
+		// get event subtotal line
262
+		$events_sub_total = self::get_event_line_item_for_ticket($total_line_item, $ticket);
263
+		// add $ticket to cart
264
+		$line_item = EE_Line_Item::new_instance(array(
265
+			'LIN_name' => $ticket->name(),
266
+			'LIN_desc' => $ticket->description() !== '' ? $ticket->description() . ' ' . $event : $event,
267
+			'LIN_unit_price' => $ticket->price(),
268
+			'LIN_quantity' => $qty,
269
+			'LIN_is_taxable' => $ticket->taxable(),
270
+			'LIN_order' => count($events_sub_total->children()),
271
+			'LIN_total' => $ticket->price() * $qty,
272
+			'LIN_type' => EEM_Line_Item::type_line_item,
273
+			'OBJ_ID' => $ticket->ID(),
274
+			'OBJ_type' => 'Ticket'
275
+		));
276
+		$line_item = apply_filters(
277
+			'FHEE__EEH_Line_Item__create_ticket_line_item__line_item',
278
+			$line_item
279
+		);
280
+		$events_sub_total->add_child_line_item($line_item);
281
+		// now add the sub-line items
282
+		$running_total_for_ticket = 0;
283
+		foreach ($ticket->prices(array('order_by' => array('PRC_order' => 'ASC'))) as $price) {
284
+			$sign = $price->is_discount() ? -1 : 1;
285
+			$price_total = $price->is_percent()
286
+				? $running_total_for_ticket * $price->amount() / 100
287
+				: $price->amount() * $qty;
288
+			$sub_line_item = EE_Line_Item::new_instance(array(
289
+				'LIN_name' => $price->name(),
290
+				'LIN_desc' => $price->desc(),
291
+				'LIN_quantity' => $price->is_percent() ? null : $qty,
292
+				'LIN_is_taxable' => false,
293
+				'LIN_order' => $price->order(),
294
+				'LIN_total' => $sign * $price_total,
295
+				'LIN_type' => EEM_Line_Item::type_sub_line_item,
296
+				'OBJ_ID' => $price->ID(),
297
+				'OBJ_type' => 'Price'
298
+			));
299
+			$sub_line_item = apply_filters(
300
+				'FHEE__EEH_Line_Item__create_ticket_line_item__sub_line_item',
301
+				$sub_line_item
302
+			);
303
+			if ($price->is_percent()) {
304
+				$sub_line_item->set_percent($sign * $price->amount());
305
+			} else {
306
+				$sub_line_item->set_unit_price($sign * $price->amount());
307
+			}
308
+			$running_total_for_ticket += $price_total;
309
+			$line_item->add_child_line_item($sub_line_item);
310
+		}
311
+		return $line_item;
312
+	}
313
+
314
+
315
+	/**
316
+	 * Adds the specified item under the pre-tax-sub-total line item. Automatically
317
+	 * re-calculates the line item totals and updates the related transaction. But
318
+	 * DOES NOT automatically upgrade the transaction's registrations' final prices (which
319
+	 * should probably change because of this).
320
+	 * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
321
+	 * after using this, to keep the registration final prices in-sync with the transaction's total.
322
+	 *
323
+	 * @param EE_Line_Item $total_line_item
324
+	 * @param EE_Line_Item $item to be added
325
+	 * @return boolean
326
+	 * @throws \EE_Error
327
+	 */
328
+	public static function add_item(EE_Line_Item $total_line_item, EE_Line_Item $item)
329
+	{
330
+		$pre_tax_subtotal = self::get_pre_tax_subtotal($total_line_item);
331
+		if ($pre_tax_subtotal instanceof EE_Line_Item) {
332
+			$success = $pre_tax_subtotal->add_child_line_item($item);
333
+		} else {
334
+			return false;
335
+		}
336
+		$total_line_item->recalculate_total_including_taxes();
337
+		return $success;
338
+	}
339
+
340
+
341
+	/**
342
+	 * cancels an existing ticket line item,
343
+	 * by decrementing it's quantity by 1 and adding a new "type_cancellation" sub-line-item.
344
+	 * ALL totals and subtotals will NEED TO BE UPDATED after performing this action
345
+	 *
346
+	 * @param EE_Line_Item $ticket_line_item
347
+	 * @param int $qty
348
+	 * @return bool success
349
+	 * @throws \EE_Error
350
+	 */
351
+	public static function cancel_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1)
352
+	{
353
+		// validate incoming line_item
354
+		if ($ticket_line_item->OBJ_type() !== 'Ticket') {
355
+			throw new EE_Error(
356
+				sprintf(
357
+					__('The supplied line item must have an Object Type of "Ticket", not %1$s.', 'event_espresso'),
358
+					$ticket_line_item->type()
359
+				)
360
+			);
361
+		}
362
+		if ($ticket_line_item->quantity() < $qty) {
363
+			throw new EE_Error(
364
+				sprintf(
365
+					__('Can not cancel %1$d ticket(s) because the supplied line item has a quantity of %2$d.', 'event_espresso'),
366
+					$qty,
367
+					$ticket_line_item->quantity()
368
+				)
369
+			);
370
+		}
371
+		// decrement ticket quantity; don't rely on auto-fixing when recalculating totals to do this
372
+		$ticket_line_item->set_quantity($ticket_line_item->quantity() - $qty);
373
+		foreach ($ticket_line_item->children() as $child_line_item) {
374
+			if ($child_line_item->is_sub_line_item()
375
+				&& !$child_line_item->is_percent()
376
+				&& !$child_line_item->is_cancellation()
377
+			) {
378
+				$child_line_item->set_quantity($child_line_item->quantity() - $qty);
379
+			}
380
+		}
381
+		// get cancellation sub line item
382
+		$cancellation_line_item = EEH_Line_Item::get_descendants_of_type(
383
+			$ticket_line_item,
384
+			EEM_Line_Item::type_cancellation
385
+		);
386
+		$cancellation_line_item = reset($cancellation_line_item);
387
+		// verify that this ticket was indeed previously cancelled
388
+		if ($cancellation_line_item instanceof EE_Line_Item) {
389
+			// increment cancelled quantity
390
+			$cancellation_line_item->set_quantity($cancellation_line_item->quantity() + $qty);
391
+		} else {
392
+			// create cancellation sub line item
393
+			$cancellation_line_item = EE_Line_Item::new_instance(array(
394
+				'LIN_name' => __('Cancellation', 'event_espresso'),
395
+				'LIN_desc' => sprintf(
396
+					_x('Cancelled %1$s : %2$s', 'Cancelled Ticket Name : 2015-01-01 11:11', 'event_espresso'),
397
+					$ticket_line_item->name(),
398
+					current_time(get_option('date_format') . ' ' . get_option('time_format'))
399
+				),
400
+				'LIN_unit_price' => 0, // $ticket_line_item->unit_price()
401
+				'LIN_quantity' => $qty,
402
+				'LIN_is_taxable' => $ticket_line_item->is_taxable(),
403
+				'LIN_order' => count($ticket_line_item->children()),
404
+				'LIN_total' => 0, // $ticket_line_item->unit_price()
405
+				'LIN_type' => EEM_Line_Item::type_cancellation,
406
+			));
407
+			$ticket_line_item->add_child_line_item($cancellation_line_item);
408
+		}
409
+		if ($ticket_line_item->save_this_and_descendants() > 0) {
410
+			// decrement parent line item quantity
411
+			$event_line_item = $ticket_line_item->parent();
412
+			if ($event_line_item instanceof EE_Line_Item && $event_line_item->OBJ_type() === 'Event') {
413
+				$event_line_item->set_quantity($event_line_item->quantity() - $qty);
414
+				$event_line_item->save();
415
+			}
416
+			EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item);
417
+			return true;
418
+		}
419
+		return false;
420
+	}
421
+
422
+
423
+	/**
424
+	 * reinstates (un-cancels?) a previously canceled ticket line item,
425
+	 * by incrementing it's quantity by 1, and decrementing it's "type_cancellation" sub-line-item.
426
+	 * ALL totals and subtotals will NEED TO BE UPDATED after performing this action
427
+	 *
428
+	 * @param EE_Line_Item $ticket_line_item
429
+	 * @param int $qty
430
+	 * @return bool success
431
+	 * @throws \EE_Error
432
+	 */
433
+	public static function reinstate_canceled_ticket_line_item(EE_Line_Item $ticket_line_item, $qty = 1)
434
+	{
435
+		// validate incoming line_item
436
+		if ($ticket_line_item->OBJ_type() !== 'Ticket') {
437
+			throw new EE_Error(
438
+				sprintf(
439
+					__('The supplied line item must have an Object Type of "Ticket", not %1$s.', 'event_espresso'),
440
+					$ticket_line_item->type()
441
+				)
442
+			);
443
+		}
444
+		// get cancellation sub line item
445
+		$cancellation_line_item = EEH_Line_Item::get_descendants_of_type(
446
+			$ticket_line_item,
447
+			EEM_Line_Item::type_cancellation
448
+		);
449
+		$cancellation_line_item = reset($cancellation_line_item);
450
+		// verify that this ticket was indeed previously cancelled
451
+		if (!$cancellation_line_item instanceof EE_Line_Item) {
452
+			return false;
453
+		}
454
+		if ($cancellation_line_item->quantity() > $qty) {
455
+			// decrement cancelled quantity
456
+			$cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty);
457
+		} elseif ($cancellation_line_item->quantity() == $qty) {
458
+			// decrement cancelled quantity in case anyone still has the object kicking around
459
+			$cancellation_line_item->set_quantity($cancellation_line_item->quantity() - $qty);
460
+			// delete because quantity will end up as 0
461
+			$cancellation_line_item->delete();
462
+			// and attempt to destroy the object,
463
+			// even though PHP won't actually destroy it until it needs the memory
464
+			unset($cancellation_line_item);
465
+		} else {
466
+			// what ?!?! negative quantity ?!?!
467
+			throw new EE_Error(
468
+				sprintf(
469
+					__(
470
+						'Can not reinstate %1$d cancelled ticket(s) because the cancelled ticket quantity is only %2$d.',
471
+						'event_espresso'
472
+					),
473
+					$qty,
474
+					$cancellation_line_item->quantity()
475
+				)
476
+			);
477
+		}
478
+		// increment ticket quantity
479
+		$ticket_line_item->set_quantity($ticket_line_item->quantity() + $qty);
480
+		if ($ticket_line_item->save_this_and_descendants() > 0) {
481
+			// increment parent line item quantity
482
+			$event_line_item = $ticket_line_item->parent();
483
+			if ($event_line_item instanceof EE_Line_Item && $event_line_item->OBJ_type() === 'Event') {
484
+				$event_line_item->set_quantity($event_line_item->quantity() + $qty);
485
+			}
486
+			EEH_Line_Item::get_grand_total_and_recalculate_everything($ticket_line_item);
487
+			return true;
488
+		}
489
+		return false;
490
+	}
491
+
492
+
493
+	/**
494
+	 * calls EEH_Line_Item::find_transaction_grand_total_for_line_item()
495
+	 * then EE_Line_Item::recalculate_total_including_taxes() on the result
496
+	 *
497
+	 * @param EE_Line_Item $line_item
498
+	 * @return \EE_Line_Item
499
+	 */
500
+	public static function get_grand_total_and_recalculate_everything(EE_Line_Item $line_item)
501
+	{
502
+		$grand_total_line_item = EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item);
503
+		return $grand_total_line_item->recalculate_total_including_taxes();
504
+	}
505
+
506
+
507
+	/**
508
+	 * Gets the line item which contains the subtotal of all the items
509
+	 *
510
+	 * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
511
+	 * @return \EE_Line_Item
512
+	 * @throws \EE_Error
513
+	 */
514
+	public static function get_pre_tax_subtotal(EE_Line_Item $total_line_item)
515
+	{
516
+		$pre_tax_subtotal = $total_line_item->get_child_line_item('pre-tax-subtotal');
517
+		return $pre_tax_subtotal instanceof EE_Line_Item
518
+			? $pre_tax_subtotal
519
+			: self::create_pre_tax_subtotal($total_line_item);
520
+	}
521
+
522
+
523
+	/**
524
+	 * Gets the line item for the taxes subtotal
525
+	 *
526
+	 * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
527
+	 * @return \EE_Line_Item
528
+	 * @throws \EE_Error
529
+	 */
530
+	public static function get_taxes_subtotal(EE_Line_Item $total_line_item)
531
+	{
532
+		$taxes = $total_line_item->get_child_line_item('taxes');
533
+		return $taxes ? $taxes : self::create_taxes_subtotal($total_line_item);
534
+	}
535
+
536
+
537
+	/**
538
+	 * sets the TXN ID on an EE_Line_Item if passed a valid EE_Transaction object
539
+	 *
540
+	 * @param EE_Line_Item $line_item
541
+	 * @param EE_Transaction $transaction
542
+	 * @return void
543
+	 * @throws \EE_Error
544
+	 */
545
+	public static function set_TXN_ID(EE_Line_Item $line_item, $transaction = null)
546
+	{
547
+		if ($transaction) {
548
+			/** @type EEM_Transaction $EEM_Transaction */
549
+			$EEM_Transaction = EE_Registry::instance()->load_model('Transaction');
550
+			$TXN_ID = $EEM_Transaction->ensure_is_ID($transaction);
551
+			$line_item->set_TXN_ID($TXN_ID);
552
+		}
553
+	}
554
+
555
+
556
+	/**
557
+	 * Creates a new default total line item for the transaction,
558
+	 * and its tickets subtotal and taxes subtotal line items (and adds the
559
+	 * existing taxes as children of the taxes subtotal line item)
560
+	 *
561
+	 * @param EE_Transaction $transaction
562
+	 * @return \EE_Line_Item of type total
563
+	 * @throws \EE_Error
564
+	 */
565
+	public static function create_total_line_item($transaction = null)
566
+	{
567
+		$total_line_item = EE_Line_Item::new_instance(array(
568
+			'LIN_code' => 'total',
569
+			'LIN_name' => __('Grand Total', 'event_espresso'),
570
+			'LIN_type' => EEM_Line_Item::type_total,
571
+			'OBJ_type' => 'Transaction'
572
+		));
573
+		$total_line_item = apply_filters(
574
+			'FHEE__EEH_Line_Item__create_total_line_item__total_line_item',
575
+			$total_line_item
576
+		);
577
+		self::set_TXN_ID($total_line_item, $transaction);
578
+		self::create_pre_tax_subtotal($total_line_item, $transaction);
579
+		self::create_taxes_subtotal($total_line_item, $transaction);
580
+		return $total_line_item;
581
+	}
582
+
583
+
584
+	/**
585
+	 * Creates a default items subtotal line item
586
+	 *
587
+	 * @param EE_Line_Item $total_line_item
588
+	 * @param EE_Transaction $transaction
589
+	 * @return EE_Line_Item
590
+	 * @throws \EE_Error
591
+	 */
592
+	protected static function create_pre_tax_subtotal(EE_Line_Item $total_line_item, $transaction = null)
593
+	{
594
+		$pre_tax_line_item = EE_Line_Item::new_instance(array(
595
+			'LIN_code' => 'pre-tax-subtotal',
596
+			'LIN_name' => __('Pre-Tax Subtotal', 'event_espresso'),
597
+			'LIN_type' => EEM_Line_Item::type_sub_total
598
+		));
599
+		$pre_tax_line_item = apply_filters(
600
+			'FHEE__EEH_Line_Item__create_pre_tax_subtotal__pre_tax_line_item',
601
+			$pre_tax_line_item
602
+		);
603
+		self::set_TXN_ID($pre_tax_line_item, $transaction);
604
+		$total_line_item->add_child_line_item($pre_tax_line_item);
605
+		self::create_event_subtotal($pre_tax_line_item, $transaction);
606
+		return $pre_tax_line_item;
607
+	}
608
+
609
+
610
+	/**
611
+	 * Creates a line item for the taxes subtotal and finds all the tax prices
612
+	 * and applies taxes to it
613
+	 *
614
+	 * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
615
+	 * @param EE_Transaction $transaction
616
+	 * @return EE_Line_Item
617
+	 * @throws \EE_Error
618
+	 */
619
+	protected static function create_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null)
620
+	{
621
+		$tax_line_item = EE_Line_Item::new_instance(array(
622
+			'LIN_code' => 'taxes',
623
+			'LIN_name' => __('Taxes', 'event_espresso'),
624
+			'LIN_type' => EEM_Line_Item::type_tax_sub_total,
625
+			'LIN_order' => 1000,// this should always come last
626
+		));
627
+		$tax_line_item = apply_filters(
628
+			'FHEE__EEH_Line_Item__create_taxes_subtotal__tax_line_item',
629
+			$tax_line_item
630
+		);
631
+		self::set_TXN_ID($tax_line_item, $transaction);
632
+		$total_line_item->add_child_line_item($tax_line_item);
633
+		// and lastly, add the actual taxes
634
+		self::apply_taxes($total_line_item);
635
+		return $tax_line_item;
636
+	}
637
+
638
+
639
+	/**
640
+	 * Creates a default items subtotal line item
641
+	 *
642
+	 * @param EE_Line_Item $pre_tax_line_item
643
+	 * @param EE_Transaction $transaction
644
+	 * @param EE_Event $event
645
+	 * @return EE_Line_Item
646
+	 * @throws \EE_Error
647
+	 */
648
+	public static function create_event_subtotal(EE_Line_Item $pre_tax_line_item, $transaction = null, $event = null)
649
+	{
650
+		$event_line_item = EE_Line_Item::new_instance(array(
651
+			'LIN_code' => self::get_event_code($event),
652
+			'LIN_name' => self::get_event_name($event),
653
+			'LIN_desc' => self::get_event_desc($event),
654
+			'LIN_type' => EEM_Line_Item::type_sub_total,
655
+			'OBJ_type' => 'Event',
656
+			'OBJ_ID' => $event instanceof EE_Event ? $event->ID() : 0
657
+		));
658
+		$event_line_item = apply_filters(
659
+			'FHEE__EEH_Line_Item__create_event_subtotal__event_line_item',
660
+			$event_line_item
661
+		);
662
+		self::set_TXN_ID($event_line_item, $transaction);
663
+		$pre_tax_line_item->add_child_line_item($event_line_item);
664
+		return $event_line_item;
665
+	}
666
+
667
+
668
+	/**
669
+	 * Gets what the event ticket's code SHOULD be
670
+	 *
671
+	 * @param EE_Event $event
672
+	 * @return string
673
+	 * @throws \EE_Error
674
+	 */
675
+	public static function get_event_code($event)
676
+	{
677
+		return 'event-' . ($event instanceof EE_Event ? $event->ID() : '0');
678
+	}
679
+
680
+	/**
681
+	 * Gets the event name
682
+	 * @param EE_Event $event
683
+	 * @return string
684
+	 */
685
+	public static function get_event_name($event)
686
+	{
687
+		return $event instanceof EE_Event ? $event->name() : __('Event', 'event_espresso');
688
+	}
689
+
690
+	/**
691
+	 * Gets the event excerpt
692
+	 * @param EE_Event $event
693
+	 * @return string
694
+	 */
695
+	public static function get_event_desc($event)
696
+	{
697
+		return $event instanceof EE_Event ? $event->short_description() : '';
698
+	}
699
+
700
+	/**
701
+	 * Given the grand total line item and a ticket, finds the event sub-total
702
+	 * line item the ticket's purchase should be added onto
703
+	 *
704
+	 * @access public
705
+	 * @param EE_Line_Item $grand_total the grand total line item
706
+	 * @param EE_Ticket $ticket
707
+	 * @throws \EE_Error
708
+	 * @return EE_Line_Item
709
+	 */
710
+	public static function get_event_line_item_for_ticket(EE_Line_Item $grand_total, EE_Ticket $ticket)
711
+	{
712
+		$first_datetime = $ticket->first_datetime();
713
+		if (!$first_datetime instanceof EE_Datetime) {
714
+			throw new EE_Error(
715
+				sprintf(__('The supplied ticket (ID %d) has no datetimes', 'event_espresso'), $ticket->ID())
716
+			);
717
+		}
718
+		$event = $first_datetime->event();
719
+		if (!$event instanceof EE_Event) {
720
+			throw new EE_Error(
721
+				sprintf(
722
+					__('The supplied ticket (ID %d) has no event data associated with it.', 'event_espresso'),
723
+					$ticket->ID()
724
+				)
725
+			);
726
+		}
727
+		$events_sub_total = EEH_Line_Item::get_event_line_item($grand_total, $event);
728
+		if (!$events_sub_total instanceof EE_Line_Item) {
729
+			throw new EE_Error(
730
+				sprintf(
731
+					__('There is no events sub-total for ticket %s on total line item %d', 'event_espresso'),
732
+					$ticket->ID(),
733
+					$grand_total->ID()
734
+				)
735
+			);
736
+		}
737
+		return $events_sub_total;
738
+	}
739
+
740
+
741
+	/**
742
+	 * Gets the event line item
743
+	 *
744
+	 * @param EE_Line_Item $grand_total
745
+	 * @param EE_Event $event
746
+	 * @return EE_Line_Item for the event subtotal which is a child of $grand_total
747
+	 * @throws \EE_Error
748
+	 */
749
+	public static function get_event_line_item(EE_Line_Item $grand_total, $event)
750
+	{
751
+		/** @type EE_Event $event */
752
+		$event = EEM_Event::instance()->ensure_is_obj($event, true);
753
+		$event_line_item = null;
754
+		$found = false;
755
+		foreach (EEH_Line_Item::get_event_subtotals($grand_total) as $event_line_item) {
756
+			// default event subtotal, we should only ever find this the first time this method is called
757
+			if (!$event_line_item->OBJ_ID()) {
758
+				// let's use this! but first... set the event details
759
+				EEH_Line_Item::set_event_subtotal_details($event_line_item, $event);
760
+				$found = true;
761
+				break;
762
+			} elseif ($event_line_item->OBJ_ID() === $event->ID()) {
763
+				// found existing line item for this event in the cart, so break out of loop and use this one
764
+				$found = true;
765
+				break;
766
+			}
767
+		}
768
+		if (!$found) {
769
+			// there is no event sub-total yet, so add it
770
+			$pre_tax_subtotal = EEH_Line_Item::get_pre_tax_subtotal($grand_total);
771
+			// create a new "event" subtotal below that
772
+			$event_line_item = EEH_Line_Item::create_event_subtotal($pre_tax_subtotal, null, $event);
773
+			// and set the event details
774
+			EEH_Line_Item::set_event_subtotal_details($event_line_item, $event);
775
+		}
776
+		return $event_line_item;
777
+	}
778
+
779
+
780
+	/**
781
+	 * Creates a default items subtotal line item
782
+	 *
783
+	 * @param EE_Line_Item $event_line_item
784
+	 * @param EE_Event $event
785
+	 * @param EE_Transaction $transaction
786
+	 * @return EE_Line_Item
787
+	 * @throws \EE_Error
788
+	 */
789
+	public static function set_event_subtotal_details(
790
+		EE_Line_Item $event_line_item,
791
+		EE_Event $event,
792
+		$transaction = null
793
+	) {
794
+		if ($event instanceof EE_Event) {
795
+			$event_line_item->set_code(self::get_event_code($event));
796
+			$event_line_item->set_name(self::get_event_name($event));
797
+			$event_line_item->set_desc(self::get_event_desc($event));
798
+			$event_line_item->set_OBJ_ID($event->ID());
799
+		}
800
+		self::set_TXN_ID($event_line_item, $transaction);
801
+	}
802
+
803
+
804
+	/**
805
+	 * Finds what taxes should apply, adds them as tax line items under the taxes sub-total,
806
+	 * and recalculates the taxes sub-total and the grand total. Resets the taxes, so
807
+	 * any old taxes are removed
808
+	 *
809
+	 * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
810
+	 * @throws \EE_Error
811
+	 */
812
+	public static function apply_taxes(EE_Line_Item $total_line_item)
813
+	{
814
+		/** @type EEM_Price $EEM_Price */
815
+		$EEM_Price = EE_Registry::instance()->load_model('Price');
816
+		// get array of taxes via Price Model
817
+		$ordered_taxes = $EEM_Price->get_all_prices_that_are_taxes();
818
+		ksort($ordered_taxes);
819
+		$taxes_line_item = self::get_taxes_subtotal($total_line_item);
820
+		// just to be safe, remove its old tax line items
821
+		$taxes_line_item->delete_children_line_items();
822
+		// loop thru taxes
823
+		foreach ($ordered_taxes as $order => $taxes) {
824
+			foreach ($taxes as $tax) {
825
+				if ($tax instanceof EE_Price) {
826
+					$tax_line_item = EE_Line_Item::new_instance(
827
+						array(
828
+							'LIN_name' => $tax->name(),
829
+							'LIN_desc' => $tax->desc(),
830
+							'LIN_percent' => $tax->amount(),
831
+							'LIN_is_taxable' => false,
832
+							'LIN_order' => $order,
833
+							'LIN_total' => 0,
834
+							'LIN_type' => EEM_Line_Item::type_tax,
835
+							'OBJ_type' => 'Price',
836
+							'OBJ_ID' => $tax->ID()
837
+						)
838
+					);
839
+					$tax_line_item = apply_filters(
840
+						'FHEE__EEH_Line_Item__apply_taxes__tax_line_item',
841
+						$tax_line_item
842
+					);
843
+					$taxes_line_item->add_child_line_item($tax_line_item);
844
+				}
845
+			}
846
+		}
847
+		$total_line_item->recalculate_total_including_taxes();
848
+	}
849
+
850
+
851
+	/**
852
+	 * Ensures that taxes have been applied to the order, if not applies them.
853
+	 * Returns the total amount of tax
854
+	 *
855
+	 * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
856
+	 * @return float
857
+	 * @throws \EE_Error
858
+	 */
859
+	public static function ensure_taxes_applied($total_line_item)
860
+	{
861
+		$taxes_subtotal = self::get_taxes_subtotal($total_line_item);
862
+		if (!$taxes_subtotal->children()) {
863
+			self::apply_taxes($total_line_item);
864
+		}
865
+		return $taxes_subtotal->total();
866
+	}
867
+
868
+
869
+	/**
870
+	 * Deletes ALL children of the passed line item
871
+	 *
872
+	 * @param EE_Line_Item $parent_line_item
873
+	 * @return bool
874
+	 * @throws \EE_Error
875
+	 */
876
+	public static function delete_all_child_items(EE_Line_Item $parent_line_item)
877
+	{
878
+		$deleted = 0;
879
+		foreach ($parent_line_item->children() as $child_line_item) {
880
+			if ($child_line_item instanceof EE_Line_Item) {
881
+				$deleted += EEH_Line_Item::delete_all_child_items($child_line_item);
882
+				if ($child_line_item->ID()) {
883
+					$child_line_item->delete();
884
+					unset($child_line_item);
885
+				} else {
886
+					$parent_line_item->delete_child_line_item($child_line_item->code());
887
+				}
888
+				$deleted++;
889
+			}
890
+		}
891
+		return $deleted;
892
+	}
893
+
894
+
895
+	/**
896
+	 * Deletes the line items as indicated by the line item code(s) provided,
897
+	 * regardless of where they're found in the line item tree. Automatically
898
+	 * re-calculates the line item totals and updates the related transaction. But
899
+	 * DOES NOT automatically upgrade the transaction's registrations' final prices (which
900
+	 * should probably change because of this).
901
+	 * You should call EE_Registration_Processor::calculate_reg_final_prices_per_line_item()
902
+	 * after using this, to keep the registration final prices in-sync with the transaction's total.
903
+	 * @param EE_Line_Item $total_line_item of type EEM_Line_Item::type_total
904
+	 * @param array|bool|string $line_item_codes
905
+	 * @return int number of items successfully removed
906
+	 */
907
+	public static function delete_items(EE_Line_Item $total_line_item, $line_item_codes = false)
908
+	{
909
+
910
+		if ($total_line_item->type() !== EEM_Line_Item::type_total) {
911
+			EE_Error::doing_it_wrong(
912
+				'EEH_Line_Item::delete_items',
913
+				__(
914
+					'This static method should only be called with a TOTAL line item, otherwise we won\'t recalculate the totals correctly',
915
+					'event_espresso'
916
+				),
917
+				'4.6.18'
918
+			);
919
+		}
920
+		do_action('AHEE_log', __FILE__, __FUNCTION__, '');
921
+
922
+		// check if only a single line_item_id was passed
923
+		if (!empty($line_item_codes) && !is_array($line_item_codes)) {
924
+			// place single line_item_id in an array to appear as multiple line_item_ids
925
+			$line_item_codes = array($line_item_codes);
926
+		}
927
+		$removals = 0;
928
+		// cycle thru line_item_ids
929
+		foreach ($line_item_codes as $line_item_id) {
930
+			$removals += $total_line_item->delete_child_line_item($line_item_id);
931
+		}
932
+
933
+		if ($removals > 0) {
934
+			$total_line_item->recalculate_taxes_and_tax_total();
935
+			return $removals;
936
+		} else {
937
+			return false;
938
+		}
939
+	}
940
+
941
+
942
+	/**
943
+	 * Overwrites the previous tax by clearing out the old taxes, and creates a new
944
+	 * tax and updates the total line item accordingly
945
+	 *
946
+	 * @param EE_Line_Item $total_line_item
947
+	 * @param float $amount
948
+	 * @param string $name
949
+	 * @param string $description
950
+	 * @param string $code
951
+	 * @param boolean $add_to_existing_line_item
952
+	 *                          if true, and a duplicate line item with the same code is found,
953
+	 *                          $amount will be added onto it; otherwise will simply set the taxes to match $amount
954
+	 * @return EE_Line_Item the new tax line item created
955
+	 * @throws \EE_Error
956
+	 */
957
+	public static function set_total_tax_to(
958
+		EE_Line_Item $total_line_item,
959
+		$amount,
960
+		$name = null,
961
+		$description = null,
962
+		$code = null,
963
+		$add_to_existing_line_item = false
964
+	) {
965
+		$tax_subtotal = self::get_taxes_subtotal($total_line_item);
966
+		$taxable_total = $total_line_item->taxable_total();
967
+
968
+		if ($add_to_existing_line_item) {
969
+			$new_tax = $tax_subtotal->get_child_line_item($code);
970
+			EEM_Line_Item::instance()->delete(
971
+				array(array('LIN_code' => array('!=', $code), 'LIN_parent' => $tax_subtotal->ID()))
972
+			);
973
+		} else {
974
+			$new_tax = null;
975
+			$tax_subtotal->delete_children_line_items();
976
+		}
977
+		if ($new_tax) {
978
+			$new_tax->set_total($new_tax->total() + $amount);
979
+			$new_tax->set_percent($taxable_total ? $new_tax->total() / $taxable_total * 100 : 0);
980
+		} else {
981
+			// no existing tax item. Create it
982
+			$new_tax = EE_Line_Item::new_instance(array(
983
+				'TXN_ID' => $total_line_item->TXN_ID(),
984
+				'LIN_name' => $name ? $name : __('Tax', 'event_espresso'),
985
+				'LIN_desc' => $description ? $description : '',
986
+				'LIN_percent' => $taxable_total ? ($amount / $taxable_total * 100) : 0,
987
+				'LIN_total' => $amount,
988
+				'LIN_parent' => $tax_subtotal->ID(),
989
+				'LIN_type' => EEM_Line_Item::type_tax,
990
+				'LIN_code' => $code
991
+			));
992
+		}
993
+
994
+		$new_tax = apply_filters(
995
+			'FHEE__EEH_Line_Item__set_total_tax_to__new_tax_subtotal',
996
+			$new_tax,
997
+			$total_line_item
998
+		);
999
+		$new_tax->save();
1000
+		$tax_subtotal->set_total($new_tax->total());
1001
+		$tax_subtotal->save();
1002
+		$total_line_item->recalculate_total_including_taxes();
1003
+		return $new_tax;
1004
+	}
1005
+
1006
+
1007
+	/**
1008
+	 * Makes all the line items which are children of $line_item taxable (or not).
1009
+	 * Does NOT save the line items
1010
+	 * @param EE_Line_Item $line_item
1011
+	 * @param string $code_substring_for_whitelist if this string is part of the line item's code
1012
+	 *  it will be whitelisted (ie, except from becoming taxable)
1013
+	 * @param boolean $taxable
1014
+	 */
1015
+	public static function set_line_items_taxable(
1016
+		EE_Line_Item $line_item,
1017
+		$taxable = true,
1018
+		$code_substring_for_whitelist = null
1019
+	) {
1020
+		$whitelisted = false;
1021
+		if ($code_substring_for_whitelist !== null) {
1022
+			$whitelisted = strpos($line_item->code(), $code_substring_for_whitelist) !== false ? true : false;
1023
+		}
1024
+		if (!$whitelisted && $line_item->is_line_item()) {
1025
+			$line_item->set_is_taxable($taxable);
1026
+		}
1027
+		foreach ($line_item->children() as $child_line_item) {
1028
+			EEH_Line_Item::set_line_items_taxable($child_line_item, $taxable, $code_substring_for_whitelist);
1029
+		}
1030
+	}
1031
+
1032
+
1033
+	/**
1034
+	 * Gets all descendants that are event subtotals
1035
+	 *
1036
+	 * @uses  EEH_Line_Item::get_subtotals_of_object_type()
1037
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1038
+	 * @return EE_Line_Item[]
1039
+	 */
1040
+	public static function get_event_subtotals(EE_Line_Item $parent_line_item)
1041
+	{
1042
+		return self::get_subtotals_of_object_type($parent_line_item, 'Event');
1043
+	}
1044
+
1045
+
1046
+	/**
1047
+	 * Gets all descendants subtotals that match the supplied object type
1048
+	 *
1049
+	 * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1050
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1051
+	 * @param string $obj_type
1052
+	 * @return EE_Line_Item[]
1053
+	 */
1054
+	public static function get_subtotals_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '')
1055
+	{
1056
+		return self::_get_descendants_by_type_and_object_type(
1057
+			$parent_line_item,
1058
+			EEM_Line_Item::type_sub_total,
1059
+			$obj_type
1060
+		);
1061
+	}
1062
+
1063
+
1064
+	/**
1065
+	 * Gets all descendants that are tickets
1066
+	 *
1067
+	 * @uses  EEH_Line_Item::get_line_items_of_object_type()
1068
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1069
+	 * @return EE_Line_Item[]
1070
+	 */
1071
+	public static function get_ticket_line_items(EE_Line_Item $parent_line_item)
1072
+	{
1073
+		return self::get_line_items_of_object_type($parent_line_item, 'Ticket');
1074
+	}
1075
+
1076
+
1077
+	/**
1078
+	 * Gets all descendants subtotals that match the supplied object type
1079
+	 *
1080
+	 * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1081
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1082
+	 * @param string $obj_type
1083
+	 * @return EE_Line_Item[]
1084
+	 */
1085
+	public static function get_line_items_of_object_type(EE_Line_Item $parent_line_item, $obj_type = '')
1086
+	{
1087
+		return self::_get_descendants_by_type_and_object_type($parent_line_item, EEM_Line_Item::type_line_item, $obj_type);
1088
+	}
1089
+
1090
+
1091
+	/**
1092
+	 * Gets all the descendants (ie, children or children of children etc) that are of the type 'tax'
1093
+	 * @uses  EEH_Line_Item::get_descendants_of_type()
1094
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1095
+	 * @return EE_Line_Item[]
1096
+	 */
1097
+	public static function get_tax_descendants(EE_Line_Item $parent_line_item)
1098
+	{
1099
+		return EEH_Line_Item::get_descendants_of_type($parent_line_item, EEM_Line_Item::type_tax);
1100
+	}
1101
+
1102
+
1103
+	/**
1104
+	 * Gets all the real items purchased which are children of this item
1105
+	 * @uses  EEH_Line_Item::get_descendants_of_type()
1106
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1107
+	 * @return EE_Line_Item[]
1108
+	 */
1109
+	public static function get_line_item_descendants(EE_Line_Item $parent_line_item)
1110
+	{
1111
+		return EEH_Line_Item::get_descendants_of_type($parent_line_item, EEM_Line_Item::type_line_item);
1112
+	}
1113
+
1114
+
1115
+	/**
1116
+	 * Gets all descendants of supplied line item that match the supplied line item type
1117
+	 *
1118
+	 * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1119
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1120
+	 * @param string $line_item_type one of the EEM_Line_Item constants
1121
+	 * @return EE_Line_Item[]
1122
+	 */
1123
+	public static function get_descendants_of_type(EE_Line_Item $parent_line_item, $line_item_type)
1124
+	{
1125
+		return self::_get_descendants_by_type_and_object_type($parent_line_item, $line_item_type, null);
1126
+	}
1127
+
1128
+
1129
+	/**
1130
+	 * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type as well
1131
+	 *
1132
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1133
+	 * @param string $line_item_type one of the EEM_Line_Item constants
1134
+	 * @param string | NULL $obj_type object model class name (minus prefix) or NULL to ignore object type when searching
1135
+	 * @return EE_Line_Item[]
1136
+	 */
1137
+	protected static function _get_descendants_by_type_and_object_type(
1138
+		EE_Line_Item $parent_line_item,
1139
+		$line_item_type,
1140
+		$obj_type = null
1141
+	) {
1142
+		$objects = array();
1143
+		foreach ($parent_line_item->children() as $child_line_item) {
1144
+			if ($child_line_item instanceof EE_Line_Item) {
1145
+				if ($child_line_item->type() === $line_item_type
1146
+					&& (
1147
+						$child_line_item->OBJ_type() === $obj_type || $obj_type === null
1148
+					)
1149
+				) {
1150
+					$objects[] = $child_line_item;
1151
+				} else {
1152
+					// go-through-all-its children looking for more matches
1153
+					$objects = array_merge(
1154
+						$objects,
1155
+						self::_get_descendants_by_type_and_object_type(
1156
+							$child_line_item,
1157
+							$line_item_type,
1158
+							$obj_type
1159
+						)
1160
+					);
1161
+				}
1162
+			}
1163
+		}
1164
+		return $objects;
1165
+	}
1166
+
1167
+
1168
+	/**
1169
+	 * Gets all descendants subtotals that match the supplied object type
1170
+	 *
1171
+	 * @uses  EEH_Line_Item::_get_descendants_by_type_and_object_type()
1172
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1173
+	 * @param string $OBJ_type object type (like Event)
1174
+	 * @param array $OBJ_IDs array of OBJ_IDs
1175
+	 * @return EE_Line_Item[]
1176
+	 */
1177
+	public static function get_line_items_by_object_type_and_IDs(
1178
+		EE_Line_Item $parent_line_item,
1179
+		$OBJ_type = '',
1180
+		$OBJ_IDs = array()
1181
+	) {
1182
+		return self::_get_descendants_by_object_type_and_object_ID($parent_line_item, $OBJ_type, $OBJ_IDs);
1183
+	}
1184
+
1185
+
1186
+	/**
1187
+	 * Gets all descendants of supplied line item that match the supplied line item type and possibly the object type as well
1188
+	 *
1189
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1190
+	 * @param string $OBJ_type object type (like Event)
1191
+	 * @param array $OBJ_IDs array of OBJ_IDs
1192
+	 * @return EE_Line_Item[]
1193
+	 */
1194
+	protected static function _get_descendants_by_object_type_and_object_ID(
1195
+		EE_Line_Item $parent_line_item,
1196
+		$OBJ_type,
1197
+		$OBJ_IDs
1198
+	) {
1199
+		$objects = array();
1200
+		foreach ($parent_line_item->children() as $child_line_item) {
1201
+			if ($child_line_item instanceof EE_Line_Item) {
1202
+				if ($child_line_item->OBJ_type() === $OBJ_type
1203
+					&& is_array($OBJ_IDs)
1204
+					&& in_array($child_line_item->OBJ_ID(), $OBJ_IDs)
1205
+				) {
1206
+					$objects[] = $child_line_item;
1207
+				} else {
1208
+					// go-through-all-its children looking for more matches
1209
+					$objects = array_merge(
1210
+						$objects,
1211
+						self::_get_descendants_by_object_type_and_object_ID(
1212
+							$child_line_item,
1213
+							$OBJ_type,
1214
+							$OBJ_IDs
1215
+						)
1216
+					);
1217
+				}
1218
+			}
1219
+		}
1220
+		return $objects;
1221
+	}
1222
+
1223
+
1224
+	/**
1225
+	 * Uses a breadth-first-search in order to find the nearest descendant of
1226
+	 * the specified type and returns it, else NULL
1227
+	 *
1228
+	 * @uses  EEH_Line_Item::_get_nearest_descendant()
1229
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1230
+	 * @param string $type like one of the EEM_Line_Item::type_*
1231
+	 * @return EE_Line_Item
1232
+	 */
1233
+	public static function get_nearest_descendant_of_type(EE_Line_Item $parent_line_item, $type)
1234
+	{
1235
+		return self::_get_nearest_descendant($parent_line_item, 'LIN_type', $type);
1236
+	}
1237
+
1238
+
1239
+	/**
1240
+	 * Uses a breadth-first-search in order to find the nearest descendant
1241
+	 * having the specified LIN_code and returns it, else NULL
1242
+	 *
1243
+	 * @uses  EEH_Line_Item::_get_nearest_descendant()
1244
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1245
+	 * @param string $code any value used for LIN_code
1246
+	 * @return EE_Line_Item
1247
+	 */
1248
+	public static function get_nearest_descendant_having_code(EE_Line_Item $parent_line_item, $code)
1249
+	{
1250
+		return self::_get_nearest_descendant($parent_line_item, 'LIN_code', $code);
1251
+	}
1252
+
1253
+
1254
+	/**
1255
+	 * Uses a breadth-first-search in order to find the nearest descendant
1256
+	 * having the specified LIN_code and returns it, else NULL
1257
+	 *
1258
+	 * @param \EE_Line_Item $parent_line_item - the line item to find descendants of
1259
+	 * @param string $search_field name of EE_Line_Item property
1260
+	 * @param string $value any value stored in $search_field
1261
+	 * @return EE_Line_Item
1262
+	 */
1263
+	protected static function _get_nearest_descendant(EE_Line_Item $parent_line_item, $search_field, $value)
1264
+	{
1265
+		foreach ($parent_line_item->children() as $child) {
1266
+			if ($child->get($search_field) == $value) {
1267
+				return $child;
1268
+			}
1269
+		}
1270
+		foreach ($parent_line_item->children() as $child) {
1271
+			$descendant_found = self::_get_nearest_descendant($child, $search_field, $value);
1272
+			if ($descendant_found) {
1273
+				return $descendant_found;
1274
+			}
1275
+		}
1276
+		return null;
1277
+	}
1278
+
1279
+
1280
+	/**
1281
+	 * if passed line item has a TXN ID, uses that to jump directly to the grand total line item for the transaction,
1282
+	 * else recursively walks up the line item tree until a parent of type total is found,
1283
+	 *
1284
+	 * @param EE_Line_Item $line_item
1285
+	 * @return \EE_Line_Item
1286
+	 * @throws \EE_Error
1287
+	 */
1288
+	public static function find_transaction_grand_total_for_line_item(EE_Line_Item $line_item)
1289
+	{
1290
+		if ($line_item->TXN_ID()) {
1291
+			$total_line_item = $line_item->transaction()->total_line_item(false);
1292
+			if ($total_line_item instanceof EE_Line_Item) {
1293
+				return $total_line_item;
1294
+			}
1295
+		} else {
1296
+			$line_item_parent = $line_item->parent();
1297
+			if ($line_item_parent instanceof EE_Line_Item) {
1298
+				if ($line_item_parent->is_total()) {
1299
+					return $line_item_parent;
1300
+				}
1301
+				return EEH_Line_Item::find_transaction_grand_total_for_line_item($line_item_parent);
1302
+			}
1303
+		}
1304
+		throw new EE_Error(
1305
+			sprintf(
1306
+				__('A valid grand total for line item %1$d was not found.', 'event_espresso'),
1307
+				$line_item->ID()
1308
+			)
1309
+		);
1310
+	}
1311
+
1312
+
1313
+	/**
1314
+	 * Prints out a representation of the line item tree
1315
+	 *
1316
+	 * @param EE_Line_Item $line_item
1317
+	 * @param int $indentation
1318
+	 * @return void
1319
+	 * @throws \EE_Error
1320
+	 */
1321
+	public static function visualize(EE_Line_Item $line_item, $indentation = 0)
1322
+	{
1323
+		echo defined('EE_TESTS_DIR') ? "\n" : '<br />';
1324
+		if (!$indentation) {
1325
+			echo defined('EE_TESTS_DIR') ? "\n" : '<br />';
1326
+		}
1327
+		for ($i = 0; $i < $indentation; $i++) {
1328
+			echo ". ";
1329
+		}
1330
+		$breakdown = '';
1331
+		if ($line_item->is_line_item()) {
1332
+			if ($line_item->is_percent()) {
1333
+				$breakdown = "{$line_item->percent()}%";
1334
+			} else {
1335
+				$breakdown = '$' . "{$line_item->unit_price()} x {$line_item->quantity()}";
1336
+			}
1337
+		}
1338
+		echo $line_item->name() . " [ ID:{$line_item->ID()} | qty:{$line_item->quantity()} ] {$line_item->type()} : " . '$' . "{$line_item->total()}";
1339
+		if ($breakdown) {
1340
+			echo " ( {$breakdown} )";
1341
+		}
1342
+		if ($line_item->is_taxable()) {
1343
+			echo "  * taxable";
1344
+		}
1345
+		if ($line_item->children()) {
1346
+			foreach ($line_item->children() as $child) {
1347
+				self::visualize($child, $indentation + 1);
1348
+			}
1349
+		}
1350
+	}
1351
+
1352
+
1353
+	/**
1354
+	 * Calculates the registration's final price, taking into account that they
1355
+	 * need to not only help pay for their OWN ticket, but also any transaction-wide surcharges and taxes,
1356
+	 * and receive a portion of any transaction-wide discounts.
1357
+	 * eg1, if I buy a $1 ticket and brent buys a $9 ticket, and we receive a $5 discount
1358
+	 * then I'll get 1/10 of that $5 discount, which is $0.50, and brent will get
1359
+	 * 9/10ths of that $5 discount, which is $4.50. So my final price should be $0.50
1360
+	 * and brent's final price should be $5.50.
1361
+	 *
1362
+	 * In order to do this, we basically need to traverse the line item tree calculating
1363
+	 * the running totals (just as if we were recalculating the total), but when we identify
1364
+	 * regular line items, we need to keep track of their share of the grand total.
1365
+	 * Also, we need to keep track of the TAXABLE total for each ticket purchase, so
1366
+	 * we can know how to apply taxes to it. (Note: "taxable total" does not equal the "pretax total"
1367
+	 * when there are non-taxable items; otherwise they would be the same)
1368
+	 *
1369
+	 * @param EE_Line_Item $line_item
1370
+	 * @param array $billable_ticket_quantities array of EE_Ticket IDs and their corresponding quantity that
1371
+	 *                                                                            can be included in price calculations at this moment
1372
+	 * @return array        keys are line items for tickets IDs and values are their share of the running total,
1373
+	 *                                          plus the key 'total', and 'taxable' which also has keys of all the ticket IDs. Eg
1374
+	 *                                          array(
1375
+	 *                                          12 => 4.3
1376
+	 *                                          23 => 8.0
1377
+	 *                                          'total' => 16.6,
1378
+	 *                                          'taxable' => array(
1379
+	 *                                          12 => 10,
1380
+	 *                                          23 => 4
1381
+	 *                                          ).
1382
+	 *                                          So to find which registrations have which final price, we need to find which line item
1383
+	 *                                          is theirs, which can be done with
1384
+	 *                                          `EEM_Line_Item::instance()->get_line_item_for_registration( $registration );`
1385
+	 */
1386
+	public static function calculate_reg_final_prices_per_line_item(EE_Line_Item $line_item, $billable_ticket_quantities = array())
1387
+	{
1388
+		// init running grand total if not already
1389
+		if (!isset($running_totals['total'])) {
1390
+			$running_totals['total'] = 0;
1391
+		}
1392
+		if (!isset($running_totals['taxable'])) {
1393
+			$running_totals['taxable'] = array('total' => 0);
1394
+		}
1395
+		foreach ($line_item->children() as $child_line_item) {
1396
+			switch ($child_line_item->type()) {
1397
+				case EEM_Line_Item::type_sub_total:
1398
+					$running_totals_from_subtotal = EEH_Line_Item::calculate_reg_final_prices_per_line_item($child_line_item, $billable_ticket_quantities);
1399
+					// combine arrays but preserve numeric keys
1400
+					$running_totals = array_replace_recursive($running_totals_from_subtotal, $running_totals);
1401
+					$running_totals['total'] += $running_totals_from_subtotal['total'];
1402
+					$running_totals['taxable']['total'] += $running_totals_from_subtotal['taxable']['total'];
1403
+					break;
1404
+
1405
+				case EEM_Line_Item::type_tax_sub_total:
1406
+					// find how much the taxes percentage is
1407
+					if ($child_line_item->percent() !== 0) {
1408
+						$tax_percent_decimal = $child_line_item->percent() / 100;
1409
+					} else {
1410
+						$tax_percent_decimal = EE_Taxes::get_total_taxes_percentage() / 100;
1411
+					}
1412
+					// and apply to all the taxable totals, and add to the pretax totals
1413
+					foreach ($running_totals as $line_item_id => $this_running_total) {
1414
+						// "total" and "taxable" array key is an exception
1415
+						if ($line_item_id === 'taxable') {
1416
+							continue;
1417
+						}
1418
+						$taxable_total = $running_totals['taxable'][ $line_item_id ];
1419
+						$running_totals[ $line_item_id ] += ($taxable_total * $tax_percent_decimal);
1420
+					}
1421
+					break;
1422
+
1423
+				case EEM_Line_Item::type_line_item:
1424
+					// ticket line items or ????
1425
+					if ($child_line_item->OBJ_type() === 'Ticket') {
1426
+						// kk it's a ticket
1427
+						if (isset($running_totals[ $child_line_item->ID() ])) {
1428
+							// huh? that shouldn't happen.
1429
+							$running_totals['total'] += $child_line_item->total();
1430
+						} else {
1431
+							// its not in our running totals yet. great.
1432
+							if ($child_line_item->is_taxable()) {
1433
+								$taxable_amount = $child_line_item->unit_price();
1434
+							} else {
1435
+								$taxable_amount = 0;
1436
+							}
1437
+							// are we only calculating totals for some tickets?
1438
+							if (isset($billable_ticket_quantities[ $child_line_item->OBJ_ID() ])) {
1439
+								$quantity = $billable_ticket_quantities[ $child_line_item->OBJ_ID() ];
1440
+								$running_totals[ $child_line_item->ID() ] = $quantity
1441
+									? $child_line_item->unit_price()
1442
+									: 0;
1443
+								$running_totals['taxable'][ $child_line_item->ID() ] = $quantity
1444
+									? $taxable_amount
1445
+									: 0;
1446
+							} else {
1447
+								$quantity = $child_line_item->quantity();
1448
+								$running_totals[ $child_line_item->ID() ] = $child_line_item->unit_price();
1449
+								$running_totals['taxable'][ $child_line_item->ID() ] = $taxable_amount;
1450
+							}
1451
+							$running_totals['taxable']['total'] += $taxable_amount * $quantity;
1452
+							$running_totals['total'] += $child_line_item->unit_price() * $quantity;
1453
+						}
1454
+					} else {
1455
+						// it's some other type of item added to the cart
1456
+						// it should affect the running totals
1457
+						// basically we want to convert it into a PERCENT modifier. Because
1458
+						// more clearly affect all registration's final price equally
1459
+						$line_items_percent_of_running_total = $running_totals['total'] > 0
1460
+							? ($child_line_item->total() / $running_totals['total']) + 1
1461
+							: 1;
1462
+						foreach ($running_totals as $line_item_id => $this_running_total) {
1463
+							// the "taxable" array key is an exception
1464
+							if ($line_item_id === 'taxable') {
1465
+								continue;
1466
+							}
1467
+							// update the running totals
1468
+							// yes this actually even works for the running grand total!
1469
+							$running_totals[ $line_item_id ] =
1470
+								$line_items_percent_of_running_total * $this_running_total;
1471
+
1472
+							if ($child_line_item->is_taxable()) {
1473
+								$running_totals['taxable'][ $line_item_id ] =
1474
+									$line_items_percent_of_running_total * $running_totals['taxable'][ $line_item_id ];
1475
+							}
1476
+						}
1477
+					}
1478
+					break;
1479
+			}
1480
+		}
1481
+		return $running_totals;
1482
+	}
1483
+
1484
+
1485
+	/**
1486
+	 * @param \EE_Line_Item $total_line_item
1487
+	 * @param \EE_Line_Item $ticket_line_item
1488
+	 * @return float | null
1489
+	 * @throws \OutOfRangeException
1490
+	 */
1491
+	public static function calculate_final_price_for_ticket_line_item(\EE_Line_Item $total_line_item, \EE_Line_Item $ticket_line_item)
1492
+	{
1493
+		static $final_prices_per_ticket_line_item = array();
1494
+		if (empty($final_prices_per_ticket_line_item)) {
1495
+			$final_prices_per_ticket_line_item = \EEH_Line_Item::calculate_reg_final_prices_per_line_item(
1496
+				$total_line_item
1497
+			);
1498
+		}
1499
+		// ok now find this new registration's final price
1500
+		if (isset($final_prices_per_ticket_line_item[ $ticket_line_item->ID() ])) {
1501
+			return $final_prices_per_ticket_line_item[ $ticket_line_item->ID() ];
1502
+		}
1503
+		$message = sprintf(
1504
+			__(
1505
+				'The final price for the ticket line item (ID:%1$d) could not be calculated.',
1506
+				'event_espresso'
1507
+			),
1508
+			$ticket_line_item->ID()
1509
+		);
1510
+		if (WP_DEBUG) {
1511
+			$message .= '<br>' . print_r($final_prices_per_ticket_line_item, true);
1512
+			throw new \OutOfRangeException($message);
1513
+		} else {
1514
+			EE_Log::instance()->log(__CLASS__, __FUNCTION__, $message);
1515
+		}
1516
+		return null;
1517
+	}
1518
+
1519
+
1520
+	/**
1521
+	 * Creates a duplicate of the line item tree, except only includes billable items
1522
+	 * and the portion of line items attributed to billable things
1523
+	 *
1524
+	 * @param EE_Line_Item $line_item
1525
+	 * @param EE_Registration[] $registrations
1526
+	 * @return \EE_Line_Item
1527
+	 * @throws \EE_Error
1528
+	 */
1529
+	public static function billable_line_item_tree(EE_Line_Item $line_item, $registrations)
1530
+	{
1531
+		$copy_li = EEH_Line_Item::billable_line_item($line_item, $registrations);
1532
+		foreach ($line_item->children() as $child_li) {
1533
+			$copy_li->add_child_line_item(EEH_Line_Item::billable_line_item_tree($child_li, $registrations));
1534
+		}
1535
+		// if this is the grand total line item, make sure the totals all add up
1536
+		// (we could have duplicated this logic AS we copied the line items, but
1537
+		// it seems DRYer this way)
1538
+		if ($copy_li->type() === EEM_Line_Item::type_total) {
1539
+			$copy_li->recalculate_total_including_taxes();
1540
+		}
1541
+		return $copy_li;
1542
+	}
1543
+
1544
+
1545
+	/**
1546
+	 * Creates a new, unsaved line item from $line_item that factors in the
1547
+	 * number of billable registrations on $registrations.
1548
+	 *
1549
+	 * @param EE_Line_Item $line_item
1550
+	 * @return EE_Line_Item
1551
+	 * @throws \EE_Error
1552
+	 * @param EE_Registration[] $registrations
1553
+	 */
1554
+	public static function billable_line_item(EE_Line_Item $line_item, $registrations)
1555
+	{
1556
+		$new_li_fields = $line_item->model_field_array();
1557
+		if ($line_item->type() === EEM_Line_Item::type_line_item &&
1558
+			$line_item->OBJ_type() === 'Ticket'
1559
+		) {
1560
+			$count = 0;
1561
+			foreach ($registrations as $registration) {
1562
+				if ($line_item->OBJ_ID() === $registration->ticket_ID() &&
1563
+					in_array($registration->status_ID(), EEM_Registration::reg_statuses_that_allow_payment())
1564
+				) {
1565
+					$count++;
1566
+				}
1567
+			}
1568
+			$new_li_fields['LIN_quantity'] = $count;
1569
+		}
1570
+		// don't set the total. We'll leave that up to the code that calculates it
1571
+		unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent'], $new_li_fields['LIN_total']);
1572
+		return EE_Line_Item::new_instance($new_li_fields);
1573
+	}
1574
+
1575
+
1576
+	/**
1577
+	 * Returns a modified line item tree where all the subtotals which have a total of 0
1578
+	 * are removed, and line items with a quantity of 0
1579
+	 *
1580
+	 * @param EE_Line_Item $line_item |null
1581
+	 * @return \EE_Line_Item|null
1582
+	 * @throws \EE_Error
1583
+	 */
1584
+	public static function non_empty_line_items(EE_Line_Item $line_item)
1585
+	{
1586
+		$copied_li = EEH_Line_Item::non_empty_line_item($line_item);
1587
+		if ($copied_li === null) {
1588
+			return null;
1589
+		}
1590
+		// if this is an event subtotal, we want to only include it if it
1591
+		// has a non-zero total and at least one ticket line item child
1592
+		$ticket_children = 0;
1593
+		foreach ($line_item->children() as $child_li) {
1594
+			$child_li_copy = EEH_Line_Item::non_empty_line_items($child_li);
1595
+			if ($child_li_copy !== null) {
1596
+				$copied_li->add_child_line_item($child_li_copy);
1597
+				if ($child_li_copy->type() === EEM_Line_Item::type_line_item &&
1598
+					$child_li_copy->OBJ_type() === 'Ticket'
1599
+				) {
1600
+					$ticket_children++;
1601
+				}
1602
+			}
1603
+		}
1604
+		// if this is an event subtotal with NO ticket children
1605
+		// we basically want to ignore it
1606
+		if ($ticket_children === 0
1607
+			&& $line_item->type() === EEM_Line_Item::type_sub_total
1608
+			&& $line_item->OBJ_type() === 'Event'
1609
+			&& $line_item->total() === 0
1610
+		) {
1611
+			return null;
1612
+		}
1613
+		return $copied_li;
1614
+	}
1615
+
1616
+
1617
+	/**
1618
+	 * Creates a new, unsaved line item, but if it's a ticket line item
1619
+	 * with a total of 0, or a subtotal of 0, returns null instead
1620
+	 *
1621
+	 * @param EE_Line_Item $line_item
1622
+	 * @return EE_Line_Item
1623
+	 * @throws \EE_Error
1624
+	 */
1625
+	public static function non_empty_line_item(EE_Line_Item $line_item)
1626
+	{
1627
+		if ($line_item->type() === EEM_Line_Item::type_line_item &&
1628
+			$line_item->OBJ_type() === 'Ticket' &&
1629
+			$line_item->quantity() === 0
1630
+		) {
1631
+			return null;
1632
+		}
1633
+		$new_li_fields = $line_item->model_field_array();
1634
+		// don't set the total. We'll leave that up to the code that calculates it
1635
+		unset($new_li_fields['LIN_ID'], $new_li_fields['LIN_parent']);
1636
+		return EE_Line_Item::new_instance($new_li_fields);
1637
+	}
1638
+
1639
+
1640
+
1641
+	/**************************************** @DEPRECATED METHODS *************************************** */
1642
+	/**
1643
+	 * @deprecated
1644
+	 * @param EE_Line_Item $total_line_item
1645
+	 * @return \EE_Line_Item
1646
+	 * @throws \EE_Error
1647
+	 */
1648
+	public static function get_items_subtotal(EE_Line_Item $total_line_item)
1649
+	{
1650
+		EE_Error::doing_it_wrong('EEH_Line_Item::get_items_subtotal()', __('Method replaced with EEH_Line_Item::get_pre_tax_subtotal()', 'event_espresso'), '4.6.0');
1651
+		return self::get_pre_tax_subtotal($total_line_item);
1652
+	}
1653
+
1654
+
1655
+	/**
1656
+	 * @deprecated
1657
+	 * @param EE_Transaction $transaction
1658
+	 * @return \EE_Line_Item
1659
+	 * @throws \EE_Error
1660
+	 */
1661
+	public static function create_default_total_line_item($transaction = null)
1662
+	{
1663
+		EE_Error::doing_it_wrong('EEH_Line_Item::create_default_total_line_item()', __('Method replaced with EEH_Line_Item::create_total_line_item()', 'event_espresso'), '4.6.0');
1664
+		return self::create_total_line_item($transaction);
1665
+	}
1666
+
1667
+
1668
+	/**
1669
+	 * @deprecated
1670
+	 * @param EE_Line_Item $total_line_item
1671
+	 * @param EE_Transaction $transaction
1672
+	 * @return \EE_Line_Item
1673
+	 * @throws \EE_Error
1674
+	 */
1675
+	public static function create_default_tickets_subtotal(EE_Line_Item $total_line_item, $transaction = null)
1676
+	{
1677
+		EE_Error::doing_it_wrong('EEH_Line_Item::create_default_tickets_subtotal()', __('Method replaced with EEH_Line_Item::create_pre_tax_subtotal()', 'event_espresso'), '4.6.0');
1678
+		return self::create_pre_tax_subtotal($total_line_item, $transaction);
1679
+	}
1680
+
1681
+
1682
+	/**
1683
+	 * @deprecated
1684
+	 * @param EE_Line_Item $total_line_item
1685
+	 * @param EE_Transaction $transaction
1686
+	 * @return \EE_Line_Item
1687
+	 * @throws \EE_Error
1688
+	 */
1689
+	public static function create_default_taxes_subtotal(EE_Line_Item $total_line_item, $transaction = null)
1690
+	{
1691
+		EE_Error::doing_it_wrong('EEH_Line_Item::create_default_taxes_subtotal()', __('Method replaced with EEH_Line_Item::create_taxes_subtotal()', 'event_espresso'), '4.6.0');
1692
+		return self::create_taxes_subtotal($total_line_item, $transaction);
1693
+	}
1694
+
1695
+
1696
+	/**
1697
+	 * @deprecated
1698
+	 * @param EE_Line_Item $total_line_item
1699
+	 * @param EE_Transaction $transaction
1700
+	 * @return \EE_Line_Item
1701
+	 * @throws \EE_Error
1702
+	 */
1703
+	public static function create_default_event_subtotal(EE_Line_Item $total_line_item, $transaction = null)
1704
+	{
1705
+		EE_Error::doing_it_wrong('EEH_Line_Item::create_default_event_subtotal()', __('Method replaced with EEH_Line_Item::create_event_subtotal()', 'event_espresso'), '4.6.0');
1706
+		return self::create_event_subtotal($total_line_item, $transaction);
1707
+	}
1708 1708
 }
Please login to merge, or discard this patch.
Spacing   +40 added lines, -40 removed lines patch added patch discarded remove patch
@@ -122,13 +122,13 @@  discard block
 block discarded – undo
122 122
      */
123 123
     public static function add_ticket_purchase(EE_Line_Item $total_line_item, EE_Ticket $ticket, $qty = 1)
124 124
     {
125
-        if (!$total_line_item instanceof EE_Line_Item || !$total_line_item->is_total()) {
125
+        if ( ! $total_line_item instanceof EE_Line_Item || ! $total_line_item->is_total()) {
126 126
             throw new EE_Error(sprintf(__('A valid line item total is required in order to add tickets. A line item of type "%s" was passed.', 'event_espresso'), $ticket->ID(), $total_line_item->ID()));
127 127
         }
128 128
         // either increment the qty for an existing ticket
129 129
         $line_item = self::increment_ticket_qty_if_already_in_cart($total_line_item, $ticket, $qty);
130 130
         // or add a new one
131
-        if (!$line_item instanceof EE_Line_Item) {
131
+        if ( ! $line_item instanceof EE_Line_Item) {
132 132
             $line_item = self::create_ticket_line_item($total_line_item, $ticket, $qty);
133 133
         }
134 134
         $total_line_item->recalculate_total_including_taxes();
@@ -177,7 +177,7 @@  discard block
 block discarded – undo
177 177
      */
178 178
     public static function increment_quantity(EE_Line_Item $line_item, $qty = 1)
179 179
     {
180
-        if (!$line_item->is_percent()) {
180
+        if ( ! $line_item->is_percent()) {
181 181
             $qty += $line_item->quantity();
182 182
             $line_item->set_quantity($qty);
183 183
             $line_item->set_total($line_item->unit_price() * $qty);
@@ -202,7 +202,7 @@  discard block
 block discarded – undo
202 202
      */
203 203
     public static function decrement_quantity(EE_Line_Item $line_item, $qty = 1)
204 204
     {
205
-        if (!$line_item->is_percent()) {
205
+        if ( ! $line_item->is_percent()) {
206 206
             $qty = $line_item->quantity() - $qty;
207 207
             $qty = max($qty, 0);
208 208
             $line_item->set_quantity($qty);
@@ -227,7 +227,7 @@  discard block
 block discarded – undo
227 227
      */
228 228
     public static function update_quantity(EE_Line_Item $line_item, $new_quantity)
229 229
     {
230
-        if (!$line_item->is_percent()) {
230
+        if ( ! $line_item->is_percent()) {
231 231
             $line_item->set_quantity($new_quantity);
232 232
             $line_item->set_total($line_item->unit_price() * $new_quantity);
233 233
             $line_item->save();
@@ -263,7 +263,7 @@  discard block
 block discarded – undo
263 263
         // add $ticket to cart
264 264
         $line_item = EE_Line_Item::new_instance(array(
265 265
             'LIN_name' => $ticket->name(),
266
-            'LIN_desc' => $ticket->description() !== '' ? $ticket->description() . ' ' . $event : $event,
266
+            'LIN_desc' => $ticket->description() !== '' ? $ticket->description().' '.$event : $event,
267 267
             'LIN_unit_price' => $ticket->price(),
268 268
             'LIN_quantity' => $qty,
269 269
             'LIN_is_taxable' => $ticket->taxable(),
@@ -372,8 +372,8 @@  discard block
 block discarded – undo
372 372
         $ticket_line_item->set_quantity($ticket_line_item->quantity() - $qty);
373 373
         foreach ($ticket_line_item->children() as $child_line_item) {
374 374
             if ($child_line_item->is_sub_line_item()
375
-                && !$child_line_item->is_percent()
376
-                && !$child_line_item->is_cancellation()
375
+                && ! $child_line_item->is_percent()
376
+                && ! $child_line_item->is_cancellation()
377 377
             ) {
378 378
                 $child_line_item->set_quantity($child_line_item->quantity() - $qty);
379 379
             }
@@ -395,7 +395,7 @@  discard block
 block discarded – undo
395 395
                 'LIN_desc' => sprintf(
396 396
                     _x('Cancelled %1$s : %2$s', 'Cancelled Ticket Name : 2015-01-01 11:11', 'event_espresso'),
397 397
                     $ticket_line_item->name(),
398
-                    current_time(get_option('date_format') . ' ' . get_option('time_format'))
398
+                    current_time(get_option('date_format').' '.get_option('time_format'))
399 399
                 ),
400 400
                 'LIN_unit_price' => 0, // $ticket_line_item->unit_price()
401 401
                 'LIN_quantity' => $qty,
@@ -448,7 +448,7 @@  discard block
 block discarded – undo
448 448
         );
449 449
         $cancellation_line_item = reset($cancellation_line_item);
450 450
         // verify that this ticket was indeed previously cancelled
451
-        if (!$cancellation_line_item instanceof EE_Line_Item) {
451
+        if ( ! $cancellation_line_item instanceof EE_Line_Item) {
452 452
             return false;
453 453
         }
454 454
         if ($cancellation_line_item->quantity() > $qty) {
@@ -622,7 +622,7 @@  discard block
 block discarded – undo
622 622
             'LIN_code' => 'taxes',
623 623
             'LIN_name' => __('Taxes', 'event_espresso'),
624 624
             'LIN_type' => EEM_Line_Item::type_tax_sub_total,
625
-            'LIN_order' => 1000,// this should always come last
625
+            'LIN_order' => 1000, // this should always come last
626 626
         ));
627 627
         $tax_line_item = apply_filters(
628 628
             'FHEE__EEH_Line_Item__create_taxes_subtotal__tax_line_item',
@@ -674,7 +674,7 @@  discard block
 block discarded – undo
674 674
      */
675 675
     public static function get_event_code($event)
676 676
     {
677
-        return 'event-' . ($event instanceof EE_Event ? $event->ID() : '0');
677
+        return 'event-'.($event instanceof EE_Event ? $event->ID() : '0');
678 678
     }
679 679
 
680 680
     /**
@@ -710,13 +710,13 @@  discard block
 block discarded – undo
710 710
     public static function get_event_line_item_for_ticket(EE_Line_Item $grand_total, EE_Ticket $ticket)
711 711
     {
712 712
         $first_datetime = $ticket->first_datetime();
713
-        if (!$first_datetime instanceof EE_Datetime) {
713
+        if ( ! $first_datetime instanceof EE_Datetime) {
714 714
             throw new EE_Error(
715 715
                 sprintf(__('The supplied ticket (ID %d) has no datetimes', 'event_espresso'), $ticket->ID())
716 716
             );
717 717
         }
718 718
         $event = $first_datetime->event();
719
-        if (!$event instanceof EE_Event) {
719
+        if ( ! $event instanceof EE_Event) {
720 720
             throw new EE_Error(
721 721
                 sprintf(
722 722
                     __('The supplied ticket (ID %d) has no event data associated with it.', 'event_espresso'),
@@ -725,7 +725,7 @@  discard block
 block discarded – undo
725 725
             );
726 726
         }
727 727
         $events_sub_total = EEH_Line_Item::get_event_line_item($grand_total, $event);
728
-        if (!$events_sub_total instanceof EE_Line_Item) {
728
+        if ( ! $events_sub_total instanceof EE_Line_Item) {
729 729
             throw new EE_Error(
730 730
                 sprintf(
731 731
                     __('There is no events sub-total for ticket %s on total line item %d', 'event_espresso'),
@@ -754,7 +754,7 @@  discard block
 block discarded – undo
754 754
         $found = false;
755 755
         foreach (EEH_Line_Item::get_event_subtotals($grand_total) as $event_line_item) {
756 756
             // default event subtotal, we should only ever find this the first time this method is called
757
-            if (!$event_line_item->OBJ_ID()) {
757
+            if ( ! $event_line_item->OBJ_ID()) {
758 758
                 // let's use this! but first... set the event details
759 759
                 EEH_Line_Item::set_event_subtotal_details($event_line_item, $event);
760 760
                 $found = true;
@@ -765,7 +765,7 @@  discard block
 block discarded – undo
765 765
                 break;
766 766
             }
767 767
         }
768
-        if (!$found) {
768
+        if ( ! $found) {
769 769
             // there is no event sub-total yet, so add it
770 770
             $pre_tax_subtotal = EEH_Line_Item::get_pre_tax_subtotal($grand_total);
771 771
             // create a new "event" subtotal below that
@@ -859,7 +859,7 @@  discard block
 block discarded – undo
859 859
     public static function ensure_taxes_applied($total_line_item)
860 860
     {
861 861
         $taxes_subtotal = self::get_taxes_subtotal($total_line_item);
862
-        if (!$taxes_subtotal->children()) {
862
+        if ( ! $taxes_subtotal->children()) {
863 863
             self::apply_taxes($total_line_item);
864 864
         }
865 865
         return $taxes_subtotal->total();
@@ -920,7 +920,7 @@  discard block
 block discarded – undo
920 920
         do_action('AHEE_log', __FILE__, __FUNCTION__, '');
921 921
 
922 922
         // check if only a single line_item_id was passed
923
-        if (!empty($line_item_codes) && !is_array($line_item_codes)) {
923
+        if ( ! empty($line_item_codes) && ! is_array($line_item_codes)) {
924 924
             // place single line_item_id in an array to appear as multiple line_item_ids
925 925
             $line_item_codes = array($line_item_codes);
926 926
         }
@@ -1021,7 +1021,7 @@  discard block
 block discarded – undo
1021 1021
         if ($code_substring_for_whitelist !== null) {
1022 1022
             $whitelisted = strpos($line_item->code(), $code_substring_for_whitelist) !== false ? true : false;
1023 1023
         }
1024
-        if (!$whitelisted && $line_item->is_line_item()) {
1024
+        if ( ! $whitelisted && $line_item->is_line_item()) {
1025 1025
             $line_item->set_is_taxable($taxable);
1026 1026
         }
1027 1027
         foreach ($line_item->children() as $child_line_item) {
@@ -1321,7 +1321,7 @@  discard block
 block discarded – undo
1321 1321
     public static function visualize(EE_Line_Item $line_item, $indentation = 0)
1322 1322
     {
1323 1323
         echo defined('EE_TESTS_DIR') ? "\n" : '<br />';
1324
-        if (!$indentation) {
1324
+        if ( ! $indentation) {
1325 1325
             echo defined('EE_TESTS_DIR') ? "\n" : '<br />';
1326 1326
         }
1327 1327
         for ($i = 0; $i < $indentation; $i++) {
@@ -1332,10 +1332,10 @@  discard block
 block discarded – undo
1332 1332
             if ($line_item->is_percent()) {
1333 1333
                 $breakdown = "{$line_item->percent()}%";
1334 1334
             } else {
1335
-                $breakdown = '$' . "{$line_item->unit_price()} x {$line_item->quantity()}";
1335
+                $breakdown = '$'."{$line_item->unit_price()} x {$line_item->quantity()}";
1336 1336
             }
1337 1337
         }
1338
-        echo $line_item->name() . " [ ID:{$line_item->ID()} | qty:{$line_item->quantity()} ] {$line_item->type()} : " . '$' . "{$line_item->total()}";
1338
+        echo $line_item->name()." [ ID:{$line_item->ID()} | qty:{$line_item->quantity()} ] {$line_item->type()} : ".'$'."{$line_item->total()}";
1339 1339
         if ($breakdown) {
1340 1340
             echo " ( {$breakdown} )";
1341 1341
         }
@@ -1386,10 +1386,10 @@  discard block
 block discarded – undo
1386 1386
     public static function calculate_reg_final_prices_per_line_item(EE_Line_Item $line_item, $billable_ticket_quantities = array())
1387 1387
     {
1388 1388
         // init running grand total if not already
1389
-        if (!isset($running_totals['total'])) {
1389
+        if ( ! isset($running_totals['total'])) {
1390 1390
             $running_totals['total'] = 0;
1391 1391
         }
1392
-        if (!isset($running_totals['taxable'])) {
1392
+        if ( ! isset($running_totals['taxable'])) {
1393 1393
             $running_totals['taxable'] = array('total' => 0);
1394 1394
         }
1395 1395
         foreach ($line_item->children() as $child_line_item) {
@@ -1415,8 +1415,8 @@  discard block
 block discarded – undo
1415 1415
                         if ($line_item_id === 'taxable') {
1416 1416
                             continue;
1417 1417
                         }
1418
-                        $taxable_total = $running_totals['taxable'][ $line_item_id ];
1419
-                        $running_totals[ $line_item_id ] += ($taxable_total * $tax_percent_decimal);
1418
+                        $taxable_total = $running_totals['taxable'][$line_item_id];
1419
+                        $running_totals[$line_item_id] += ($taxable_total * $tax_percent_decimal);
1420 1420
                     }
1421 1421
                     break;
1422 1422
 
@@ -1424,7 +1424,7 @@  discard block
 block discarded – undo
1424 1424
                     // ticket line items or ????
1425 1425
                     if ($child_line_item->OBJ_type() === 'Ticket') {
1426 1426
                         // kk it's a ticket
1427
-                        if (isset($running_totals[ $child_line_item->ID() ])) {
1427
+                        if (isset($running_totals[$child_line_item->ID()])) {
1428 1428
                             // huh? that shouldn't happen.
1429 1429
                             $running_totals['total'] += $child_line_item->total();
1430 1430
                         } else {
@@ -1435,18 +1435,18 @@  discard block
 block discarded – undo
1435 1435
                                 $taxable_amount = 0;
1436 1436
                             }
1437 1437
                             // are we only calculating totals for some tickets?
1438
-                            if (isset($billable_ticket_quantities[ $child_line_item->OBJ_ID() ])) {
1439
-                                $quantity = $billable_ticket_quantities[ $child_line_item->OBJ_ID() ];
1440
-                                $running_totals[ $child_line_item->ID() ] = $quantity
1438
+                            if (isset($billable_ticket_quantities[$child_line_item->OBJ_ID()])) {
1439
+                                $quantity = $billable_ticket_quantities[$child_line_item->OBJ_ID()];
1440
+                                $running_totals[$child_line_item->ID()] = $quantity
1441 1441
                                     ? $child_line_item->unit_price()
1442 1442
                                     : 0;
1443
-                                $running_totals['taxable'][ $child_line_item->ID() ] = $quantity
1443
+                                $running_totals['taxable'][$child_line_item->ID()] = $quantity
1444 1444
                                     ? $taxable_amount
1445 1445
                                     : 0;
1446 1446
                             } else {
1447 1447
                                 $quantity = $child_line_item->quantity();
1448
-                                $running_totals[ $child_line_item->ID() ] = $child_line_item->unit_price();
1449
-                                $running_totals['taxable'][ $child_line_item->ID() ] = $taxable_amount;
1448
+                                $running_totals[$child_line_item->ID()] = $child_line_item->unit_price();
1449
+                                $running_totals['taxable'][$child_line_item->ID()] = $taxable_amount;
1450 1450
                             }
1451 1451
                             $running_totals['taxable']['total'] += $taxable_amount * $quantity;
1452 1452
                             $running_totals['total'] += $child_line_item->unit_price() * $quantity;
@@ -1466,12 +1466,12 @@  discard block
 block discarded – undo
1466 1466
                             }
1467 1467
                             // update the running totals
1468 1468
                             // yes this actually even works for the running grand total!
1469
-                            $running_totals[ $line_item_id ] =
1469
+                            $running_totals[$line_item_id] =
1470 1470
                                 $line_items_percent_of_running_total * $this_running_total;
1471 1471
 
1472 1472
                             if ($child_line_item->is_taxable()) {
1473
-                                $running_totals['taxable'][ $line_item_id ] =
1474
-                                    $line_items_percent_of_running_total * $running_totals['taxable'][ $line_item_id ];
1473
+                                $running_totals['taxable'][$line_item_id] =
1474
+                                    $line_items_percent_of_running_total * $running_totals['taxable'][$line_item_id];
1475 1475
                             }
1476 1476
                         }
1477 1477
                     }
@@ -1497,8 +1497,8 @@  discard block
 block discarded – undo
1497 1497
             );
1498 1498
         }
1499 1499
         // ok now find this new registration's final price
1500
-        if (isset($final_prices_per_ticket_line_item[ $ticket_line_item->ID() ])) {
1501
-            return $final_prices_per_ticket_line_item[ $ticket_line_item->ID() ];
1500
+        if (isset($final_prices_per_ticket_line_item[$ticket_line_item->ID()])) {
1501
+            return $final_prices_per_ticket_line_item[$ticket_line_item->ID()];
1502 1502
         }
1503 1503
         $message = sprintf(
1504 1504
             __(
@@ -1508,7 +1508,7 @@  discard block
 block discarded – undo
1508 1508
             $ticket_line_item->ID()
1509 1509
         );
1510 1510
         if (WP_DEBUG) {
1511
-            $message .= '<br>' . print_r($final_prices_per_ticket_line_item, true);
1511
+            $message .= '<br>'.print_r($final_prices_per_ticket_line_item, true);
1512 1512
             throw new \OutOfRangeException($message);
1513 1513
         } else {
1514 1514
             EE_Log::instance()->log(__CLASS__, __FUNCTION__, $message);
Please login to merge, or discard this patch.
core/helpers/EEH_DTT_Helper.helper.php 3 patches
Doc Comments   +5 added lines, -5 removed lines patch added patch discarded remove patch
@@ -197,8 +197,8 @@  discard block
 block discarded – undo
197 197
      * Get Timezone Offset for given timezone object.
198 198
      *
199 199
      * @param DateTimeZone $date_time_zone
200
-     * @param null         $time
201
-     * @return mixed
200
+     * @param integer|null         $time
201
+     * @return integer
202 202
      * @throws InvalidArgumentException
203 203
      * @throws InvalidDataTypeException
204 204
      * @throws InvalidInterfaceException
@@ -320,7 +320,7 @@  discard block
 block discarded – undo
320 320
      * @param  DateTime   $DateTime DateTime object
321 321
      * @param  string     $period   a value to indicate what interval is being used in the calculation. The options are
322 322
      *                              'years', 'months', 'days', 'hours', 'minutes', 'seconds'. Defaults to years.
323
-     * @param  int|string $value    What you want to increment the date by
323
+     * @param  integer $value    What you want to increment the date by
324 324
      * @param  string     $operand  What operand you wish to use for the calculation
325 325
      * @return DateTime return whatever type came in.
326 326
      * @throws Exception
@@ -856,7 +856,7 @@  discard block
 block discarded – undo
856 856
      * this method will add that "1" into your date regardless of the format.
857 857
      *
858 858
      * @param string $month
859
-     * @return string
859
+     * @return integer
860 860
      */
861 861
     public static function first_of_month_timestamp($month = '')
862 862
     {
@@ -1004,7 +1004,7 @@  discard block
 block discarded – undo
1004 1004
     /**
1005 1005
      * Shim for the WP function `get_user_locale` that was added in WordPress 4.7.0
1006 1006
      *
1007
-     * @param int|WP_User $user_id
1007
+     * @param integer $user_id
1008 1008
      * @return string
1009 1009
      */
1010 1010
     public static function get_user_locale($user_id = 0)
Please login to merge, or discard this patch.
Indentation   +981 added lines, -981 removed lines patch added patch discarded remove patch
@@ -17,1043 +17,1043 @@
 block discarded – undo
17 17
 {
18 18
 
19 19
 
20
-    /**
21
-     * return the timezone set for the WP install
22
-     *
23
-     * @return string valid timezone string for PHP DateTimeZone() class
24
-     * @throws InvalidArgumentException
25
-     * @throws InvalidDataTypeException
26
-     * @throws InvalidInterfaceException
27
-     */
28
-    public static function get_timezone()
29
-    {
30
-        return EEH_DTT_Helper::get_valid_timezone_string();
31
-    }
20
+	/**
21
+	 * return the timezone set for the WP install
22
+	 *
23
+	 * @return string valid timezone string for PHP DateTimeZone() class
24
+	 * @throws InvalidArgumentException
25
+	 * @throws InvalidDataTypeException
26
+	 * @throws InvalidInterfaceException
27
+	 */
28
+	public static function get_timezone()
29
+	{
30
+		return EEH_DTT_Helper::get_valid_timezone_string();
31
+	}
32 32
 
33 33
 
34
-    /**
35
-     * get_valid_timezone_string
36
-     *    ensures that a valid timezone string is returned
37
-     *
38
-     * @param string $timezone_string
39
-     * @return string
40
-     * @throws InvalidArgumentException
41
-     * @throws InvalidDataTypeException
42
-     * @throws InvalidInterfaceException
43
-     */
44
-    public static function get_valid_timezone_string($timezone_string = '')
45
-    {
46
-        return self::getHelperAdapter()->getValidTimezoneString($timezone_string);
47
-    }
34
+	/**
35
+	 * get_valid_timezone_string
36
+	 *    ensures that a valid timezone string is returned
37
+	 *
38
+	 * @param string $timezone_string
39
+	 * @return string
40
+	 * @throws InvalidArgumentException
41
+	 * @throws InvalidDataTypeException
42
+	 * @throws InvalidInterfaceException
43
+	 */
44
+	public static function get_valid_timezone_string($timezone_string = '')
45
+	{
46
+		return self::getHelperAdapter()->getValidTimezoneString($timezone_string);
47
+	}
48 48
 
49 49
 
50
-    /**
51
-     * This only purpose for this static method is to validate that the incoming timezone is a valid php timezone.
52
-     *
53
-     * @static
54
-     * @param  string $timezone_string Timezone string to check
55
-     * @param bool    $throw_error
56
-     * @return bool
57
-     * @throws InvalidArgumentException
58
-     * @throws InvalidDataTypeException
59
-     * @throws InvalidInterfaceException
60
-     */
61
-    public static function validate_timezone($timezone_string, $throw_error = true)
62
-    {
63
-        return self::getHelperAdapter()->validateTimezone($timezone_string, $throw_error);
64
-    }
50
+	/**
51
+	 * This only purpose for this static method is to validate that the incoming timezone is a valid php timezone.
52
+	 *
53
+	 * @static
54
+	 * @param  string $timezone_string Timezone string to check
55
+	 * @param bool    $throw_error
56
+	 * @return bool
57
+	 * @throws InvalidArgumentException
58
+	 * @throws InvalidDataTypeException
59
+	 * @throws InvalidInterfaceException
60
+	 */
61
+	public static function validate_timezone($timezone_string, $throw_error = true)
62
+	{
63
+		return self::getHelperAdapter()->validateTimezone($timezone_string, $throw_error);
64
+	}
65 65
 
66 66
 
67
-    /**
68
-     * This returns a string that can represent the provided gmt offset in format that can be passed into
69
-     * DateTimeZone.  This is NOT a string that can be passed as a value on the WordPress timezone_string option.
70
-     *
71
-     * @param float|string $gmt_offset
72
-     * @return string
73
-     * @throws InvalidArgumentException
74
-     * @throws InvalidDataTypeException
75
-     * @throws InvalidInterfaceException
76
-     */
77
-    public static function get_timezone_string_from_gmt_offset($gmt_offset = '')
78
-    {
79
-        return self::getHelperAdapter()->getTimezoneStringFromGmtOffset($gmt_offset);
80
-    }
67
+	/**
68
+	 * This returns a string that can represent the provided gmt offset in format that can be passed into
69
+	 * DateTimeZone.  This is NOT a string that can be passed as a value on the WordPress timezone_string option.
70
+	 *
71
+	 * @param float|string $gmt_offset
72
+	 * @return string
73
+	 * @throws InvalidArgumentException
74
+	 * @throws InvalidDataTypeException
75
+	 * @throws InvalidInterfaceException
76
+	 */
77
+	public static function get_timezone_string_from_gmt_offset($gmt_offset = '')
78
+	{
79
+		return self::getHelperAdapter()->getTimezoneStringFromGmtOffset($gmt_offset);
80
+	}
81 81
 
82 82
 
83
-    /**
84
-     * Gets the site's GMT offset based on either the timezone string
85
-     * (in which case teh gmt offset will vary depending on the location's
86
-     * observance of daylight savings time) or the gmt_offset wp option
87
-     *
88
-     * @return int seconds offset
89
-     * @throws InvalidArgumentException
90
-     * @throws InvalidDataTypeException
91
-     * @throws InvalidInterfaceException
92
-     */
93
-    public static function get_site_timezone_gmt_offset()
94
-    {
95
-        return self::getHelperAdapter()->getSiteTimezoneGmtOffset();
96
-    }
83
+	/**
84
+	 * Gets the site's GMT offset based on either the timezone string
85
+	 * (in which case teh gmt offset will vary depending on the location's
86
+	 * observance of daylight savings time) or the gmt_offset wp option
87
+	 *
88
+	 * @return int seconds offset
89
+	 * @throws InvalidArgumentException
90
+	 * @throws InvalidDataTypeException
91
+	 * @throws InvalidInterfaceException
92
+	 */
93
+	public static function get_site_timezone_gmt_offset()
94
+	{
95
+		return self::getHelperAdapter()->getSiteTimezoneGmtOffset();
96
+	}
97 97
 
98 98
 
99
-    /**
100
-     * Depending on PHP version,
101
-     * there might not be valid current timezone strings to match these gmt_offsets in its timezone tables.
102
-     * To get around that, for these fringe timezones we bump them to a known valid offset.
103
-     * This method should ONLY be called after first verifying an timezone_string cannot be retrieved for the offset.
104
-     *
105
-     * @deprecated 4.9.54.rc    Developers this was always meant to only be an internally used method.  This will be
106
-     *                          removed in a future version of EE.
107
-     * @param int $gmt_offset
108
-     * @return int
109
-     * @throws InvalidArgumentException
110
-     * @throws InvalidDataTypeException
111
-     * @throws InvalidInterfaceException
112
-     */
113
-    public static function adjust_invalid_gmt_offsets($gmt_offset = 0)
114
-    {
115
-        return self::getHelperAdapter()->adjustInvalidGmtOffsets($gmt_offset);
116
-    }
99
+	/**
100
+	 * Depending on PHP version,
101
+	 * there might not be valid current timezone strings to match these gmt_offsets in its timezone tables.
102
+	 * To get around that, for these fringe timezones we bump them to a known valid offset.
103
+	 * This method should ONLY be called after first verifying an timezone_string cannot be retrieved for the offset.
104
+	 *
105
+	 * @deprecated 4.9.54.rc    Developers this was always meant to only be an internally used method.  This will be
106
+	 *                          removed in a future version of EE.
107
+	 * @param int $gmt_offset
108
+	 * @return int
109
+	 * @throws InvalidArgumentException
110
+	 * @throws InvalidDataTypeException
111
+	 * @throws InvalidInterfaceException
112
+	 */
113
+	public static function adjust_invalid_gmt_offsets($gmt_offset = 0)
114
+	{
115
+		return self::getHelperAdapter()->adjustInvalidGmtOffsets($gmt_offset);
116
+	}
117 117
 
118 118
 
119
-    /**
120
-     * get_timezone_string_from_abbreviations_list
121
-     *
122
-     * @deprecated 4.9.54.rc  Developers, this was never intended to be public.  This is a soft deprecation for now.
123
-     *                        If you are using this, you'll want to work out an alternate way of getting the value.
124
-     * @param int  $gmt_offset
125
-     * @param bool $coerce If true, we attempt to coerce with our adjustment table @see self::adjust_invalid_gmt_offset.
126
-     * @return string
127
-     * @throws EE_Error
128
-     * @throws InvalidArgumentException
129
-     * @throws InvalidDataTypeException
130
-     * @throws InvalidInterfaceException
131
-     */
132
-    public static function get_timezone_string_from_abbreviations_list($gmt_offset = 0, $coerce = true)
133
-    {
134
-        $gmt_offset =  (int) $gmt_offset;
135
-        /** @var array[] $abbreviations */
136
-        $abbreviations = DateTimeZone::listAbbreviations();
137
-        foreach ($abbreviations as $abbreviation) {
138
-            foreach ($abbreviation as $timezone) {
139
-                if ((int) $timezone['offset'] === $gmt_offset && (bool) $timezone['dst'] === false) {
140
-                    try {
141
-                        $offset = self::get_timezone_offset(new DateTimeZone($timezone['timezone_id']));
142
-                        if ($offset !== $gmt_offset) {
143
-                            continue;
144
-                        }
145
-                        return $timezone['timezone_id'];
146
-                    } catch (Exception $e) {
147
-                        continue;
148
-                    }
149
-                }
150
-            }
151
-        }
152
-        // if $coerce is true, let's see if we can get a timezone string after the offset is adjusted
153
-        if ($coerce === true) {
154
-            $timezone_string = self::get_timezone_string_from_abbreviations_list(
155
-                self::adjust_invalid_gmt_offsets($gmt_offset),
156
-                false
157
-            );
158
-            if ($timezone_string) {
159
-                return $timezone_string;
160
-            }
161
-        }
162
-        throw new EE_Error(
163
-            sprintf(
164
-                esc_html__(
165
-                    'The provided GMT offset (%1$s), is invalid, please check with %2$sthis list%3$s for what valid timezones can be used',
166
-                    'event_espresso'
167
-                ),
168
-                $gmt_offset / HOUR_IN_SECONDS,
169
-                '<a href="http://www.php.net/manual/en/timezones.php">',
170
-                '</a>'
171
-            )
172
-        );
173
-    }
119
+	/**
120
+	 * get_timezone_string_from_abbreviations_list
121
+	 *
122
+	 * @deprecated 4.9.54.rc  Developers, this was never intended to be public.  This is a soft deprecation for now.
123
+	 *                        If you are using this, you'll want to work out an alternate way of getting the value.
124
+	 * @param int  $gmt_offset
125
+	 * @param bool $coerce If true, we attempt to coerce with our adjustment table @see self::adjust_invalid_gmt_offset.
126
+	 * @return string
127
+	 * @throws EE_Error
128
+	 * @throws InvalidArgumentException
129
+	 * @throws InvalidDataTypeException
130
+	 * @throws InvalidInterfaceException
131
+	 */
132
+	public static function get_timezone_string_from_abbreviations_list($gmt_offset = 0, $coerce = true)
133
+	{
134
+		$gmt_offset =  (int) $gmt_offset;
135
+		/** @var array[] $abbreviations */
136
+		$abbreviations = DateTimeZone::listAbbreviations();
137
+		foreach ($abbreviations as $abbreviation) {
138
+			foreach ($abbreviation as $timezone) {
139
+				if ((int) $timezone['offset'] === $gmt_offset && (bool) $timezone['dst'] === false) {
140
+					try {
141
+						$offset = self::get_timezone_offset(new DateTimeZone($timezone['timezone_id']));
142
+						if ($offset !== $gmt_offset) {
143
+							continue;
144
+						}
145
+						return $timezone['timezone_id'];
146
+					} catch (Exception $e) {
147
+						continue;
148
+					}
149
+				}
150
+			}
151
+		}
152
+		// if $coerce is true, let's see if we can get a timezone string after the offset is adjusted
153
+		if ($coerce === true) {
154
+			$timezone_string = self::get_timezone_string_from_abbreviations_list(
155
+				self::adjust_invalid_gmt_offsets($gmt_offset),
156
+				false
157
+			);
158
+			if ($timezone_string) {
159
+				return $timezone_string;
160
+			}
161
+		}
162
+		throw new EE_Error(
163
+			sprintf(
164
+				esc_html__(
165
+					'The provided GMT offset (%1$s), is invalid, please check with %2$sthis list%3$s for what valid timezones can be used',
166
+					'event_espresso'
167
+				),
168
+				$gmt_offset / HOUR_IN_SECONDS,
169
+				'<a href="http://www.php.net/manual/en/timezones.php">',
170
+				'</a>'
171
+			)
172
+		);
173
+	}
174 174
 
175 175
 
176
-    /**
177
-     * Get Timezone Transitions
178
-     *
179
-     * @param DateTimeZone $date_time_zone
180
-     * @param int|null     $time
181
-     * @param bool         $first_only
182
-     * @return array
183
-     * @throws InvalidArgumentException
184
-     * @throws InvalidDataTypeException
185
-     * @throws InvalidInterfaceException
186
-     */
187
-    public static function get_timezone_transitions(DateTimeZone $date_time_zone, $time = null, $first_only = true)
188
-    {
189
-        return self::getHelperAdapter()->getTimezoneTransitions($date_time_zone, $time, $first_only);
190
-    }
176
+	/**
177
+	 * Get Timezone Transitions
178
+	 *
179
+	 * @param DateTimeZone $date_time_zone
180
+	 * @param int|null     $time
181
+	 * @param bool         $first_only
182
+	 * @return array
183
+	 * @throws InvalidArgumentException
184
+	 * @throws InvalidDataTypeException
185
+	 * @throws InvalidInterfaceException
186
+	 */
187
+	public static function get_timezone_transitions(DateTimeZone $date_time_zone, $time = null, $first_only = true)
188
+	{
189
+		return self::getHelperAdapter()->getTimezoneTransitions($date_time_zone, $time, $first_only);
190
+	}
191 191
 
192 192
 
193
-    /**
194
-     * Get Timezone Offset for given timezone object.
195
-     *
196
-     * @param DateTimeZone $date_time_zone
197
-     * @param null         $time
198
-     * @return mixed
199
-     * @throws InvalidArgumentException
200
-     * @throws InvalidDataTypeException
201
-     * @throws InvalidInterfaceException
202
-     */
203
-    public static function get_timezone_offset(DateTimeZone $date_time_zone, $time = null)
204
-    {
205
-        return self::getHelperAdapter()->getTimezoneOffset($date_time_zone, $time);
206
-    }
193
+	/**
194
+	 * Get Timezone Offset for given timezone object.
195
+	 *
196
+	 * @param DateTimeZone $date_time_zone
197
+	 * @param null         $time
198
+	 * @return mixed
199
+	 * @throws InvalidArgumentException
200
+	 * @throws InvalidDataTypeException
201
+	 * @throws InvalidInterfaceException
202
+	 */
203
+	public static function get_timezone_offset(DateTimeZone $date_time_zone, $time = null)
204
+	{
205
+		return self::getHelperAdapter()->getTimezoneOffset($date_time_zone, $time);
206
+	}
207 207
 
208 208
 
209
-    /**
210
-     * Prints a select input for the given timezone string.
211
-     * @param string $timezone_string
212
-     * @deprecatd 4.9.54.rc   Soft deprecation.  Consider using \EEH_DTT_Helper::wp_timezone_choice instead.
213
-     * @throws InvalidArgumentException
214
-     * @throws InvalidDataTypeException
215
-     * @throws InvalidInterfaceException
216
-     */
217
-    public static function timezone_select_input($timezone_string = '')
218
-    {
219
-        self::getHelperAdapter()->timezoneSelectInput($timezone_string);
220
-    }
209
+	/**
210
+	 * Prints a select input for the given timezone string.
211
+	 * @param string $timezone_string
212
+	 * @deprecatd 4.9.54.rc   Soft deprecation.  Consider using \EEH_DTT_Helper::wp_timezone_choice instead.
213
+	 * @throws InvalidArgumentException
214
+	 * @throws InvalidDataTypeException
215
+	 * @throws InvalidInterfaceException
216
+	 */
217
+	public static function timezone_select_input($timezone_string = '')
218
+	{
219
+		self::getHelperAdapter()->timezoneSelectInput($timezone_string);
220
+	}
221 221
 
222 222
 
223
-    /**
224
-     * This method will take an incoming unix timestamp and add the offset to it for the given timezone_string.
225
-     * If no unix timestamp is given then time() is used.  If no timezone is given then the set timezone string for
226
-     * the site is used.
227
-     * This is used typically when using a Unix timestamp any core WP functions that expect their specially
228
-     * computed timestamp (i.e. date_i18n() )
229
-     *
230
-     * @param int    $unix_timestamp                  if 0, then time() will be used.
231
-     * @param string $timezone_string                 timezone_string. If empty, then the current set timezone for the
232
-     *                                                site will be used.
233
-     * @return int $unix_timestamp with the offset applied for the given timezone.
234
-     * @throws InvalidArgumentException
235
-     * @throws InvalidDataTypeException
236
-     * @throws InvalidInterfaceException
237
-     */
238
-    public static function get_timestamp_with_offset($unix_timestamp = 0, $timezone_string = '')
239
-    {
240
-        return self::getHelperAdapter()->getTimestampWithOffset($unix_timestamp, $timezone_string);
241
-    }
223
+	/**
224
+	 * This method will take an incoming unix timestamp and add the offset to it for the given timezone_string.
225
+	 * If no unix timestamp is given then time() is used.  If no timezone is given then the set timezone string for
226
+	 * the site is used.
227
+	 * This is used typically when using a Unix timestamp any core WP functions that expect their specially
228
+	 * computed timestamp (i.e. date_i18n() )
229
+	 *
230
+	 * @param int    $unix_timestamp                  if 0, then time() will be used.
231
+	 * @param string $timezone_string                 timezone_string. If empty, then the current set timezone for the
232
+	 *                                                site will be used.
233
+	 * @return int $unix_timestamp with the offset applied for the given timezone.
234
+	 * @throws InvalidArgumentException
235
+	 * @throws InvalidDataTypeException
236
+	 * @throws InvalidInterfaceException
237
+	 */
238
+	public static function get_timestamp_with_offset($unix_timestamp = 0, $timezone_string = '')
239
+	{
240
+		return self::getHelperAdapter()->getTimestampWithOffset($unix_timestamp, $timezone_string);
241
+	}
242 242
 
243 243
 
244
-    /**
245
-     *    _set_date_time_field
246
-     *    modifies EE_Base_Class EE_Datetime_Field objects
247
-     *
248
-     * @param  EE_Base_Class $obj                 EE_Base_Class object
249
-     * @param    DateTime    $DateTime            PHP DateTime object
250
-     * @param  string        $datetime_field_name the datetime fieldname to be manipulated
251
-     * @return EE_Base_Class
252
-     * @throws EE_Error
253
-     */
254
-    protected static function _set_date_time_field(EE_Base_Class $obj, DateTime $DateTime, $datetime_field_name)
255
-    {
256
-        // grab current datetime format
257
-        $current_format = $obj->get_format();
258
-        // set new full timestamp format
259
-        $obj->set_date_format(EE_Datetime_Field::mysql_date_format);
260
-        $obj->set_time_format(EE_Datetime_Field::mysql_time_format);
261
-        // set the new date value using a full timestamp format so that no data is lost
262
-        $obj->set($datetime_field_name, $DateTime->format(EE_Datetime_Field::mysql_timestamp_format));
263
-        // reset datetime formats
264
-        $obj->set_date_format($current_format[0]);
265
-        $obj->set_time_format($current_format[1]);
266
-        return $obj;
267
-    }
244
+	/**
245
+	 *    _set_date_time_field
246
+	 *    modifies EE_Base_Class EE_Datetime_Field objects
247
+	 *
248
+	 * @param  EE_Base_Class $obj                 EE_Base_Class object
249
+	 * @param    DateTime    $DateTime            PHP DateTime object
250
+	 * @param  string        $datetime_field_name the datetime fieldname to be manipulated
251
+	 * @return EE_Base_Class
252
+	 * @throws EE_Error
253
+	 */
254
+	protected static function _set_date_time_field(EE_Base_Class $obj, DateTime $DateTime, $datetime_field_name)
255
+	{
256
+		// grab current datetime format
257
+		$current_format = $obj->get_format();
258
+		// set new full timestamp format
259
+		$obj->set_date_format(EE_Datetime_Field::mysql_date_format);
260
+		$obj->set_time_format(EE_Datetime_Field::mysql_time_format);
261
+		// set the new date value using a full timestamp format so that no data is lost
262
+		$obj->set($datetime_field_name, $DateTime->format(EE_Datetime_Field::mysql_timestamp_format));
263
+		// reset datetime formats
264
+		$obj->set_date_format($current_format[0]);
265
+		$obj->set_time_format($current_format[1]);
266
+		return $obj;
267
+	}
268 268
 
269 269
 
270
-    /**
271
-     *    date_time_add
272
-     *    helper for doing simple datetime calculations on a given datetime from EE_Base_Class
273
-     *    and modifying it IN the EE_Base_Class so you don't have to do anything else.
274
-     *
275
-     * @param  EE_Base_Class $obj                 EE_Base_Class object
276
-     * @param  string        $datetime_field_name name of the EE_Datetime_Filed datatype db column to be manipulated
277
-     * @param  string        $period              what you are adding. The options are (years, months, days, hours,
278
-     *                                            minutes, seconds) defaults to years
279
-     * @param  integer       $value               what you want to increment the time by
280
-     * @return EE_Base_Class return the EE_Base_Class object so right away you can do something with it
281
-     *                                            (chaining)
282
-     * @throws EE_Error
283
-     * @throws Exception
284
-     */
285
-    public static function date_time_add(EE_Base_Class $obj, $datetime_field_name, $period = 'years', $value = 1)
286
-    {
287
-        // get the raw UTC date.
288
-        $DateTime = $obj->get_DateTime_object($datetime_field_name);
289
-        $DateTime = EEH_DTT_Helper::calc_date($DateTime, $period, $value);
290
-        return EEH_DTT_Helper::_set_date_time_field($obj, $DateTime, $datetime_field_name);
291
-    }
270
+	/**
271
+	 *    date_time_add
272
+	 *    helper for doing simple datetime calculations on a given datetime from EE_Base_Class
273
+	 *    and modifying it IN the EE_Base_Class so you don't have to do anything else.
274
+	 *
275
+	 * @param  EE_Base_Class $obj                 EE_Base_Class object
276
+	 * @param  string        $datetime_field_name name of the EE_Datetime_Filed datatype db column to be manipulated
277
+	 * @param  string        $period              what you are adding. The options are (years, months, days, hours,
278
+	 *                                            minutes, seconds) defaults to years
279
+	 * @param  integer       $value               what you want to increment the time by
280
+	 * @return EE_Base_Class return the EE_Base_Class object so right away you can do something with it
281
+	 *                                            (chaining)
282
+	 * @throws EE_Error
283
+	 * @throws Exception
284
+	 */
285
+	public static function date_time_add(EE_Base_Class $obj, $datetime_field_name, $period = 'years', $value = 1)
286
+	{
287
+		// get the raw UTC date.
288
+		$DateTime = $obj->get_DateTime_object($datetime_field_name);
289
+		$DateTime = EEH_DTT_Helper::calc_date($DateTime, $period, $value);
290
+		return EEH_DTT_Helper::_set_date_time_field($obj, $DateTime, $datetime_field_name);
291
+	}
292 292
 
293 293
 
294
-    /**
295
-     *    date_time_subtract
296
-     *    same as date_time_add except subtracting value instead of adding.
297
-     *
298
-     * @param EE_Base_Class $obj
299
-     * @param  string       $datetime_field_name name of the EE_Datetime_Filed datatype db column to be manipulated
300
-     * @param string        $period
301
-     * @param int           $value
302
-     * @return EE_Base_Class
303
-     * @throws EE_Error
304
-     * @throws Exception
305
-     */
306
-    public static function date_time_subtract(EE_Base_Class $obj, $datetime_field_name, $period = 'years', $value = 1)
307
-    {
308
-        // get the raw UTC date
309
-        $DateTime = $obj->get_DateTime_object($datetime_field_name);
310
-        $DateTime = EEH_DTT_Helper::calc_date($DateTime, $period, $value, '-');
311
-        return EEH_DTT_Helper::_set_date_time_field($obj, $DateTime, $datetime_field_name);
312
-    }
294
+	/**
295
+	 *    date_time_subtract
296
+	 *    same as date_time_add except subtracting value instead of adding.
297
+	 *
298
+	 * @param EE_Base_Class $obj
299
+	 * @param  string       $datetime_field_name name of the EE_Datetime_Filed datatype db column to be manipulated
300
+	 * @param string        $period
301
+	 * @param int           $value
302
+	 * @return EE_Base_Class
303
+	 * @throws EE_Error
304
+	 * @throws Exception
305
+	 */
306
+	public static function date_time_subtract(EE_Base_Class $obj, $datetime_field_name, $period = 'years', $value = 1)
307
+	{
308
+		// get the raw UTC date
309
+		$DateTime = $obj->get_DateTime_object($datetime_field_name);
310
+		$DateTime = EEH_DTT_Helper::calc_date($DateTime, $period, $value, '-');
311
+		return EEH_DTT_Helper::_set_date_time_field($obj, $DateTime, $datetime_field_name);
312
+	}
313 313
 
314 314
 
315
-    /**
316
-     * Simply takes an incoming DateTime object and does calculations on it based on the incoming parameters
317
-     *
318
-     * @param  DateTime   $DateTime DateTime object
319
-     * @param  string     $period   a value to indicate what interval is being used in the calculation. The options are
320
-     *                              'years', 'months', 'days', 'hours', 'minutes', 'seconds'. Defaults to years.
321
-     * @param  int|string $value    What you want to increment the date by
322
-     * @param  string     $operand  What operand you wish to use for the calculation
323
-     * @return DateTime return whatever type came in.
324
-     * @throws Exception
325
-     * @throws EE_Error
326
-     */
327
-    protected static function _modify_datetime_object(DateTime $DateTime, $period = 'years', $value = 1, $operand = '+')
328
-    {
329
-        if (! $DateTime instanceof DateTime) {
330
-            throw new EE_Error(
331
-                sprintf(
332
-                    esc_html__('Expected a PHP DateTime object, but instead received %1$s', 'event_espresso'),
333
-                    print_r($DateTime, true)
334
-                )
335
-            );
336
-        }
337
-        switch ($period) {
338
-            case 'years':
339
-                $value = 'P' . $value . 'Y';
340
-                break;
341
-            case 'months':
342
-                $value = 'P' . $value . 'M';
343
-                break;
344
-            case 'weeks':
345
-                $value = 'P' . $value . 'W';
346
-                break;
347
-            case 'days':
348
-                $value = 'P' . $value . 'D';
349
-                break;
350
-            case 'hours':
351
-                $value = 'PT' . $value . 'H';
352
-                break;
353
-            case 'minutes':
354
-                $value = 'PT' . $value . 'M';
355
-                break;
356
-            case 'seconds':
357
-                $value = 'PT' . $value . 'S';
358
-                break;
359
-        }
360
-        switch ($operand) {
361
-            case '+':
362
-                $DateTime->add(new DateInterval($value));
363
-                break;
364
-            case '-':
365
-                $DateTime->sub(new DateInterval($value));
366
-                break;
367
-        }
368
-        return $DateTime;
369
-    }
315
+	/**
316
+	 * Simply takes an incoming DateTime object and does calculations on it based on the incoming parameters
317
+	 *
318
+	 * @param  DateTime   $DateTime DateTime object
319
+	 * @param  string     $period   a value to indicate what interval is being used in the calculation. The options are
320
+	 *                              'years', 'months', 'days', 'hours', 'minutes', 'seconds'. Defaults to years.
321
+	 * @param  int|string $value    What you want to increment the date by
322
+	 * @param  string     $operand  What operand you wish to use for the calculation
323
+	 * @return DateTime return whatever type came in.
324
+	 * @throws Exception
325
+	 * @throws EE_Error
326
+	 */
327
+	protected static function _modify_datetime_object(DateTime $DateTime, $period = 'years', $value = 1, $operand = '+')
328
+	{
329
+		if (! $DateTime instanceof DateTime) {
330
+			throw new EE_Error(
331
+				sprintf(
332
+					esc_html__('Expected a PHP DateTime object, but instead received %1$s', 'event_espresso'),
333
+					print_r($DateTime, true)
334
+				)
335
+			);
336
+		}
337
+		switch ($period) {
338
+			case 'years':
339
+				$value = 'P' . $value . 'Y';
340
+				break;
341
+			case 'months':
342
+				$value = 'P' . $value . 'M';
343
+				break;
344
+			case 'weeks':
345
+				$value = 'P' . $value . 'W';
346
+				break;
347
+			case 'days':
348
+				$value = 'P' . $value . 'D';
349
+				break;
350
+			case 'hours':
351
+				$value = 'PT' . $value . 'H';
352
+				break;
353
+			case 'minutes':
354
+				$value = 'PT' . $value . 'M';
355
+				break;
356
+			case 'seconds':
357
+				$value = 'PT' . $value . 'S';
358
+				break;
359
+		}
360
+		switch ($operand) {
361
+			case '+':
362
+				$DateTime->add(new DateInterval($value));
363
+				break;
364
+			case '-':
365
+				$DateTime->sub(new DateInterval($value));
366
+				break;
367
+		}
368
+		return $DateTime;
369
+	}
370 370
 
371 371
 
372
-    /**
373
-     * Simply takes an incoming Unix timestamp and does calculations on it based on the incoming parameters
374
-     *
375
-     * @param  int     $timestamp Unix timestamp
376
-     * @param  string  $period    a value to indicate what interval is being used in the calculation. The options are
377
-     *                            'years', 'months', 'days', 'hours', 'minutes', 'seconds'. Defaults to years.
378
-     * @param  integer $value     What you want to increment the date by
379
-     * @param  string  $operand   What operand you wish to use for the calculation
380
-     * @return int
381
-     * @throws EE_Error
382
-     */
383
-    protected static function _modify_timestamp($timestamp, $period = 'years', $value = 1, $operand = '+')
384
-    {
385
-        if (! preg_match(EE_Datetime_Field::unix_timestamp_regex, $timestamp)) {
386
-            throw new EE_Error(
387
-                sprintf(
388
-                    esc_html__('Expected a Unix timestamp, but instead received %1$s', 'event_espresso'),
389
-                    print_r($timestamp, true)
390
-                )
391
-            );
392
-        }
393
-        switch ($period) {
394
-            case 'years':
395
-                $value = YEAR_IN_SECONDS * $value;
396
-                break;
397
-            case 'months':
398
-                $value = YEAR_IN_SECONDS / 12 * $value;
399
-                break;
400
-            case 'weeks':
401
-                $value = WEEK_IN_SECONDS * $value;
402
-                break;
403
-            case 'days':
404
-                $value = DAY_IN_SECONDS * $value;
405
-                break;
406
-            case 'hours':
407
-                $value = HOUR_IN_SECONDS * $value;
408
-                break;
409
-            case 'minutes':
410
-                $value = MINUTE_IN_SECONDS * $value;
411
-                break;
412
-        }
413
-        switch ($operand) {
414
-            case '+':
415
-                $timestamp += $value;
416
-                break;
417
-            case '-':
418
-                $timestamp -= $value;
419
-                break;
420
-        }
421
-        return $timestamp;
422
-    }
372
+	/**
373
+	 * Simply takes an incoming Unix timestamp and does calculations on it based on the incoming parameters
374
+	 *
375
+	 * @param  int     $timestamp Unix timestamp
376
+	 * @param  string  $period    a value to indicate what interval is being used in the calculation. The options are
377
+	 *                            'years', 'months', 'days', 'hours', 'minutes', 'seconds'. Defaults to years.
378
+	 * @param  integer $value     What you want to increment the date by
379
+	 * @param  string  $operand   What operand you wish to use for the calculation
380
+	 * @return int
381
+	 * @throws EE_Error
382
+	 */
383
+	protected static function _modify_timestamp($timestamp, $period = 'years', $value = 1, $operand = '+')
384
+	{
385
+		if (! preg_match(EE_Datetime_Field::unix_timestamp_regex, $timestamp)) {
386
+			throw new EE_Error(
387
+				sprintf(
388
+					esc_html__('Expected a Unix timestamp, but instead received %1$s', 'event_espresso'),
389
+					print_r($timestamp, true)
390
+				)
391
+			);
392
+		}
393
+		switch ($period) {
394
+			case 'years':
395
+				$value = YEAR_IN_SECONDS * $value;
396
+				break;
397
+			case 'months':
398
+				$value = YEAR_IN_SECONDS / 12 * $value;
399
+				break;
400
+			case 'weeks':
401
+				$value = WEEK_IN_SECONDS * $value;
402
+				break;
403
+			case 'days':
404
+				$value = DAY_IN_SECONDS * $value;
405
+				break;
406
+			case 'hours':
407
+				$value = HOUR_IN_SECONDS * $value;
408
+				break;
409
+			case 'minutes':
410
+				$value = MINUTE_IN_SECONDS * $value;
411
+				break;
412
+		}
413
+		switch ($operand) {
414
+			case '+':
415
+				$timestamp += $value;
416
+				break;
417
+			case '-':
418
+				$timestamp -= $value;
419
+				break;
420
+		}
421
+		return $timestamp;
422
+	}
423 423
 
424 424
 
425
-    /**
426
-     * Simply takes an incoming UTC timestamp or DateTime object and does calculations on it based on the incoming
427
-     * parameters and returns the new timestamp or DateTime.
428
-     *
429
-     * @param  int | DateTime $DateTime_or_timestamp DateTime object or Unix timestamp
430
-     * @param  string         $period                a value to indicate what interval is being used in the
431
-     *                                               calculation. The options are 'years', 'months', 'days', 'hours',
432
-     *                                               'minutes', 'seconds'. Defaults to years.
433
-     * @param  integer        $value                 What you want to increment the date by
434
-     * @param  string         $operand               What operand you wish to use for the calculation
435
-     * @return mixed string|DateTime          return whatever type came in.
436
-     * @throws Exception
437
-     * @throws EE_Error
438
-     */
439
-    public static function calc_date($DateTime_or_timestamp, $period = 'years', $value = 1, $operand = '+')
440
-    {
441
-        if ($DateTime_or_timestamp instanceof DateTime) {
442
-            return EEH_DTT_Helper::_modify_datetime_object(
443
-                $DateTime_or_timestamp,
444
-                $period,
445
-                $value,
446
-                $operand
447
-            );
448
-        }
449
-        if (preg_match(EE_Datetime_Field::unix_timestamp_regex, $DateTime_or_timestamp)) {
450
-            return EEH_DTT_Helper::_modify_timestamp(
451
-                $DateTime_or_timestamp,
452
-                $period,
453
-                $value,
454
-                $operand
455
-            );
456
-        }
457
-        // error
458
-        return $DateTime_or_timestamp;
459
-    }
425
+	/**
426
+	 * Simply takes an incoming UTC timestamp or DateTime object and does calculations on it based on the incoming
427
+	 * parameters and returns the new timestamp or DateTime.
428
+	 *
429
+	 * @param  int | DateTime $DateTime_or_timestamp DateTime object or Unix timestamp
430
+	 * @param  string         $period                a value to indicate what interval is being used in the
431
+	 *                                               calculation. The options are 'years', 'months', 'days', 'hours',
432
+	 *                                               'minutes', 'seconds'. Defaults to years.
433
+	 * @param  integer        $value                 What you want to increment the date by
434
+	 * @param  string         $operand               What operand you wish to use for the calculation
435
+	 * @return mixed string|DateTime          return whatever type came in.
436
+	 * @throws Exception
437
+	 * @throws EE_Error
438
+	 */
439
+	public static function calc_date($DateTime_or_timestamp, $period = 'years', $value = 1, $operand = '+')
440
+	{
441
+		if ($DateTime_or_timestamp instanceof DateTime) {
442
+			return EEH_DTT_Helper::_modify_datetime_object(
443
+				$DateTime_or_timestamp,
444
+				$period,
445
+				$value,
446
+				$operand
447
+			);
448
+		}
449
+		if (preg_match(EE_Datetime_Field::unix_timestamp_regex, $DateTime_or_timestamp)) {
450
+			return EEH_DTT_Helper::_modify_timestamp(
451
+				$DateTime_or_timestamp,
452
+				$period,
453
+				$value,
454
+				$operand
455
+			);
456
+		}
457
+		// error
458
+		return $DateTime_or_timestamp;
459
+	}
460 460
 
461 461
 
462
-    /**
463
-     * The purpose of this helper method is to receive an incoming format string in php date/time format
464
-     * and spit out the js and moment.js equivalent formats.
465
-     * Note, if no format string is given, then it is assumed the user wants what is set for WP.
466
-     * Note, js date and time formats are those used by the jquery-ui datepicker and the jquery-ui date-
467
-     * time picker.
468
-     *
469
-     * @see http://stackoverflow.com/posts/16725290/ for the code inspiration.
470
-     * @param string $date_format_string
471
-     * @param string $time_format_string
472
-     * @return array
473
-     *              array(
474
-     *              'js' => array (
475
-     *              'date' => //date format
476
-     *              'time' => //time format
477
-     *              ),
478
-     *              'moment' => //date and time format.
479
-     *              )
480
-     */
481
-    public static function convert_php_to_js_and_moment_date_formats(
482
-        $date_format_string = null,
483
-        $time_format_string = null
484
-    ) {
485
-        if ($date_format_string === null) {
486
-            $date_format_string = (string) get_option('date_format');
487
-        }
488
-        if ($time_format_string === null) {
489
-            $time_format_string = (string) get_option('time_format');
490
-        }
491
-        $date_format = self::_php_to_js_moment_converter($date_format_string);
492
-        $time_format = self::_php_to_js_moment_converter($time_format_string);
493
-        return array(
494
-            'js'     => array(
495
-                'date' => $date_format['js'],
496
-                'time' => $time_format['js'],
497
-            ),
498
-            'moment' => $date_format['moment'] . ' ' . $time_format['moment'],
499
-            'moment_split' => array(
500
-                'date' => $date_format['moment'],
501
-                'time' => $time_format['moment']
502
-            )
503
-        );
504
-    }
462
+	/**
463
+	 * The purpose of this helper method is to receive an incoming format string in php date/time format
464
+	 * and spit out the js and moment.js equivalent formats.
465
+	 * Note, if no format string is given, then it is assumed the user wants what is set for WP.
466
+	 * Note, js date and time formats are those used by the jquery-ui datepicker and the jquery-ui date-
467
+	 * time picker.
468
+	 *
469
+	 * @see http://stackoverflow.com/posts/16725290/ for the code inspiration.
470
+	 * @param string $date_format_string
471
+	 * @param string $time_format_string
472
+	 * @return array
473
+	 *              array(
474
+	 *              'js' => array (
475
+	 *              'date' => //date format
476
+	 *              'time' => //time format
477
+	 *              ),
478
+	 *              'moment' => //date and time format.
479
+	 *              )
480
+	 */
481
+	public static function convert_php_to_js_and_moment_date_formats(
482
+		$date_format_string = null,
483
+		$time_format_string = null
484
+	) {
485
+		if ($date_format_string === null) {
486
+			$date_format_string = (string) get_option('date_format');
487
+		}
488
+		if ($time_format_string === null) {
489
+			$time_format_string = (string) get_option('time_format');
490
+		}
491
+		$date_format = self::_php_to_js_moment_converter($date_format_string);
492
+		$time_format = self::_php_to_js_moment_converter($time_format_string);
493
+		return array(
494
+			'js'     => array(
495
+				'date' => $date_format['js'],
496
+				'time' => $time_format['js'],
497
+			),
498
+			'moment' => $date_format['moment'] . ' ' . $time_format['moment'],
499
+			'moment_split' => array(
500
+				'date' => $date_format['moment'],
501
+				'time' => $time_format['moment']
502
+			)
503
+		);
504
+	}
505 505
 
506 506
 
507
-    /**
508
-     * This converts incoming format string into js and moment variations.
509
-     *
510
-     * @param string $format_string incoming php format string
511
-     * @return array js and moment formats.
512
-     */
513
-    protected static function _php_to_js_moment_converter($format_string)
514
-    {
515
-        /**
516
-         * This is a map of symbols for formats.
517
-         * The index is the php symbol, the equivalent values are in the array.
518
-         *
519
-         * @var array
520
-         */
521
-        $symbols_map          = array(
522
-            // Day
523
-            // 01
524
-            'd' => array(
525
-                'js'     => 'dd',
526
-                'moment' => 'DD',
527
-            ),
528
-            // Mon
529
-            'D' => array(
530
-                'js'     => 'D',
531
-                'moment' => 'ddd',
532
-            ),
533
-            // 1,2,...31
534
-            'j' => array(
535
-                'js'     => 'd',
536
-                'moment' => 'D',
537
-            ),
538
-            // Monday
539
-            'l' => array(
540
-                'js'     => 'DD',
541
-                'moment' => 'dddd',
542
-            ),
543
-            // ISO numeric representation of the day of the week (1-6)
544
-            'N' => array(
545
-                'js'     => '',
546
-                'moment' => 'E',
547
-            ),
548
-            // st,nd.rd
549
-            'S' => array(
550
-                'js'     => '',
551
-                'moment' => 'o',
552
-            ),
553
-            // numeric representation of day of week (0-6)
554
-            'w' => array(
555
-                'js'     => '',
556
-                'moment' => 'd',
557
-            ),
558
-            // day of year starting from 0 (0-365)
559
-            'z' => array(
560
-                'js'     => 'o',
561
-                'moment' => 'DDD' // note moment does not start with 0 so will need to modify by subtracting 1
562
-            ),
563
-            // Week
564
-            // ISO-8601 week number of year (weeks starting on monday)
565
-            'W' => array(
566
-                'js'     => '',
567
-                'moment' => 'w',
568
-            ),
569
-            // Month
570
-            // January...December
571
-            'F' => array(
572
-                'js'     => 'MM',
573
-                'moment' => 'MMMM',
574
-            ),
575
-            // 01...12
576
-            'm' => array(
577
-                'js'     => 'mm',
578
-                'moment' => 'MM',
579
-            ),
580
-            // Jan...Dec
581
-            'M' => array(
582
-                'js'     => 'M',
583
-                'moment' => 'MMM',
584
-            ),
585
-            // 1-12
586
-            'n' => array(
587
-                'js'     => 'm',
588
-                'moment' => 'M',
589
-            ),
590
-            // number of days in given month
591
-            't' => array(
592
-                'js'     => '',
593
-                'moment' => '',
594
-            ),
595
-            // Year
596
-            // whether leap year or not 1/0
597
-            'L' => array(
598
-                'js'     => '',
599
-                'moment' => '',
600
-            ),
601
-            // ISO-8601 year number
602
-            'o' => array(
603
-                'js'     => '',
604
-                'moment' => 'GGGG',
605
-            ),
606
-            // 1999...2003
607
-            'Y' => array(
608
-                'js'     => 'yy',
609
-                'moment' => 'YYYY',
610
-            ),
611
-            // 99...03
612
-            'y' => array(
613
-                'js'     => 'y',
614
-                'moment' => 'YY',
615
-            ),
616
-            // Time
617
-            // am/pm
618
-            'a' => array(
619
-                'js'     => 'tt',
620
-                'moment' => 'a',
621
-            ),
622
-            // AM/PM
623
-            'A' => array(
624
-                'js'     => 'TT',
625
-                'moment' => 'A',
626
-            ),
627
-            // Swatch Internet Time?!?
628
-            'B' => array(
629
-                'js'     => '',
630
-                'moment' => '',
631
-            ),
632
-            // 1...12
633
-            'g' => array(
634
-                'js'     => 'h',
635
-                'moment' => 'h',
636
-            ),
637
-            // 0...23
638
-            'G' => array(
639
-                'js'     => 'H',
640
-                'moment' => 'H',
641
-            ),
642
-            // 01...12
643
-            'h' => array(
644
-                'js'     => 'hh',
645
-                'moment' => 'hh',
646
-            ),
647
-            // 00...23
648
-            'H' => array(
649
-                'js'     => 'HH',
650
-                'moment' => 'HH',
651
-            ),
652
-            // 00..59
653
-            'i' => array(
654
-                'js'     => 'mm',
655
-                'moment' => 'mm',
656
-            ),
657
-            // seconds... 00...59
658
-            's' => array(
659
-                'js'     => 'ss',
660
-                'moment' => 'ss',
661
-            ),
662
-            // microseconds
663
-            'u' => array(
664
-                'js'     => '',
665
-                'moment' => '',
666
-            ),
667
-        );
668
-        $jquery_ui_format     = '';
669
-        $moment_format        = '';
670
-        $escaping             = false;
671
-        $format_string_length = strlen($format_string);
672
-        for ($i = 0; $i < $format_string_length; $i++) {
673
-            $char = $format_string[ $i ];
674
-            if ($char === '\\') { // PHP date format escaping character
675
-                $i++;
676
-                if ($escaping) {
677
-                    $jquery_ui_format .= $format_string[ $i ];
678
-                    $moment_format    .= $format_string[ $i ];
679
-                } else {
680
-                    $jquery_ui_format .= '\'' . $format_string[ $i ];
681
-                    $moment_format    .= $format_string[ $i ];
682
-                }
683
-                $escaping = true;
684
-            } else {
685
-                if ($escaping) {
686
-                    $jquery_ui_format .= "'";
687
-                    $moment_format    .= "'";
688
-                    $escaping         = false;
689
-                }
690
-                if (isset($symbols_map[ $char ])) {
691
-                    $jquery_ui_format .= $symbols_map[ $char ]['js'];
692
-                    $moment_format    .= $symbols_map[ $char ]['moment'];
693
-                } else {
694
-                    $jquery_ui_format .= $char;
695
-                    $moment_format    .= $char;
696
-                }
697
-            }
698
-        }
699
-        return array('js' => $jquery_ui_format, 'moment' => $moment_format);
700
-    }
507
+	/**
508
+	 * This converts incoming format string into js and moment variations.
509
+	 *
510
+	 * @param string $format_string incoming php format string
511
+	 * @return array js and moment formats.
512
+	 */
513
+	protected static function _php_to_js_moment_converter($format_string)
514
+	{
515
+		/**
516
+		 * This is a map of symbols for formats.
517
+		 * The index is the php symbol, the equivalent values are in the array.
518
+		 *
519
+		 * @var array
520
+		 */
521
+		$symbols_map          = array(
522
+			// Day
523
+			// 01
524
+			'd' => array(
525
+				'js'     => 'dd',
526
+				'moment' => 'DD',
527
+			),
528
+			// Mon
529
+			'D' => array(
530
+				'js'     => 'D',
531
+				'moment' => 'ddd',
532
+			),
533
+			// 1,2,...31
534
+			'j' => array(
535
+				'js'     => 'd',
536
+				'moment' => 'D',
537
+			),
538
+			// Monday
539
+			'l' => array(
540
+				'js'     => 'DD',
541
+				'moment' => 'dddd',
542
+			),
543
+			// ISO numeric representation of the day of the week (1-6)
544
+			'N' => array(
545
+				'js'     => '',
546
+				'moment' => 'E',
547
+			),
548
+			// st,nd.rd
549
+			'S' => array(
550
+				'js'     => '',
551
+				'moment' => 'o',
552
+			),
553
+			// numeric representation of day of week (0-6)
554
+			'w' => array(
555
+				'js'     => '',
556
+				'moment' => 'd',
557
+			),
558
+			// day of year starting from 0 (0-365)
559
+			'z' => array(
560
+				'js'     => 'o',
561
+				'moment' => 'DDD' // note moment does not start with 0 so will need to modify by subtracting 1
562
+			),
563
+			// Week
564
+			// ISO-8601 week number of year (weeks starting on monday)
565
+			'W' => array(
566
+				'js'     => '',
567
+				'moment' => 'w',
568
+			),
569
+			// Month
570
+			// January...December
571
+			'F' => array(
572
+				'js'     => 'MM',
573
+				'moment' => 'MMMM',
574
+			),
575
+			// 01...12
576
+			'm' => array(
577
+				'js'     => 'mm',
578
+				'moment' => 'MM',
579
+			),
580
+			// Jan...Dec
581
+			'M' => array(
582
+				'js'     => 'M',
583
+				'moment' => 'MMM',
584
+			),
585
+			// 1-12
586
+			'n' => array(
587
+				'js'     => 'm',
588
+				'moment' => 'M',
589
+			),
590
+			// number of days in given month
591
+			't' => array(
592
+				'js'     => '',
593
+				'moment' => '',
594
+			),
595
+			// Year
596
+			// whether leap year or not 1/0
597
+			'L' => array(
598
+				'js'     => '',
599
+				'moment' => '',
600
+			),
601
+			// ISO-8601 year number
602
+			'o' => array(
603
+				'js'     => '',
604
+				'moment' => 'GGGG',
605
+			),
606
+			// 1999...2003
607
+			'Y' => array(
608
+				'js'     => 'yy',
609
+				'moment' => 'YYYY',
610
+			),
611
+			// 99...03
612
+			'y' => array(
613
+				'js'     => 'y',
614
+				'moment' => 'YY',
615
+			),
616
+			// Time
617
+			// am/pm
618
+			'a' => array(
619
+				'js'     => 'tt',
620
+				'moment' => 'a',
621
+			),
622
+			// AM/PM
623
+			'A' => array(
624
+				'js'     => 'TT',
625
+				'moment' => 'A',
626
+			),
627
+			// Swatch Internet Time?!?
628
+			'B' => array(
629
+				'js'     => '',
630
+				'moment' => '',
631
+			),
632
+			// 1...12
633
+			'g' => array(
634
+				'js'     => 'h',
635
+				'moment' => 'h',
636
+			),
637
+			// 0...23
638
+			'G' => array(
639
+				'js'     => 'H',
640
+				'moment' => 'H',
641
+			),
642
+			// 01...12
643
+			'h' => array(
644
+				'js'     => 'hh',
645
+				'moment' => 'hh',
646
+			),
647
+			// 00...23
648
+			'H' => array(
649
+				'js'     => 'HH',
650
+				'moment' => 'HH',
651
+			),
652
+			// 00..59
653
+			'i' => array(
654
+				'js'     => 'mm',
655
+				'moment' => 'mm',
656
+			),
657
+			// seconds... 00...59
658
+			's' => array(
659
+				'js'     => 'ss',
660
+				'moment' => 'ss',
661
+			),
662
+			// microseconds
663
+			'u' => array(
664
+				'js'     => '',
665
+				'moment' => '',
666
+			),
667
+		);
668
+		$jquery_ui_format     = '';
669
+		$moment_format        = '';
670
+		$escaping             = false;
671
+		$format_string_length = strlen($format_string);
672
+		for ($i = 0; $i < $format_string_length; $i++) {
673
+			$char = $format_string[ $i ];
674
+			if ($char === '\\') { // PHP date format escaping character
675
+				$i++;
676
+				if ($escaping) {
677
+					$jquery_ui_format .= $format_string[ $i ];
678
+					$moment_format    .= $format_string[ $i ];
679
+				} else {
680
+					$jquery_ui_format .= '\'' . $format_string[ $i ];
681
+					$moment_format    .= $format_string[ $i ];
682
+				}
683
+				$escaping = true;
684
+			} else {
685
+				if ($escaping) {
686
+					$jquery_ui_format .= "'";
687
+					$moment_format    .= "'";
688
+					$escaping         = false;
689
+				}
690
+				if (isset($symbols_map[ $char ])) {
691
+					$jquery_ui_format .= $symbols_map[ $char ]['js'];
692
+					$moment_format    .= $symbols_map[ $char ]['moment'];
693
+				} else {
694
+					$jquery_ui_format .= $char;
695
+					$moment_format    .= $char;
696
+				}
697
+			}
698
+		}
699
+		return array('js' => $jquery_ui_format, 'moment' => $moment_format);
700
+	}
701 701
 
702 702
 
703
-    /**
704
-     * This takes an incoming format string and validates it to ensure it will work fine with PHP.
705
-     *
706
-     * @param string $format_string   Incoming format string for php date().
707
-     * @return mixed bool|array  If all is okay then TRUE is returned.  Otherwise an array of validation
708
-     *                                errors is returned.  So for client code calling, check for is_array() to
709
-     *                                indicate failed validations.
710
-     */
711
-    public static function validate_format_string($format_string)
712
-    {
713
-        $error_msg = array();
714
-        // time format checks
715
-        switch (true) {
716
-            case strpos($format_string, 'h') !== false:
717
-            case strpos($format_string, 'g') !== false:
718
-                /**
719
-                 * if the time string has a lowercase 'h' which == 12 hour time format and there
720
-                 * is not any ante meridiem format ('a' or 'A').  Then throw an error because its
721
-                 * too ambiguous and PHP won't be able to figure out whether 1 = 1pm or 1am.
722
-                 */
723
-                if (stripos($format_string, 'A') === false) {
724
-                    $error_msg[] = esc_html__(
725
-                        'There is a  time format for 12 hour time but no  "a" or "A" to indicate am/pm.  Without this distinction, PHP is unable to determine if a "1" for the hour value equals "1pm" or "1am".',
726
-                        'event_espresso'
727
-                    );
728
-                }
729
-                break;
730
-        }
731
-        return empty($error_msg) ? true : $error_msg;
732
-    }
703
+	/**
704
+	 * This takes an incoming format string and validates it to ensure it will work fine with PHP.
705
+	 *
706
+	 * @param string $format_string   Incoming format string for php date().
707
+	 * @return mixed bool|array  If all is okay then TRUE is returned.  Otherwise an array of validation
708
+	 *                                errors is returned.  So for client code calling, check for is_array() to
709
+	 *                                indicate failed validations.
710
+	 */
711
+	public static function validate_format_string($format_string)
712
+	{
713
+		$error_msg = array();
714
+		// time format checks
715
+		switch (true) {
716
+			case strpos($format_string, 'h') !== false:
717
+			case strpos($format_string, 'g') !== false:
718
+				/**
719
+				 * if the time string has a lowercase 'h' which == 12 hour time format and there
720
+				 * is not any ante meridiem format ('a' or 'A').  Then throw an error because its
721
+				 * too ambiguous and PHP won't be able to figure out whether 1 = 1pm or 1am.
722
+				 */
723
+				if (stripos($format_string, 'A') === false) {
724
+					$error_msg[] = esc_html__(
725
+						'There is a  time format for 12 hour time but no  "a" or "A" to indicate am/pm.  Without this distinction, PHP is unable to determine if a "1" for the hour value equals "1pm" or "1am".',
726
+						'event_espresso'
727
+					);
728
+				}
729
+				break;
730
+		}
731
+		return empty($error_msg) ? true : $error_msg;
732
+	}
733 733
 
734 734
 
735
-    /**
736
-     *     If the the first date starts at midnight on one day, and the next date ends at midnight on the
737
-     *     very next day then this method will return true.
738
-     *    If $date_1 = 2015-12-15 00:00:00 and $date_2 = 2015-12-16 00:00:00 then this function will return true.
739
-     *    If $date_1 = 2015-12-15 03:00:00 and $date_2 = 2015-12_16 03:00:00 then this function will return false.
740
-     *    If $date_1 = 2015-12-15 00:00:00 and $date_2 = 2015-12-15 00:00:00 then this function will return true.
741
-     *
742
-     * @param mixed $date_1
743
-     * @param mixed $date_2
744
-     * @return bool
745
-     */
746
-    public static function dates_represent_one_24_hour_date($date_1, $date_2)
747
-    {
735
+	/**
736
+	 *     If the the first date starts at midnight on one day, and the next date ends at midnight on the
737
+	 *     very next day then this method will return true.
738
+	 *    If $date_1 = 2015-12-15 00:00:00 and $date_2 = 2015-12-16 00:00:00 then this function will return true.
739
+	 *    If $date_1 = 2015-12-15 03:00:00 and $date_2 = 2015-12_16 03:00:00 then this function will return false.
740
+	 *    If $date_1 = 2015-12-15 00:00:00 and $date_2 = 2015-12-15 00:00:00 then this function will return true.
741
+	 *
742
+	 * @param mixed $date_1
743
+	 * @param mixed $date_2
744
+	 * @return bool
745
+	 */
746
+	public static function dates_represent_one_24_hour_date($date_1, $date_2)
747
+	{
748 748
 
749
-        if ((! $date_1 instanceof DateTime || ! $date_2 instanceof DateTime)
750
-            || ($date_1->format(EE_Datetime_Field::mysql_time_format) !== '00:00:00'
751
-                || $date_2->format(
752
-                    EE_Datetime_Field::mysql_time_format
753
-                ) !== '00:00:00')
754
-        ) {
755
-            return false;
756
-        }
757
-        return $date_2->format('U') - $date_1->format('U') === 86400;
758
-    }
749
+		if ((! $date_1 instanceof DateTime || ! $date_2 instanceof DateTime)
750
+			|| ($date_1->format(EE_Datetime_Field::mysql_time_format) !== '00:00:00'
751
+				|| $date_2->format(
752
+					EE_Datetime_Field::mysql_time_format
753
+				) !== '00:00:00')
754
+		) {
755
+			return false;
756
+		}
757
+		return $date_2->format('U') - $date_1->format('U') === 86400;
758
+	}
759 759
 
760 760
 
761
-    /**
762
-     * This returns the appropriate query interval string that can be used in sql queries involving mysql Date
763
-     * Functions.
764
-     *
765
-     * @param string $timezone_string    A timezone string in a valid format to instantiate a DateTimeZone object.
766
-     * @param string $field_for_interval The Database field that is the interval is applied to in the query.
767
-     * @return string
768
-     */
769
-    public static function get_sql_query_interval_for_offset($timezone_string, $field_for_interval)
770
-    {
771
-        try {
772
-            /** need to account for timezone offset on the selects */
773
-            $DateTimeZone = new DateTimeZone($timezone_string);
774
-        } catch (Exception $e) {
775
-            $DateTimeZone = null;
776
-        }
777
-        /**
778
-         * Note get_option( 'gmt_offset') returns a value in hours, whereas DateTimeZone::getOffset returns values in seconds.
779
-         * Hence we do the calc for DateTimeZone::getOffset.
780
-         */
781
-        $offset         = $DateTimeZone instanceof DateTimeZone
782
-            ? $DateTimeZone->getOffset(new DateTime('now')) / HOUR_IN_SECONDS
783
-            : (float) get_option('gmt_offset');
784
-        $query_interval = $offset < 0
785
-            ? 'DATE_SUB(' . $field_for_interval . ', INTERVAL ' . $offset * -1 . ' HOUR)'
786
-            : 'DATE_ADD(' . $field_for_interval . ', INTERVAL ' . $offset . ' HOUR)';
787
-        return $query_interval;
788
-    }
761
+	/**
762
+	 * This returns the appropriate query interval string that can be used in sql queries involving mysql Date
763
+	 * Functions.
764
+	 *
765
+	 * @param string $timezone_string    A timezone string in a valid format to instantiate a DateTimeZone object.
766
+	 * @param string $field_for_interval The Database field that is the interval is applied to in the query.
767
+	 * @return string
768
+	 */
769
+	public static function get_sql_query_interval_for_offset($timezone_string, $field_for_interval)
770
+	{
771
+		try {
772
+			/** need to account for timezone offset on the selects */
773
+			$DateTimeZone = new DateTimeZone($timezone_string);
774
+		} catch (Exception $e) {
775
+			$DateTimeZone = null;
776
+		}
777
+		/**
778
+		 * Note get_option( 'gmt_offset') returns a value in hours, whereas DateTimeZone::getOffset returns values in seconds.
779
+		 * Hence we do the calc for DateTimeZone::getOffset.
780
+		 */
781
+		$offset         = $DateTimeZone instanceof DateTimeZone
782
+			? $DateTimeZone->getOffset(new DateTime('now')) / HOUR_IN_SECONDS
783
+			: (float) get_option('gmt_offset');
784
+		$query_interval = $offset < 0
785
+			? 'DATE_SUB(' . $field_for_interval . ', INTERVAL ' . $offset * -1 . ' HOUR)'
786
+			: 'DATE_ADD(' . $field_for_interval . ', INTERVAL ' . $offset . ' HOUR)';
787
+		return $query_interval;
788
+	}
789 789
 
790 790
 
791
-    /**
792
-     * Retrieves the site's default timezone and returns it formatted so it's ready for display
793
-     * to users. If you want to customize how its displayed feel free to fetch the 'timezone_string'
794
-     * and 'gmt_offset' WordPress options directly; or use the filter
795
-     * FHEE__EEH_DTT_Helper__get_timezone_string_for_display
796
-     * (although note that we remove any HTML that may be added)
797
-     *
798
-     * @return string
799
-     */
800
-    public static function get_timezone_string_for_display()
801
-    {
802
-        $pretty_timezone = apply_filters('FHEE__EEH_DTT_Helper__get_timezone_string_for_display', '');
803
-        if (! empty($pretty_timezone)) {
804
-            return esc_html($pretty_timezone);
805
-        }
806
-        $timezone_string = get_option('timezone_string');
807
-        if ($timezone_string) {
808
-            static $mo_loaded = false;
809
-            // Load translations for continents and cities just like wp_timezone_choice does
810
-            if (! $mo_loaded) {
811
-                $locale = get_locale();
812
-                $mofile = WP_LANG_DIR . '/continents-cities-' . $locale . '.mo';
813
-                load_textdomain('continents-cities', $mofile);
814
-                $mo_loaded = true;
815
-            }
816
-            // well that was easy.
817
-            $parts = explode('/', $timezone_string);
818
-            // remove the continent
819
-            unset($parts[0]);
820
-            $t_parts = array();
821
-            // phpcs:disable WordPress.WP.I18n.NonSingularStringLiteralText
822
-            // phpcs:disable WordPress.WP.I18n.TextDomainMismatch
823
-            // disabled because this code is copied from WordPress and is a WordPress domain
824
-            foreach ($parts as $part) {
825
-                $t_parts[] = translate(str_replace('_', ' ', $part), 'continents-cities');
826
-            }
827
-            return implode(' - ', $t_parts);
828
-            // phpcs:enable
829
-        }
830
-        // they haven't set the timezone string, so let's return a string like "UTC+1"
831
-        $gmt_offset = get_option('gmt_offset');
832
-        $prefix     = (int) $gmt_offset >= 0 ? '+' : '';
833
-        $parts      = explode('.', (string) $gmt_offset);
834
-        if (count($parts) === 1) {
835
-            $parts[1] = '00';
836
-        } else {
837
-            // convert the part after the decimal, eg "5" (from x.5) or "25" (from x.25)
838
-            // to minutes, eg 30 or 15, respectively
839
-            $hour_fraction = (float) ('0.' . $parts[1]);
840
-            $parts[1]      = (string) $hour_fraction * 60;
841
-        }
842
-        return sprintf(__('UTC%1$s', 'event_espresso'), $prefix . implode(':', $parts));
843
-    }
791
+	/**
792
+	 * Retrieves the site's default timezone and returns it formatted so it's ready for display
793
+	 * to users. If you want to customize how its displayed feel free to fetch the 'timezone_string'
794
+	 * and 'gmt_offset' WordPress options directly; or use the filter
795
+	 * FHEE__EEH_DTT_Helper__get_timezone_string_for_display
796
+	 * (although note that we remove any HTML that may be added)
797
+	 *
798
+	 * @return string
799
+	 */
800
+	public static function get_timezone_string_for_display()
801
+	{
802
+		$pretty_timezone = apply_filters('FHEE__EEH_DTT_Helper__get_timezone_string_for_display', '');
803
+		if (! empty($pretty_timezone)) {
804
+			return esc_html($pretty_timezone);
805
+		}
806
+		$timezone_string = get_option('timezone_string');
807
+		if ($timezone_string) {
808
+			static $mo_loaded = false;
809
+			// Load translations for continents and cities just like wp_timezone_choice does
810
+			if (! $mo_loaded) {
811
+				$locale = get_locale();
812
+				$mofile = WP_LANG_DIR . '/continents-cities-' . $locale . '.mo';
813
+				load_textdomain('continents-cities', $mofile);
814
+				$mo_loaded = true;
815
+			}
816
+			// well that was easy.
817
+			$parts = explode('/', $timezone_string);
818
+			// remove the continent
819
+			unset($parts[0]);
820
+			$t_parts = array();
821
+			// phpcs:disable WordPress.WP.I18n.NonSingularStringLiteralText
822
+			// phpcs:disable WordPress.WP.I18n.TextDomainMismatch
823
+			// disabled because this code is copied from WordPress and is a WordPress domain
824
+			foreach ($parts as $part) {
825
+				$t_parts[] = translate(str_replace('_', ' ', $part), 'continents-cities');
826
+			}
827
+			return implode(' - ', $t_parts);
828
+			// phpcs:enable
829
+		}
830
+		// they haven't set the timezone string, so let's return a string like "UTC+1"
831
+		$gmt_offset = get_option('gmt_offset');
832
+		$prefix     = (int) $gmt_offset >= 0 ? '+' : '';
833
+		$parts      = explode('.', (string) $gmt_offset);
834
+		if (count($parts) === 1) {
835
+			$parts[1] = '00';
836
+		} else {
837
+			// convert the part after the decimal, eg "5" (from x.5) or "25" (from x.25)
838
+			// to minutes, eg 30 or 15, respectively
839
+			$hour_fraction = (float) ('0.' . $parts[1]);
840
+			$parts[1]      = (string) $hour_fraction * 60;
841
+		}
842
+		return sprintf(__('UTC%1$s', 'event_espresso'), $prefix . implode(':', $parts));
843
+	}
844 844
 
845 845
 
846 846
 
847
-    /**
848
-     * So PHP does this awesome thing where if you are trying to get a timestamp
849
-     * for a month using a string like "February" or "February 2017",
850
-     * and you don't specify a day as part of your string,
851
-     * then PHP will use whatever the current day of the month is.
852
-     * IF the current day of the month happens to be the 30th or 31st,
853
-     * then PHP gets really confused by a date like February 30,
854
-     * so instead of saying
855
-     *      "Hey February only has 28 days (this year)...
856
-     *      ...you must have meant the last day of the month!"
857
-     * PHP does the next most logical thing, and bumps the date up to March 2nd,
858
-     * because someone requesting February 30th obviously meant March 1st!
859
-     * The way around this is to always set the day to the first,
860
-     * so that the month will stay on the month you wanted.
861
-     * this method will add that "1" into your date regardless of the format.
862
-     *
863
-     * @param string $month
864
-     * @return string
865
-     */
866
-    public static function first_of_month_timestamp($month = '')
867
-    {
868
-        $month = (string) $month;
869
-        $year  = '';
870
-        // check if the incoming string has a year in it or not
871
-        if (preg_match('/\b\d{4}\b/', $month, $matches)) {
872
-            $year = $matches[0];
873
-            // ten remove that from the month string as well as any spaces
874
-            $month = trim(str_replace($year, '', $month));
875
-            // add a space before the year
876
-            $year = " {$year}";
877
-        }
878
-        // return timestamp for something like "February 1 2017"
879
-        return strtotime("{$month} 1{$year}");
880
-    }
847
+	/**
848
+	 * So PHP does this awesome thing where if you are trying to get a timestamp
849
+	 * for a month using a string like "February" or "February 2017",
850
+	 * and you don't specify a day as part of your string,
851
+	 * then PHP will use whatever the current day of the month is.
852
+	 * IF the current day of the month happens to be the 30th or 31st,
853
+	 * then PHP gets really confused by a date like February 30,
854
+	 * so instead of saying
855
+	 *      "Hey February only has 28 days (this year)...
856
+	 *      ...you must have meant the last day of the month!"
857
+	 * PHP does the next most logical thing, and bumps the date up to March 2nd,
858
+	 * because someone requesting February 30th obviously meant March 1st!
859
+	 * The way around this is to always set the day to the first,
860
+	 * so that the month will stay on the month you wanted.
861
+	 * this method will add that "1" into your date regardless of the format.
862
+	 *
863
+	 * @param string $month
864
+	 * @return string
865
+	 */
866
+	public static function first_of_month_timestamp($month = '')
867
+	{
868
+		$month = (string) $month;
869
+		$year  = '';
870
+		// check if the incoming string has a year in it or not
871
+		if (preg_match('/\b\d{4}\b/', $month, $matches)) {
872
+			$year = $matches[0];
873
+			// ten remove that from the month string as well as any spaces
874
+			$month = trim(str_replace($year, '', $month));
875
+			// add a space before the year
876
+			$year = " {$year}";
877
+		}
878
+		// return timestamp for something like "February 1 2017"
879
+		return strtotime("{$month} 1{$year}");
880
+	}
881 881
 
882 882
 
883
-    /**
884
-     * This simply returns the timestamp for tomorrow (midnight next day) in this sites timezone.  So it may be midnight
885
-     * for this sites timezone, but the timestamp could be some other time GMT.
886
-     */
887
-    public static function tomorrow()
888
-    {
889
-        // The multiplication of -1 ensures that we switch positive offsets to negative and negative offsets to positive
890
-        // before adding to the timestamp.  Why? Because we want tomorrow to be for midnight the next day in THIS timezone
891
-        // not an offset from midnight in UTC.  So if we're starting with UTC 00:00:00, then we want to make sure the
892
-        // final timestamp is equivalent to midnight in this timezone as represented in GMT.
893
-        return strtotime('tomorrow') + (self::get_site_timezone_gmt_offset() * -1);
894
-    }
883
+	/**
884
+	 * This simply returns the timestamp for tomorrow (midnight next day) in this sites timezone.  So it may be midnight
885
+	 * for this sites timezone, but the timestamp could be some other time GMT.
886
+	 */
887
+	public static function tomorrow()
888
+	{
889
+		// The multiplication of -1 ensures that we switch positive offsets to negative and negative offsets to positive
890
+		// before adding to the timestamp.  Why? Because we want tomorrow to be for midnight the next day in THIS timezone
891
+		// not an offset from midnight in UTC.  So if we're starting with UTC 00:00:00, then we want to make sure the
892
+		// final timestamp is equivalent to midnight in this timezone as represented in GMT.
893
+		return strtotime('tomorrow') + (self::get_site_timezone_gmt_offset() * -1);
894
+	}
895 895
 
896 896
 
897
-    /**
898
-     * **
899
-     * Gives a nicely-formatted list of timezone strings.
900
-     * Copied from the core wp function by the same name so we could customize to remove UTC offsets.
901
-     *
902
-     * @since     4.9.40.rc.008
903
-     * @staticvar bool $mo_loaded
904
-     * @staticvar string $locale_loaded
905
-     * @param string $selected_zone Selected timezone.
906
-     * @param string $locale        Optional. Locale to load the timezones in. Default current site locale.
907
-     * @return string
908
-     */
909
-    public static function wp_timezone_choice($selected_zone, $locale = null)
910
-    {
911
-        static $mo_loaded = false, $locale_loaded = null;
912
-        $continents = array(
913
-            'Africa',
914
-            'America',
915
-            'Antarctica',
916
-            'Arctic',
917
-            'Asia',
918
-            'Atlantic',
919
-            'Australia',
920
-            'Europe',
921
-            'Indian',
922
-            'Pacific',
923
-        );
924
-        // Load translations for continents and cities.
925
-        if (! $mo_loaded || $locale !== $locale_loaded) {
926
-            $locale_loaded = $locale ? $locale : get_locale();
927
-            $mofile        = WP_LANG_DIR . '/continents-cities-' . $locale_loaded . '.mo';
928
-            unload_textdomain('continents-cities');
929
-            load_textdomain('continents-cities', $mofile);
930
-            $mo_loaded = true;
931
-        }
932
-        $zone_data = array();
933
-        foreach (timezone_identifiers_list() as $zone) {
934
-            $zone = explode('/', $zone);
935
-            if (! in_array($zone[0], $continents, true)) {
936
-                continue;
937
-            }
938
-            // This determines what gets set and translated - we don't translate Etc/* strings here, they are done later
939
-            $exists      = array(
940
-                0 => isset($zone[0]) && $zone[0],
941
-                1 => isset($zone[1]) && $zone[1],
942
-                2 => isset($zone[2]) && $zone[2],
943
-            );
944
-            $exists[3]   = $exists[0] && $zone[0] !== 'Etc';
945
-            $exists[4]   = $exists[1] && $exists[3];
946
-            $exists[5]   = $exists[2] && $exists[3];
947
-            // phpcs:disable WordPress.WP.I18n.NonSingularStringLiteralText
948
-            // phpcs:disable WordPress.WP.I18n.TextDomainMismatch
949
-            // disabled because this code is copied from WordPress and is a WordPress domain
950
-            $zone_data[] = array(
951
-                'continent'   => $exists[0] ? $zone[0] : '',
952
-                'city'        => $exists[1] ? $zone[1] : '',
953
-                'subcity'     => $exists[2] ? $zone[2] : '',
954
-                't_continent' => $exists[3]
955
-                    ? translate(str_replace('_', ' ', $zone[0]), 'continents-cities')
956
-                    : '',
957
-                't_city'      => $exists[4]
958
-                    ? translate(str_replace('_', ' ', $zone[1]), 'continents-cities')
959
-                    : '',
960
-                't_subcity'   => $exists[5]
961
-                    ? translate(str_replace('_', ' ', $zone[2]), 'continents-cities')
962
-                    : '',
963
-            );
964
-            // phpcs:enable
965
-        }
966
-        usort($zone_data, '_wp_timezone_choice_usort_callback');
967
-        $structure = array();
968
-        if (empty($selected_zone)) {
969
-            $structure[] = '<option selected="selected" value="">' . __('Select a city', 'event_espresso') . '</option>';
970
-        }
971
-        foreach ($zone_data as $key => $zone) {
972
-            // Build value in an array to join later
973
-            $value = array($zone['continent']);
974
-            if (empty($zone['city'])) {
975
-                // It's at the continent level (generally won't happen)
976
-                $display = $zone['t_continent'];
977
-            } else {
978
-                // It's inside a continent group
979
-                // Continent optgroup
980
-                if (! isset($zone_data[ $key - 1 ]) || $zone_data[ $key - 1 ]['continent'] !== $zone['continent']) {
981
-                    $label       = $zone['t_continent'];
982
-                    $structure[] = '<optgroup label="' . esc_attr($label) . '">';
983
-                }
984
-                // Add the city to the value
985
-                $value[] = $zone['city'];
986
-                $display = $zone['t_city'];
987
-                if (! empty($zone['subcity'])) {
988
-                    // Add the subcity to the value
989
-                    $value[] = $zone['subcity'];
990
-                    $display .= ' - ' . $zone['t_subcity'];
991
-                }
992
-            }
993
-            // Build the value
994
-            $value       = implode('/', $value);
995
-            $selected    = $value === $selected_zone ? ' selected="selected"' : '';
996
-            $structure[] = '<option value="' . esc_attr($value) . '"' . $selected . '>'
997
-                           . esc_html($display)
998
-                           . '</option>';
999
-            // Close continent optgroup
1000
-            if (! empty($zone['city'])
1001
-                && (
1002
-                    ! isset($zone_data[ $key + 1 ])
1003
-                    || (isset($zone_data[ $key + 1 ]) && $zone_data[ $key + 1 ]['continent'] !== $zone['continent'])
1004
-                )
1005
-            ) {
1006
-                $structure[] = '</optgroup>';
1007
-            }
1008
-        }
1009
-        return implode("\n", $structure);
1010
-    }
897
+	/**
898
+	 * **
899
+	 * Gives a nicely-formatted list of timezone strings.
900
+	 * Copied from the core wp function by the same name so we could customize to remove UTC offsets.
901
+	 *
902
+	 * @since     4.9.40.rc.008
903
+	 * @staticvar bool $mo_loaded
904
+	 * @staticvar string $locale_loaded
905
+	 * @param string $selected_zone Selected timezone.
906
+	 * @param string $locale        Optional. Locale to load the timezones in. Default current site locale.
907
+	 * @return string
908
+	 */
909
+	public static function wp_timezone_choice($selected_zone, $locale = null)
910
+	{
911
+		static $mo_loaded = false, $locale_loaded = null;
912
+		$continents = array(
913
+			'Africa',
914
+			'America',
915
+			'Antarctica',
916
+			'Arctic',
917
+			'Asia',
918
+			'Atlantic',
919
+			'Australia',
920
+			'Europe',
921
+			'Indian',
922
+			'Pacific',
923
+		);
924
+		// Load translations for continents and cities.
925
+		if (! $mo_loaded || $locale !== $locale_loaded) {
926
+			$locale_loaded = $locale ? $locale : get_locale();
927
+			$mofile        = WP_LANG_DIR . '/continents-cities-' . $locale_loaded . '.mo';
928
+			unload_textdomain('continents-cities');
929
+			load_textdomain('continents-cities', $mofile);
930
+			$mo_loaded = true;
931
+		}
932
+		$zone_data = array();
933
+		foreach (timezone_identifiers_list() as $zone) {
934
+			$zone = explode('/', $zone);
935
+			if (! in_array($zone[0], $continents, true)) {
936
+				continue;
937
+			}
938
+			// This determines what gets set and translated - we don't translate Etc/* strings here, they are done later
939
+			$exists      = array(
940
+				0 => isset($zone[0]) && $zone[0],
941
+				1 => isset($zone[1]) && $zone[1],
942
+				2 => isset($zone[2]) && $zone[2],
943
+			);
944
+			$exists[3]   = $exists[0] && $zone[0] !== 'Etc';
945
+			$exists[4]   = $exists[1] && $exists[3];
946
+			$exists[5]   = $exists[2] && $exists[3];
947
+			// phpcs:disable WordPress.WP.I18n.NonSingularStringLiteralText
948
+			// phpcs:disable WordPress.WP.I18n.TextDomainMismatch
949
+			// disabled because this code is copied from WordPress and is a WordPress domain
950
+			$zone_data[] = array(
951
+				'continent'   => $exists[0] ? $zone[0] : '',
952
+				'city'        => $exists[1] ? $zone[1] : '',
953
+				'subcity'     => $exists[2] ? $zone[2] : '',
954
+				't_continent' => $exists[3]
955
+					? translate(str_replace('_', ' ', $zone[0]), 'continents-cities')
956
+					: '',
957
+				't_city'      => $exists[4]
958
+					? translate(str_replace('_', ' ', $zone[1]), 'continents-cities')
959
+					: '',
960
+				't_subcity'   => $exists[5]
961
+					? translate(str_replace('_', ' ', $zone[2]), 'continents-cities')
962
+					: '',
963
+			);
964
+			// phpcs:enable
965
+		}
966
+		usort($zone_data, '_wp_timezone_choice_usort_callback');
967
+		$structure = array();
968
+		if (empty($selected_zone)) {
969
+			$structure[] = '<option selected="selected" value="">' . __('Select a city', 'event_espresso') . '</option>';
970
+		}
971
+		foreach ($zone_data as $key => $zone) {
972
+			// Build value in an array to join later
973
+			$value = array($zone['continent']);
974
+			if (empty($zone['city'])) {
975
+				// It's at the continent level (generally won't happen)
976
+				$display = $zone['t_continent'];
977
+			} else {
978
+				// It's inside a continent group
979
+				// Continent optgroup
980
+				if (! isset($zone_data[ $key - 1 ]) || $zone_data[ $key - 1 ]['continent'] !== $zone['continent']) {
981
+					$label       = $zone['t_continent'];
982
+					$structure[] = '<optgroup label="' . esc_attr($label) . '">';
983
+				}
984
+				// Add the city to the value
985
+				$value[] = $zone['city'];
986
+				$display = $zone['t_city'];
987
+				if (! empty($zone['subcity'])) {
988
+					// Add the subcity to the value
989
+					$value[] = $zone['subcity'];
990
+					$display .= ' - ' . $zone['t_subcity'];
991
+				}
992
+			}
993
+			// Build the value
994
+			$value       = implode('/', $value);
995
+			$selected    = $value === $selected_zone ? ' selected="selected"' : '';
996
+			$structure[] = '<option value="' . esc_attr($value) . '"' . $selected . '>'
997
+						   . esc_html($display)
998
+						   . '</option>';
999
+			// Close continent optgroup
1000
+			if (! empty($zone['city'])
1001
+				&& (
1002
+					! isset($zone_data[ $key + 1 ])
1003
+					|| (isset($zone_data[ $key + 1 ]) && $zone_data[ $key + 1 ]['continent'] !== $zone['continent'])
1004
+				)
1005
+			) {
1006
+				$structure[] = '</optgroup>';
1007
+			}
1008
+		}
1009
+		return implode("\n", $structure);
1010
+	}
1011 1011
 
1012 1012
 
1013
-    /**
1014
-     * Shim for the WP function `get_user_locale` that was added in WordPress 4.7.0
1015
-     *
1016
-     * @param int|WP_User $user_id
1017
-     * @return string
1018
-     */
1019
-    public static function get_user_locale($user_id = 0)
1020
-    {
1021
-        if (function_exists('get_user_locale')) {
1022
-            return get_user_locale($user_id);
1023
-        }
1024
-        return get_locale();
1025
-    }
1013
+	/**
1014
+	 * Shim for the WP function `get_user_locale` that was added in WordPress 4.7.0
1015
+	 *
1016
+	 * @param int|WP_User $user_id
1017
+	 * @return string
1018
+	 */
1019
+	public static function get_user_locale($user_id = 0)
1020
+	{
1021
+		if (function_exists('get_user_locale')) {
1022
+			return get_user_locale($user_id);
1023
+		}
1024
+		return get_locale();
1025
+	}
1026 1026
 
1027 1027
 
1028
-    /**
1029
-     * Return the appropriate helper adapter for DTT related things.
1030
-     *
1031
-     * @return HelperInterface
1032
-     * @throws InvalidArgumentException
1033
-     * @throws InvalidDataTypeException
1034
-     * @throws InvalidInterfaceException
1035
-     */
1036
-    private static function getHelperAdapter()
1037
-    {
1038
-        $dtt_helper_fqcn = PHP_VERSION_ID < 50600
1039
-            ? 'EventEspresso\core\services\helpers\datetime\PhpCompatLessFiveSixHelper'
1040
-            : 'EventEspresso\core\services\helpers\datetime\PhpCompatGreaterFiveSixHelper';
1041
-        return LoaderFactory::getLoader()->getShared($dtt_helper_fqcn);
1042
-    }
1028
+	/**
1029
+	 * Return the appropriate helper adapter for DTT related things.
1030
+	 *
1031
+	 * @return HelperInterface
1032
+	 * @throws InvalidArgumentException
1033
+	 * @throws InvalidDataTypeException
1034
+	 * @throws InvalidInterfaceException
1035
+	 */
1036
+	private static function getHelperAdapter()
1037
+	{
1038
+		$dtt_helper_fqcn = PHP_VERSION_ID < 50600
1039
+			? 'EventEspresso\core\services\helpers\datetime\PhpCompatLessFiveSixHelper'
1040
+			: 'EventEspresso\core\services\helpers\datetime\PhpCompatGreaterFiveSixHelper';
1041
+		return LoaderFactory::getLoader()->getShared($dtt_helper_fqcn);
1042
+	}
1043 1043
 
1044 1044
 
1045
-    /**
1046
-     * Helper function for setting the timezone on a DateTime object.
1047
-     * This is implemented to standardize a workaround for a PHP bug outlined in
1048
-     * https://events.codebasehq.com/projects/event-espresso/tickets/11407 and
1049
-     * https://events.codebasehq.com/projects/event-espresso/tickets/11233
1050
-     *
1051
-     * @param DateTime     $datetime
1052
-     * @param DateTimeZone $timezone
1053
-     */
1054
-    public static function setTimezone(DateTime $datetime, DateTimeZone $timezone)
1055
-    {
1056
-        $datetime->setTimezone($timezone);
1057
-        $datetime->getTimestamp();
1058
-    }
1045
+	/**
1046
+	 * Helper function for setting the timezone on a DateTime object.
1047
+	 * This is implemented to standardize a workaround for a PHP bug outlined in
1048
+	 * https://events.codebasehq.com/projects/event-espresso/tickets/11407 and
1049
+	 * https://events.codebasehq.com/projects/event-espresso/tickets/11233
1050
+	 *
1051
+	 * @param DateTime     $datetime
1052
+	 * @param DateTimeZone $timezone
1053
+	 */
1054
+	public static function setTimezone(DateTime $datetime, DateTimeZone $timezone)
1055
+	{
1056
+		$datetime->setTimezone($timezone);
1057
+		$datetime->getTimestamp();
1058
+	}
1059 1059
 }
Please login to merge, or discard this patch.
Spacing   +42 added lines, -42 removed lines patch added patch discarded remove patch
@@ -131,7 +131,7 @@  discard block
 block discarded – undo
131 131
      */
132 132
     public static function get_timezone_string_from_abbreviations_list($gmt_offset = 0, $coerce = true)
133 133
     {
134
-        $gmt_offset =  (int) $gmt_offset;
134
+        $gmt_offset = (int) $gmt_offset;
135 135
         /** @var array[] $abbreviations */
136 136
         $abbreviations = DateTimeZone::listAbbreviations();
137 137
         foreach ($abbreviations as $abbreviation) {
@@ -326,7 +326,7 @@  discard block
 block discarded – undo
326 326
      */
327 327
     protected static function _modify_datetime_object(DateTime $DateTime, $period = 'years', $value = 1, $operand = '+')
328 328
     {
329
-        if (! $DateTime instanceof DateTime) {
329
+        if ( ! $DateTime instanceof DateTime) {
330 330
             throw new EE_Error(
331 331
                 sprintf(
332 332
                     esc_html__('Expected a PHP DateTime object, but instead received %1$s', 'event_espresso'),
@@ -336,25 +336,25 @@  discard block
 block discarded – undo
336 336
         }
337 337
         switch ($period) {
338 338
             case 'years':
339
-                $value = 'P' . $value . 'Y';
339
+                $value = 'P'.$value.'Y';
340 340
                 break;
341 341
             case 'months':
342
-                $value = 'P' . $value . 'M';
342
+                $value = 'P'.$value.'M';
343 343
                 break;
344 344
             case 'weeks':
345
-                $value = 'P' . $value . 'W';
345
+                $value = 'P'.$value.'W';
346 346
                 break;
347 347
             case 'days':
348
-                $value = 'P' . $value . 'D';
348
+                $value = 'P'.$value.'D';
349 349
                 break;
350 350
             case 'hours':
351
-                $value = 'PT' . $value . 'H';
351
+                $value = 'PT'.$value.'H';
352 352
                 break;
353 353
             case 'minutes':
354
-                $value = 'PT' . $value . 'M';
354
+                $value = 'PT'.$value.'M';
355 355
                 break;
356 356
             case 'seconds':
357
-                $value = 'PT' . $value . 'S';
357
+                $value = 'PT'.$value.'S';
358 358
                 break;
359 359
         }
360 360
         switch ($operand) {
@@ -382,7 +382,7 @@  discard block
 block discarded – undo
382 382
      */
383 383
     protected static function _modify_timestamp($timestamp, $period = 'years', $value = 1, $operand = '+')
384 384
     {
385
-        if (! preg_match(EE_Datetime_Field::unix_timestamp_regex, $timestamp)) {
385
+        if ( ! preg_match(EE_Datetime_Field::unix_timestamp_regex, $timestamp)) {
386 386
             throw new EE_Error(
387 387
                 sprintf(
388 388
                     esc_html__('Expected a Unix timestamp, but instead received %1$s', 'event_espresso'),
@@ -495,7 +495,7 @@  discard block
 block discarded – undo
495 495
                 'date' => $date_format['js'],
496 496
                 'time' => $time_format['js'],
497 497
             ),
498
-            'moment' => $date_format['moment'] . ' ' . $time_format['moment'],
498
+            'moment' => $date_format['moment'].' '.$time_format['moment'],
499 499
             'moment_split' => array(
500 500
                 'date' => $date_format['moment'],
501 501
                 'time' => $time_format['moment']
@@ -518,7 +518,7 @@  discard block
 block discarded – undo
518 518
          *
519 519
          * @var array
520 520
          */
521
-        $symbols_map          = array(
521
+        $symbols_map = array(
522 522
             // Day
523 523
             // 01
524 524
             'd' => array(
@@ -670,26 +670,26 @@  discard block
 block discarded – undo
670 670
         $escaping             = false;
671 671
         $format_string_length = strlen($format_string);
672 672
         for ($i = 0; $i < $format_string_length; $i++) {
673
-            $char = $format_string[ $i ];
673
+            $char = $format_string[$i];
674 674
             if ($char === '\\') { // PHP date format escaping character
675 675
                 $i++;
676 676
                 if ($escaping) {
677
-                    $jquery_ui_format .= $format_string[ $i ];
678
-                    $moment_format    .= $format_string[ $i ];
677
+                    $jquery_ui_format .= $format_string[$i];
678
+                    $moment_format    .= $format_string[$i];
679 679
                 } else {
680
-                    $jquery_ui_format .= '\'' . $format_string[ $i ];
681
-                    $moment_format    .= $format_string[ $i ];
680
+                    $jquery_ui_format .= '\''.$format_string[$i];
681
+                    $moment_format    .= $format_string[$i];
682 682
                 }
683 683
                 $escaping = true;
684 684
             } else {
685 685
                 if ($escaping) {
686 686
                     $jquery_ui_format .= "'";
687 687
                     $moment_format    .= "'";
688
-                    $escaping         = false;
688
+                    $escaping = false;
689 689
                 }
690
-                if (isset($symbols_map[ $char ])) {
691
-                    $jquery_ui_format .= $symbols_map[ $char ]['js'];
692
-                    $moment_format    .= $symbols_map[ $char ]['moment'];
690
+                if (isset($symbols_map[$char])) {
691
+                    $jquery_ui_format .= $symbols_map[$char]['js'];
692
+                    $moment_format    .= $symbols_map[$char]['moment'];
693 693
                 } else {
694 694
                     $jquery_ui_format .= $char;
695 695
                     $moment_format    .= $char;
@@ -746,7 +746,7 @@  discard block
 block discarded – undo
746 746
     public static function dates_represent_one_24_hour_date($date_1, $date_2)
747 747
     {
748 748
 
749
-        if ((! $date_1 instanceof DateTime || ! $date_2 instanceof DateTime)
749
+        if (( ! $date_1 instanceof DateTime || ! $date_2 instanceof DateTime)
750 750
             || ($date_1->format(EE_Datetime_Field::mysql_time_format) !== '00:00:00'
751 751
                 || $date_2->format(
752 752
                     EE_Datetime_Field::mysql_time_format
@@ -782,8 +782,8 @@  discard block
 block discarded – undo
782 782
             ? $DateTimeZone->getOffset(new DateTime('now')) / HOUR_IN_SECONDS
783 783
             : (float) get_option('gmt_offset');
784 784
         $query_interval = $offset < 0
785
-            ? 'DATE_SUB(' . $field_for_interval . ', INTERVAL ' . $offset * -1 . ' HOUR)'
786
-            : 'DATE_ADD(' . $field_for_interval . ', INTERVAL ' . $offset . ' HOUR)';
785
+            ? 'DATE_SUB('.$field_for_interval.', INTERVAL '.$offset * -1.' HOUR)'
786
+            : 'DATE_ADD('.$field_for_interval.', INTERVAL '.$offset.' HOUR)';
787 787
         return $query_interval;
788 788
     }
789 789
 
@@ -800,16 +800,16 @@  discard block
 block discarded – undo
800 800
     public static function get_timezone_string_for_display()
801 801
     {
802 802
         $pretty_timezone = apply_filters('FHEE__EEH_DTT_Helper__get_timezone_string_for_display', '');
803
-        if (! empty($pretty_timezone)) {
803
+        if ( ! empty($pretty_timezone)) {
804 804
             return esc_html($pretty_timezone);
805 805
         }
806 806
         $timezone_string = get_option('timezone_string');
807 807
         if ($timezone_string) {
808 808
             static $mo_loaded = false;
809 809
             // Load translations for continents and cities just like wp_timezone_choice does
810
-            if (! $mo_loaded) {
810
+            if ( ! $mo_loaded) {
811 811
                 $locale = get_locale();
812
-                $mofile = WP_LANG_DIR . '/continents-cities-' . $locale . '.mo';
812
+                $mofile = WP_LANG_DIR.'/continents-cities-'.$locale.'.mo';
813 813
                 load_textdomain('continents-cities', $mofile);
814 814
                 $mo_loaded = true;
815 815
             }
@@ -836,10 +836,10 @@  discard block
 block discarded – undo
836 836
         } else {
837 837
             // convert the part after the decimal, eg "5" (from x.5) or "25" (from x.25)
838 838
             // to minutes, eg 30 or 15, respectively
839
-            $hour_fraction = (float) ('0.' . $parts[1]);
839
+            $hour_fraction = (float) ('0.'.$parts[1]);
840 840
             $parts[1]      = (string) $hour_fraction * 60;
841 841
         }
842
-        return sprintf(__('UTC%1$s', 'event_espresso'), $prefix . implode(':', $parts));
842
+        return sprintf(__('UTC%1$s', 'event_espresso'), $prefix.implode(':', $parts));
843 843
     }
844 844
 
845 845
 
@@ -922,9 +922,9 @@  discard block
 block discarded – undo
922 922
             'Pacific',
923 923
         );
924 924
         // Load translations for continents and cities.
925
-        if (! $mo_loaded || $locale !== $locale_loaded) {
925
+        if ( ! $mo_loaded || $locale !== $locale_loaded) {
926 926
             $locale_loaded = $locale ? $locale : get_locale();
927
-            $mofile        = WP_LANG_DIR . '/continents-cities-' . $locale_loaded . '.mo';
927
+            $mofile        = WP_LANG_DIR.'/continents-cities-'.$locale_loaded.'.mo';
928 928
             unload_textdomain('continents-cities');
929 929
             load_textdomain('continents-cities', $mofile);
930 930
             $mo_loaded = true;
@@ -932,11 +932,11 @@  discard block
 block discarded – undo
932 932
         $zone_data = array();
933 933
         foreach (timezone_identifiers_list() as $zone) {
934 934
             $zone = explode('/', $zone);
935
-            if (! in_array($zone[0], $continents, true)) {
935
+            if ( ! in_array($zone[0], $continents, true)) {
936 936
                 continue;
937 937
             }
938 938
             // This determines what gets set and translated - we don't translate Etc/* strings here, they are done later
939
-            $exists      = array(
939
+            $exists = array(
940 940
                 0 => isset($zone[0]) && $zone[0],
941 941
                 1 => isset($zone[1]) && $zone[1],
942 942
                 2 => isset($zone[2]) && $zone[2],
@@ -966,7 +966,7 @@  discard block
 block discarded – undo
966 966
         usort($zone_data, '_wp_timezone_choice_usort_callback');
967 967
         $structure = array();
968 968
         if (empty($selected_zone)) {
969
-            $structure[] = '<option selected="selected" value="">' . __('Select a city', 'event_espresso') . '</option>';
969
+            $structure[] = '<option selected="selected" value="">'.__('Select a city', 'event_espresso').'</option>';
970 970
         }
971 971
         foreach ($zone_data as $key => $zone) {
972 972
             // Build value in an array to join later
@@ -977,30 +977,30 @@  discard block
 block discarded – undo
977 977
             } else {
978 978
                 // It's inside a continent group
979 979
                 // Continent optgroup
980
-                if (! isset($zone_data[ $key - 1 ]) || $zone_data[ $key - 1 ]['continent'] !== $zone['continent']) {
980
+                if ( ! isset($zone_data[$key - 1]) || $zone_data[$key - 1]['continent'] !== $zone['continent']) {
981 981
                     $label       = $zone['t_continent'];
982
-                    $structure[] = '<optgroup label="' . esc_attr($label) . '">';
982
+                    $structure[] = '<optgroup label="'.esc_attr($label).'">';
983 983
                 }
984 984
                 // Add the city to the value
985 985
                 $value[] = $zone['city'];
986 986
                 $display = $zone['t_city'];
987
-                if (! empty($zone['subcity'])) {
987
+                if ( ! empty($zone['subcity'])) {
988 988
                     // Add the subcity to the value
989 989
                     $value[] = $zone['subcity'];
990
-                    $display .= ' - ' . $zone['t_subcity'];
990
+                    $display .= ' - '.$zone['t_subcity'];
991 991
                 }
992 992
             }
993 993
             // Build the value
994 994
             $value       = implode('/', $value);
995 995
             $selected    = $value === $selected_zone ? ' selected="selected"' : '';
996
-            $structure[] = '<option value="' . esc_attr($value) . '"' . $selected . '>'
996
+            $structure[] = '<option value="'.esc_attr($value).'"'.$selected.'>'
997 997
                            . esc_html($display)
998 998
                            . '</option>';
999 999
             // Close continent optgroup
1000
-            if (! empty($zone['city'])
1000
+            if ( ! empty($zone['city'])
1001 1001
                 && (
1002
-                    ! isset($zone_data[ $key + 1 ])
1003
-                    || (isset($zone_data[ $key + 1 ]) && $zone_data[ $key + 1 ]['continent'] !== $zone['continent'])
1002
+                    ! isset($zone_data[$key + 1])
1003
+                    || (isset($zone_data[$key + 1]) && $zone_data[$key + 1]['continent'] !== $zone['continent'])
1004 1004
                 )
1005 1005
             ) {
1006 1006
                 $structure[] = '</optgroup>';
Please login to merge, or discard this patch.
core/services/helpers/datetime/HelperInterface.php 1 patch
Indentation   +90 added lines, -90 removed lines patch added patch discarded remove patch
@@ -16,94 +16,94 @@
 block discarded – undo
16 16
 interface HelperInterface
17 17
 {
18 18
 
19
-    /**
20
-     * Ensures that a valid timezone string is returned.
21
-     *
22
-     * @param string $timezone_string  When not provided then attempt to use the timezone_string set in the WP Time
23
-     *                                 settings (or derive from set UTC offset).
24
-     * @return string
25
-     */
26
-    public function getValidTimezoneString($timezone_string = '');
27
-
28
-
29
-    /**
30
-     * The only purpose for this static method is to validate that the incoming timezone is a valid php timezone.
31
-     *
32
-     * @param string $timezone_string
33
-     * @param bool   $throw_error
34
-     * @return bool
35
-     */
36
-    public function validateTimezone($timezone_string, $throw_error = true);
37
-
38
-
39
-    /**
40
-     * Returns a timezone string for the provided gmt_offset.
41
-     * @param float|string $gmt_offset
42
-     * @return string
43
-     */
44
-    public function getTimezoneStringFromGmtOffset($gmt_offset = '');
45
-
46
-
47
-    /**
48
-     * Gets the site's GMT offset based on either the timezone string
49
-     * (in which case the gmt offset will vary depending on the location's
50
-     * observance of daylight savings time) or the gmt_offset wp option
51
-     *
52
-     * @return int  seconds offset
53
-     */
54
-    public function getSiteTimezoneGmtOffset();
55
-
56
-
57
-    /**
58
-     * Get timezone transitions
59
-     * @param DateTimeZone $date_time_zone
60
-     * @param int|null     $time
61
-     * @param bool         $first_only
62
-     * @return array
63
-     */
64
-    public function getTimezoneTransitions(DateTimeZone $date_time_zone, $time = null, $first_only = true);
65
-
66
-
67
-    /**
68
-     * Get Timezone offset for given timezone object
69
-     * @param DateTimeZone $date_time_zone
70
-     * @param null|int         $time
71
-     * @return int
72
-     */
73
-    public function getTimezoneOffset(DateTimeZone $date_time_zone, $time = null);
74
-
75
-
76
-    /**
77
-     * Provide a timezone select input
78
-     * @param string $timezone_string
79
-     * @return string
80
-     */
81
-    public function timezoneSelectInput($timezone_string = '');
82
-
83
-
84
-    /**
85
-     * This method will take an incoming unix timestamp and add the offset to it for the given timezone_string.
86
-     * If no unix timestamp is given then time() is used.  If no timezone is given then the set timezone string for
87
-     * the site is used.
88
-     * This is used typically when using a Unix timestamp any core WP functions that expect their specially
89
-     * computed timestamp (i.e. date_i18n() )
90
-     *
91
-     * @param int    $unix_timestamp    if 0, then time() will be used.
92
-     * @param string $timezone_string timezone_string. If empty, then the current set timezone for the
93
-     *                                site will be used.
94
-     * @return int      unix_timestamp value with the offset applied for the given timezone.
95
-     */
96
-    public function getTimestampWithOffset($unix_timestamp = 0, $timezone_string = '');
97
-
98
-
99
-    /**
100
-     * Depending on PHP version,
101
-     * there might not be valid current timezone strings to match these gmt_offsets in its timezone tables.
102
-     * To get around that, for these fringe timezones we bump them to a known valid offset.
103
-     * This method should ONLY be called after first verifying an timezone_string cannot be retrieved for the offset.
104
-     *
105
-     * @param int $gmt_offset
106
-     * @return int
107
-     */
108
-    public function adjustInvalidGmtOffsets($gmt_offset);
19
+	/**
20
+	 * Ensures that a valid timezone string is returned.
21
+	 *
22
+	 * @param string $timezone_string  When not provided then attempt to use the timezone_string set in the WP Time
23
+	 *                                 settings (or derive from set UTC offset).
24
+	 * @return string
25
+	 */
26
+	public function getValidTimezoneString($timezone_string = '');
27
+
28
+
29
+	/**
30
+	 * The only purpose for this static method is to validate that the incoming timezone is a valid php timezone.
31
+	 *
32
+	 * @param string $timezone_string
33
+	 * @param bool   $throw_error
34
+	 * @return bool
35
+	 */
36
+	public function validateTimezone($timezone_string, $throw_error = true);
37
+
38
+
39
+	/**
40
+	 * Returns a timezone string for the provided gmt_offset.
41
+	 * @param float|string $gmt_offset
42
+	 * @return string
43
+	 */
44
+	public function getTimezoneStringFromGmtOffset($gmt_offset = '');
45
+
46
+
47
+	/**
48
+	 * Gets the site's GMT offset based on either the timezone string
49
+	 * (in which case the gmt offset will vary depending on the location's
50
+	 * observance of daylight savings time) or the gmt_offset wp option
51
+	 *
52
+	 * @return int  seconds offset
53
+	 */
54
+	public function getSiteTimezoneGmtOffset();
55
+
56
+
57
+	/**
58
+	 * Get timezone transitions
59
+	 * @param DateTimeZone $date_time_zone
60
+	 * @param int|null     $time
61
+	 * @param bool         $first_only
62
+	 * @return array
63
+	 */
64
+	public function getTimezoneTransitions(DateTimeZone $date_time_zone, $time = null, $first_only = true);
65
+
66
+
67
+	/**
68
+	 * Get Timezone offset for given timezone object
69
+	 * @param DateTimeZone $date_time_zone
70
+	 * @param null|int         $time
71
+	 * @return int
72
+	 */
73
+	public function getTimezoneOffset(DateTimeZone $date_time_zone, $time = null);
74
+
75
+
76
+	/**
77
+	 * Provide a timezone select input
78
+	 * @param string $timezone_string
79
+	 * @return string
80
+	 */
81
+	public function timezoneSelectInput($timezone_string = '');
82
+
83
+
84
+	/**
85
+	 * This method will take an incoming unix timestamp and add the offset to it for the given timezone_string.
86
+	 * If no unix timestamp is given then time() is used.  If no timezone is given then the set timezone string for
87
+	 * the site is used.
88
+	 * This is used typically when using a Unix timestamp any core WP functions that expect their specially
89
+	 * computed timestamp (i.e. date_i18n() )
90
+	 *
91
+	 * @param int    $unix_timestamp    if 0, then time() will be used.
92
+	 * @param string $timezone_string timezone_string. If empty, then the current set timezone for the
93
+	 *                                site will be used.
94
+	 * @return int      unix_timestamp value with the offset applied for the given timezone.
95
+	 */
96
+	public function getTimestampWithOffset($unix_timestamp = 0, $timezone_string = '');
97
+
98
+
99
+	/**
100
+	 * Depending on PHP version,
101
+	 * there might not be valid current timezone strings to match these gmt_offsets in its timezone tables.
102
+	 * To get around that, for these fringe timezones we bump them to a known valid offset.
103
+	 * This method should ONLY be called after first verifying an timezone_string cannot be retrieved for the offset.
104
+	 *
105
+	 * @param int $gmt_offset
106
+	 * @return int
107
+	 */
108
+	public function adjustInvalidGmtOffsets($gmt_offset);
109 109
 }
Please login to merge, or discard this patch.
core/services/context/ContextChecker.php 2 patches
Spacing   +1 added lines, -1 removed lines patch added patch discarded remove patch
@@ -87,7 +87,7 @@
 block discarded – undo
87 87
     {
88 88
         $this->evaluation_callback = $evaluation_callback instanceof Closure
89 89
             ? $evaluation_callback
90
-            : function (ContextInterface $context, $acceptable_values) {
90
+            : function(ContextInterface $context, $acceptable_values) {
91 91
                 return in_array($context->slug(), $acceptable_values, true);
92 92
             };
93 93
     }
Please login to merge, or discard this patch.
Indentation   +141 added lines, -141 removed lines patch added patch discarded remove patch
@@ -17,145 +17,145 @@
 block discarded – undo
17 17
 class ContextChecker
18 18
 {
19 19
 
20
-    /**
21
-     * A unique string used to identify where this ContextChecker is being employed
22
-     * Is currently only used within the hook name for the filterable return value of isAllowed().
23
-     *
24
-     * @var string $identifier
25
-     */
26
-    private $identifier;
27
-
28
-    /**
29
-     * A list of values to be compared against the slug of the Context class passed to isAllowed()
30
-     *
31
-     * @var array $acceptable_values
32
-     */
33
-    private $acceptable_values;
34
-
35
-    /**
36
-     * Closure that will be called to perform the evaluation within isAllowed().
37
-     * If none is provided, then a simple type sensitive in_array() check will be used
38
-     * and return true if the incoming Context::slug() is found within the array of $acceptable_values.
39
-     *
40
-     * @var Closure $evaluation_callback
41
-     */
42
-    private $evaluation_callback;
43
-
44
-
45
-    /**
46
-     * ContextChecker constructor.
47
-     *
48
-     * @param string       $identifier
49
-     * @param array        $acceptable_values
50
-     * @param Closure|null $evaluation_callback [optional]
51
-     */
52
-    public function __construct($identifier, array $acceptable_values, Closure $evaluation_callback = null)
53
-    {
54
-        $this->setIdentifier($identifier);
55
-        $this->setAcceptableValues($acceptable_values);
56
-        $this->setEvaluationCallback($evaluation_callback);
57
-    }
58
-
59
-
60
-    /**
61
-     * @param string $identifier
62
-     */
63
-    private function setIdentifier($identifier)
64
-    {
65
-        $this->identifier = sanitize_key($identifier);
66
-    }
67
-
68
-
69
-    /**
70
-     * @param array $acceptable_values
71
-     */
72
-    private function setAcceptableValues(array $acceptable_values)
73
-    {
74
-        $this->acceptable_values = $acceptable_values;
75
-    }
76
-
77
-
78
-    /**
79
-     * @param Closure $evaluation_callback
80
-     */
81
-    private function setEvaluationCallback(Closure $evaluation_callback = null)
82
-    {
83
-        $this->evaluation_callback = $evaluation_callback instanceof Closure
84
-            ? $evaluation_callback
85
-            : function (ContextInterface $context, $acceptable_values) {
86
-                return in_array($context->slug(), $acceptable_values, true);
87
-            };
88
-    }
89
-
90
-
91
-    /**
92
-     * @return string
93
-     */
94
-    protected function identifier()
95
-    {
96
-        return $this->identifier;
97
-    }
98
-
99
-
100
-    /**
101
-     * @return array
102
-     */
103
-    protected function acceptableValues()
104
-    {
105
-        return apply_filters(
106
-            "FHEE__EventEspresso_core_domain_entities_context_ContextChecker__{$this->identifier}__acceptableValues",
107
-            $this->acceptable_values
108
-        );
109
-    }
110
-
111
-
112
-    /**
113
-     * @return Closure
114
-     */
115
-    protected function evaluationCallback()
116
-    {
117
-        return $this->evaluation_callback;
118
-    }
119
-
120
-
121
-    /**
122
-     * Returns true if the incoming Context class slug matches one of the preset acceptable values.
123
-     * The result is filterable using the identifier for this ContextChecker.
124
-     * example:
125
-     * If this ContextChecker's $identifier was set to "registration-checkout-type",
126
-     * then the filter here would be named:
127
-     *  "FHEE__EventEspresso_core_domain_entities_context_ContextChecker__registration-checkout-type__isAllowed".
128
-     * Other code could hook into the filter in isAllowed() using the above name
129
-     * and test for additional acceptable values.
130
-     * So if the set of $acceptable_values was: [ "initial-visit",  "revisit" ]
131
-     * then adding a filter to
132
-     *  "FHEE__EventEspresso_core_domain_entities_context_ContextChecker__registration-checkout-type__isAllowed",
133
-     * would allow you to perform your own conditional and allow "wait-list-checkout" as an acceptable value.
134
-     *  example:
135
-     *      add_filter(
136
-     *          'FHEE__EventEspresso_core_domain_entities_context_ContextChecker__registration-checkout-type__isAllowed',
137
-     *          function ($is_allowed, ContextInterface $context) { return $context->slug() === 'wait-list-checkout'
138
-     *                  ? true
139
-     *                  : $is_allowed;
140
-     *          },
141
-     *          10,
142
-     *          2
143
-     *      );
144
-     *
145
-     * @param ContextInterface $context
146
-     * @return boolean
147
-     */
148
-    public function isAllowed(ContextInterface $context)
149
-    {
150
-        $evaluation_callback = $this->evaluationCallback();
151
-        return filter_var(
152
-            apply_filters(
153
-                "FHEE__EventEspresso_core_domain_entities_context_ContextChecker__{$this->identifier}__isAllowed",
154
-                $evaluation_callback($context, $this->acceptableValues()),
155
-                $context,
156
-                $this
157
-            ),
158
-            FILTER_VALIDATE_BOOLEAN
159
-        );
160
-    }
20
+	/**
21
+	 * A unique string used to identify where this ContextChecker is being employed
22
+	 * Is currently only used within the hook name for the filterable return value of isAllowed().
23
+	 *
24
+	 * @var string $identifier
25
+	 */
26
+	private $identifier;
27
+
28
+	/**
29
+	 * A list of values to be compared against the slug of the Context class passed to isAllowed()
30
+	 *
31
+	 * @var array $acceptable_values
32
+	 */
33
+	private $acceptable_values;
34
+
35
+	/**
36
+	 * Closure that will be called to perform the evaluation within isAllowed().
37
+	 * If none is provided, then a simple type sensitive in_array() check will be used
38
+	 * and return true if the incoming Context::slug() is found within the array of $acceptable_values.
39
+	 *
40
+	 * @var Closure $evaluation_callback
41
+	 */
42
+	private $evaluation_callback;
43
+
44
+
45
+	/**
46
+	 * ContextChecker constructor.
47
+	 *
48
+	 * @param string       $identifier
49
+	 * @param array        $acceptable_values
50
+	 * @param Closure|null $evaluation_callback [optional]
51
+	 */
52
+	public function __construct($identifier, array $acceptable_values, Closure $evaluation_callback = null)
53
+	{
54
+		$this->setIdentifier($identifier);
55
+		$this->setAcceptableValues($acceptable_values);
56
+		$this->setEvaluationCallback($evaluation_callback);
57
+	}
58
+
59
+
60
+	/**
61
+	 * @param string $identifier
62
+	 */
63
+	private function setIdentifier($identifier)
64
+	{
65
+		$this->identifier = sanitize_key($identifier);
66
+	}
67
+
68
+
69
+	/**
70
+	 * @param array $acceptable_values
71
+	 */
72
+	private function setAcceptableValues(array $acceptable_values)
73
+	{
74
+		$this->acceptable_values = $acceptable_values;
75
+	}
76
+
77
+
78
+	/**
79
+	 * @param Closure $evaluation_callback
80
+	 */
81
+	private function setEvaluationCallback(Closure $evaluation_callback = null)
82
+	{
83
+		$this->evaluation_callback = $evaluation_callback instanceof Closure
84
+			? $evaluation_callback
85
+			: function (ContextInterface $context, $acceptable_values) {
86
+				return in_array($context->slug(), $acceptable_values, true);
87
+			};
88
+	}
89
+
90
+
91
+	/**
92
+	 * @return string
93
+	 */
94
+	protected function identifier()
95
+	{
96
+		return $this->identifier;
97
+	}
98
+
99
+
100
+	/**
101
+	 * @return array
102
+	 */
103
+	protected function acceptableValues()
104
+	{
105
+		return apply_filters(
106
+			"FHEE__EventEspresso_core_domain_entities_context_ContextChecker__{$this->identifier}__acceptableValues",
107
+			$this->acceptable_values
108
+		);
109
+	}
110
+
111
+
112
+	/**
113
+	 * @return Closure
114
+	 */
115
+	protected function evaluationCallback()
116
+	{
117
+		return $this->evaluation_callback;
118
+	}
119
+
120
+
121
+	/**
122
+	 * Returns true if the incoming Context class slug matches one of the preset acceptable values.
123
+	 * The result is filterable using the identifier for this ContextChecker.
124
+	 * example:
125
+	 * If this ContextChecker's $identifier was set to "registration-checkout-type",
126
+	 * then the filter here would be named:
127
+	 *  "FHEE__EventEspresso_core_domain_entities_context_ContextChecker__registration-checkout-type__isAllowed".
128
+	 * Other code could hook into the filter in isAllowed() using the above name
129
+	 * and test for additional acceptable values.
130
+	 * So if the set of $acceptable_values was: [ "initial-visit",  "revisit" ]
131
+	 * then adding a filter to
132
+	 *  "FHEE__EventEspresso_core_domain_entities_context_ContextChecker__registration-checkout-type__isAllowed",
133
+	 * would allow you to perform your own conditional and allow "wait-list-checkout" as an acceptable value.
134
+	 *  example:
135
+	 *      add_filter(
136
+	 *          'FHEE__EventEspresso_core_domain_entities_context_ContextChecker__registration-checkout-type__isAllowed',
137
+	 *          function ($is_allowed, ContextInterface $context) { return $context->slug() === 'wait-list-checkout'
138
+	 *                  ? true
139
+	 *                  : $is_allowed;
140
+	 *          },
141
+	 *          10,
142
+	 *          2
143
+	 *      );
144
+	 *
145
+	 * @param ContextInterface $context
146
+	 * @return boolean
147
+	 */
148
+	public function isAllowed(ContextInterface $context)
149
+	{
150
+		$evaluation_callback = $this->evaluationCallback();
151
+		return filter_var(
152
+			apply_filters(
153
+				"FHEE__EventEspresso_core_domain_entities_context_ContextChecker__{$this->identifier}__isAllowed",
154
+				$evaluation_callback($context, $this->acceptableValues()),
155
+				$context,
156
+				$this
157
+			),
158
+			FILTER_VALIDATE_BOOLEAN
159
+		);
160
+	}
161 161
 }
Please login to merge, or discard this patch.
core/domain/entities/contexts/ContextInterface.php 1 patch
Indentation   +8 added lines, -8 removed lines patch added patch discarded remove patch
@@ -18,14 +18,14 @@
 block discarded – undo
18 18
 interface ContextInterface
19 19
 {
20 20
 
21
-    /**
22
-     * @return string
23
-     */
24
-    public function slug();
21
+	/**
22
+	 * @return string
23
+	 */
24
+	public function slug();
25 25
 
26 26
 
27
-    /**
28
-     * @return string
29
-     */
30
-    public function description();
27
+	/**
28
+	 * @return string
29
+	 */
30
+	public function description();
31 31
 }
Please login to merge, or discard this patch.
core/db_models/EEM_Question.model.php 3 patches
Doc Comments   +2 added lines, -2 removed lines patch added patch discarded remove patch
@@ -364,7 +364,7 @@  discard block
 block discarded – undo
364 364
      * Gets an array for converting between QST_system and QST_IDs for system questions. Eg, if you want to know
365 365
      * which system question QST_ID corresponds to the QST_system 'city', use EEM_Question::instance()->get_Question_ID_from_system_string('city');
366 366
      * @param $QST_system
367
-     * @return int of QST_ID for the question that corresponds to that QST_system
367
+     * @return string of QST_ID for the question that corresponds to that QST_system
368 368
      */
369 369
     public function get_Question_ID_from_system_string($QST_system)
370 370
     {
@@ -425,7 +425,7 @@  discard block
 block discarded – undo
425 425
 
426 426
 
427 427
     /**
428
-     * @return array
428
+     * @return EEM_Question
429 429
      */
430 430
     public function question_descriptions()
431 431
     {
Please login to merge, or discard this patch.
Indentation   +425 added lines, -425 removed lines patch added patch discarded remove patch
@@ -10,429 +10,429 @@
 block discarded – undo
10 10
 class EEM_Question extends EEM_Soft_Delete_Base
11 11
 {
12 12
 
13
-    // constant used to indicate that the question type is COUNTRY
14
-    const QST_type_country = 'COUNTRY';
15
-
16
-    // constant used to indicate that the question type is DATE
17
-    const QST_type_date = 'DATE';
18
-
19
-    // constant used to indicate that the question type is DROPDOWN
20
-    const QST_type_dropdown = 'DROPDOWN';
21
-
22
-    // constant used to indicate that the question type is CHECKBOX
23
-    const QST_type_checkbox = 'CHECKBOX';
24
-
25
-    // constant used to indicate that the question type is RADIO_BTN
26
-    const QST_type_radio = 'RADIO_BTN';
27
-
28
-    // constant used to indicate that the question type is STATE
29
-    const QST_type_state = 'STATE';
30
-
31
-    // constant used to indicate that the question type is TEXT
32
-    const QST_type_text = 'TEXT';
33
-
34
-    // constant used to indicate that the question type is TEXTAREA
35
-    const QST_type_textarea = 'TEXTAREA';
36
-
37
-    // constant used to indicate that the question type is a TEXTAREA that allows simple html
38
-    const QST_type_html_textarea = 'HTML_TEXTAREA';
39
-
40
-    // constant used to indicate that the question type is an email input
41
-    const QST_type_email = 'EMAIL';
42
-
43
-    // constant used to indicate that the question type is a US-formatted phone number
44
-    const QST_type_us_phone = 'US_PHONE';
45
-
46
-    // constant used to indicate that the question type is an integer (whole number)
47
-    const QST_type_int = 'INTEGER';
48
-
49
-    // constant used to indicate that the question type is a decimal (float)
50
-    const QST_type_decimal = 'DECIMAL';
51
-
52
-    // constant used to indicate that the question type is a valid URL
53
-    const QST_type_url = 'URL';
54
-
55
-    // constant used to indicate that the question type is a YEAR
56
-    const QST_type_year = 'YEAR';
57
-
58
-    // constant used to indicate that the question type is a multi-select
59
-    const QST_type_multi_select = 'MULTI_SELECT';
60
-
61
-    /**
62
-     * Question types that are interchangeable, even after answers have been provided for them.
63
-     * Top-level keys are category slugs, next level is an array of question types. If question types
64
-     * aren't in this array, it is assumed they AREN'T interchangeable with any other question types.
65
-     *
66
-     * @access protected
67
-     * @var array $_question_type_categories {
68
-     * @type string $text
69
-     * @type string $single -answer-enum
70
-     * @type string $multi -answer-enum
71
-     *                    }
72
-     */
73
-    protected $_question_type_categories = array();
74
-
75
-    /**
76
-     * lists all the question types which should be allowed. Ideally, this will be extensible.
77
-     *
78
-     * @access protected
79
-     * @var array $_allowed_question_types
80
-     */
81
-    protected $_allowed_question_types = array();
82
-
83
-    /**
84
-     * brief descriptions for all the question types
85
-     *
86
-     * @access protected
87
-     * @var EEM_Question $_instance
88
-     */
89
-    protected $_question_descriptions;
90
-
91
-
92
-    /**
93
-     * Question types that should have an admin-defined max input length
94
-     * @var array
95
-     */
96
-    protected $question_types_with_max_lengh;
97
-
98
-
99
-    // private instance of the Attendee object
100
-    protected static $_instance = null;
101
-
102
-
103
-    /**
104
-     * EEM_Question constructor.
105
-     *
106
-     * @param null $timezone
107
-     */
108
-    protected function __construct($timezone = null)
109
-    {
110
-        $this->singular_item = __('Question', 'event_espresso');
111
-        $this->plural_item = __('Questions', 'event_espresso');
112
-        $this->_allowed_question_types = apply_filters(
113
-            'FHEE__EEM_Question__construct__allowed_question_types',
114
-            array(
115
-                EEM_Question::QST_type_text => __('Text', 'event_espresso'),
116
-                EEM_Question::QST_type_textarea => __('Textarea', 'event_espresso'),
117
-                EEM_Question::QST_type_checkbox => __('Checkboxes', 'event_espresso'),
118
-                EEM_Question::QST_type_radio => __('Radio Buttons', 'event_espresso'),
119
-                EEM_Question::QST_type_dropdown => __('Dropdown', 'event_espresso'),
120
-                EEM_Question::QST_type_state => __('State/Province Dropdown', 'event_espresso'),
121
-                EEM_Question::QST_type_country => __('Country Dropdown', 'event_espresso'),
122
-                EEM_Question::QST_type_date => __('Date Picker', 'event_espresso'),
123
-                EEM_Question::QST_type_html_textarea => __('HTML Textarea', 'event_espresso'),
124
-                EEM_Question::QST_type_email => __('Email', 'event_espresso'),
125
-                EEM_Question::QST_type_us_phone => __('USA - Format Phone', 'event_espresso'),
126
-                EEM_Question::QST_type_decimal => __('Number', 'event_espresso'),
127
-                EEM_Question::QST_type_int => __('Whole Number', 'event_espresso'),
128
-                EEM_Question::QST_type_url => __('URL', 'event_espresso'),
129
-                EEM_Question::QST_type_year => __('Year', 'event_espresso'),
130
-                EEM_Question::QST_type_multi_select => __('Multi Select', 'event_espresso')
131
-            )
132
-        );
133
-        $this->_question_descriptions = apply_filters(
134
-            'FHEE__EEM_Question__construct__allowed_question_types',
135
-            array(
136
-                EEM_Question::QST_type_text => __('A single line text input field', 'event_espresso'),
137
-                EEM_Question::QST_type_textarea => __('A multi line text input field', 'event_espresso'),
138
-                EEM_Question::QST_type_checkbox => __('Allows multiple preset options to be selected', 'event_espresso'),
139
-                EEM_Question::QST_type_radio => __('Allows a single preset option to be selected', 'event_espresso'),
140
-                EEM_Question::QST_type_dropdown => __('A dropdown that allows a single selection', 'event_espresso'),
141
-                EEM_Question::QST_type_state => __('A dropdown that lists states/provinces', 'event_espresso'),
142
-                EEM_Question::QST_type_country => __('A dropdown that lists countries', 'event_espresso'),
143
-                EEM_Question::QST_type_date => __('A popup calendar that allows date selections', 'event_espresso'),
144
-                EEM_Question::QST_type_html_textarea => __('A multi line text input field that allows HTML', 'event_espresso'),
145
-                EEM_Question::QST_type_email => __('A text field that must contain a valid Email address', 'event_espresso'),
146
-                EEM_Question::QST_type_us_phone => __('A text field that must contain a valid US phone number', 'event_espresso'),
147
-                EEM_Question::QST_type_decimal => __('A text field that allows number values with decimals', 'event_espresso'),
148
-                EEM_Question::QST_type_int => __('A text field that only allows whole numbers (no decimals)', 'event_espresso'),
149
-                EEM_Question::QST_type_url => __('A text field that must contain a valid URL', 'event_espresso'),
150
-                EEM_Question::QST_type_year => __('A dropdown that lists the last 100 years', 'event_espresso'),
151
-                EEM_Question::QST_type_multi_select => __('A dropdown that allows multiple selections', 'event_espresso')
152
-            )
153
-        );
154
-        $this->_question_type_categories = (array) apply_filters(
155
-            'FHEE__EEM_Question__construct__question_type_categories',
156
-            array(
157
-                'text' => array(
158
-                    EEM_Question::QST_type_text,
159
-                    EEM_Question::QST_type_textarea,
160
-                    EEM_Question::QST_type_date,
161
-                    EEM_Question::QST_type_html_textarea,
162
-                    EEM_Question::QST_type_email,
163
-                    EEM_Question::QST_type_us_phone,
164
-                    EEM_Question::QST_type_decimal,
165
-                    EEM_Question::QST_type_int,
166
-                    EEM_Question::QST_type_url,
167
-                    EEM_Question::QST_type_year
168
-                ),
169
-                'single-answer-enum' => array(
170
-                    EEM_Question::QST_type_radio,
171
-                    EEM_Question::QST_type_dropdown
172
-                ),
173
-                'multi-answer-enum' => array(
174
-                    EEM_Question::QST_type_checkbox,
175
-                    EEM_Question::QST_type_multi_select
176
-                )
177
-            )
178
-        );
179
-        $this->question_types_with_max_lengh = apply_filters(
180
-            'FHEE__EEM_Question___construct__question_types_with_max_length',
181
-            array(
182
-                EEM_Question::QST_type_text,
183
-                EEM_Question::QST_type_textarea,
184
-                EEM_Question::QST_type_html_textarea
185
-            )
186
-        );
187
-
188
-        $this->_tables = array(
189
-            'Question' => new EE_Primary_Table('esp_question', 'QST_ID')
190
-        );
191
-        $this->_fields = array(
192
-            'Question' => array(
193
-                'QST_ID' => new EE_Primary_Key_Int_Field('QST_ID', __('Question ID', 'event_espresso')),
194
-                'QST_display_text' => new EE_Post_Content_Field('QST_display_text', __('Question Text', 'event_espresso'), true, ''),
195
-                'QST_admin_label' => new EE_Plain_Text_Field('QST_admin_label', __('Question Label (admin-only)', 'event_espresso'), true, ''),
196
-                'QST_system' => new EE_Plain_Text_Field('QST_system', __('Internal string ID for question', 'event_espresso'), false, ''),
197
-                'QST_type' => new EE_Enum_Text_Field('QST_type', __('Question Type', 'event_espresso'), false, 'TEXT', $this->_allowed_question_types),
198
-                'QST_required' => new EE_Boolean_Field('QST_required', __('Required Question?', 'event_espresso'), false, false),
199
-                'QST_required_text' => new EE_Simple_HTML_Field('QST_required_text', __('Text to Display if Not Provided', 'event_espresso'), true, ''),
200
-                'QST_order' => new EE_Integer_Field('QST_order', __('Question Order', 'event_espresso'), false, 0),
201
-                'QST_admin_only' => new EE_Boolean_Field('QST_admin_only', __('Admin-Only Question?', 'event_espresso'), false, false),
202
-                'QST_max' => new EE_Infinite_Integer_Field('QST_max', __('Max Size', 'event_espresso'), false, EE_INF),
203
-                'QST_wp_user' => new EE_WP_User_Field('QST_wp_user', __('Question Creator ID', 'event_espresso'), false),
204
-                'QST_deleted' => new EE_Trashed_Flag_Field('QST_deleted', __('Flag Indicating question was deleted', 'event_espresso'), false, false)
205
-            )
206
-        );
207
-        $this->_model_relations = array(
208
-            'Question_Group' => new EE_HABTM_Relation('Question_Group_Question'),
209
-            'Question_Option' => new EE_Has_Many_Relation(),
210
-            'Answer' => new EE_Has_Many_Relation(),
211
-            'WP_User' => new EE_Belongs_To_Relation(),
212
-            // for QST_order column
213
-            'Question_Group_Question' => new EE_Has_Many_Relation()
214
-        );
215
-        // this model is generally available for reading
216
-        $this->_cap_restriction_generators[ EEM_Base::caps_read ] = new EE_Restriction_Generator_Public();
217
-        $this->_cap_restriction_generators[ EEM_Base::caps_read_admin ] = new EE_Restriction_Generator_Reg_Form('QST_system');
218
-        $this->_cap_restriction_generators[ EEM_Base::caps_edit ] = new EE_Restriction_Generator_Reg_Form('QST_system');
219
-        $this->_cap_restriction_generators[ EEM_Base::caps_delete ] = new EE_Restriction_Generator_Reg_Form('QST_system');
220
-        parent::__construct($timezone);
221
-    }
222
-
223
-    /**
224
-     * Returns the list of allowed question types, which are normally: 'TEXT','TEXTAREA','RADIO_BTN','DROPDOWN','CHECKBOX','DATE'
225
-     * but they can be extended
226
-     * @return string[]
227
-     */
228
-    public function allowed_question_types()
229
-    {
230
-        return $this->_allowed_question_types;
231
-    }
232
-
233
-    /**
234
-     * Gets all the question types in the same category
235
-     * @param string $question_type one of EEM_Question::allowed_question_types(
236
-     * @return string[] like EEM_Question::allowed_question_types()
237
-     */
238
-    public function question_types_in_same_category($question_type)
239
-    {
240
-        $question_types = array($question_type);
241
-        foreach ($this->_question_type_categories as $category => $question_types_in_category) {
242
-            if (in_array($question_type, $question_types_in_category)) {
243
-                $question_types = $question_types_in_category;
244
-                break;
245
-            }
246
-        }
247
-
248
-        return array_intersect_key($this->allowed_question_types(), array_flip($question_types));
249
-    }
250
-
251
-    /**
252
-     * Determines if the given question type is in the given question type category
253
-     * @param string $question_type one of EEM_Question::allowed_question_types()
254
-     * @param string $category one of the top-level keys of EEM_Question::question_type_categories()
255
-     * @return boolean
256
-     */
257
-    public function question_type_is_in_category($question_type, $category)
258
-    {
259
-        if (!isset($this->_question_type_categories[ $category ])) {
260
-            return false;
261
-        }
262
-        return in_array($question_type, $this->_question_type_categories[ $category ]);
263
-    }
264
-
265
-
266
-    /**
267
-     * Returns all the question types in the given category
268
-     * @param string $category
269
-     * @return array|mixed
270
-     */
271
-    public function question_types_in_category($category)
272
-    {
273
-        if (isset($this->_question_type_categories[ $category ])) {
274
-            return $this->_question_type_categories[ $category ];
275
-        }
276
-        return array();
277
-    }
278
-
279
-
280
-    /**
281
-     * Returns all the question types that should have question options
282
-     * @return array
283
-     */
284
-    public function question_types_with_options()
285
-    {
286
-        return array_merge(
287
-            $this->question_types_in_category('single-answer-enum'),
288
-            $this->question_types_in_category('multi-answer-enum')
289
-        );
290
-    }
291
-
292
-    /**
293
-     * Returns the question type categories 2d array
294
-     * @return array see EEM_Question::_question_type_categories
295
-     */
296
-    public function question_type_categories()
297
-    {
298
-        return $this->_question_type_categories;
299
-    }
300
-
301
-    /**
302
-     * Returns an array of all the QST_system values that can be allowed in the system question group
303
-     * identified by $system_question_group_id
304
-     * @param string $system_question_group_id QSG_system
305
-     * @return array of system question names (QST_system)
306
-     */
307
-    public function allowed_system_questions_in_system_question_group($system_question_group_id)
308
-    {
309
-        $question_system_ids = array();
310
-        switch ($system_question_group_id) {
311
-            case EEM_Question_Group::system_personal:
312
-                $question_system_ids = array(
313
-                    EEM_Attendee::system_question_fname,
314
-                    EEM_Attendee::system_question_lname,
315
-                    EEM_Attendee::system_question_email,
316
-                    EEM_Attendee::system_question_phone
317
-                );
318
-                break;
319
-            case EEM_Question_Group::system_address:
320
-                $question_system_ids = array(
321
-                    EEM_Attendee::system_question_address,
322
-                    EEM_Attendee::system_question_address2,
323
-                    EEM_Attendee::system_question_city,
324
-                    EEM_Attendee::system_question_state,
325
-                    EEM_Attendee::system_question_country,
326
-                    EEM_Attendee::system_question_zip,
327
-                    EEM_Attendee::system_question_phone
328
-                );
329
-                break;
330
-        }
331
-        return apply_filters('FHEE__EEM_Question__system_questions_allowed_in_system_question_group__return', $question_system_ids, $system_question_group_id);
332
-    }
333
-
334
-    /**
335
-     * Returns an array of all the QST_system values that are required in the system question group
336
-     * identified by $system_question_group_id
337
-     * @param string $system_question_group_id QSG_system
338
-     * @return array of system question names (QST_system)
339
-     */
340
-    public function required_system_questions_in_system_question_group($system_question_group_id)
341
-    {
342
-        $question_system_ids = null;
343
-        switch ($system_question_group_id) {
344
-            case EEM_Question_Group::system_personal:
345
-                $question_system_ids = array(
346
-                    EEM_Attendee::system_question_fname,
347
-                    EEM_Attendee::system_question_email,
348
-                );
349
-                break;
350
-            default:
351
-                $question_system_ids = array();
352
-        }
353
-        return apply_filters('FHEE__EEM_Question__system_questions_required_in_system_question_group', $question_system_ids, $system_question_group_id);
354
-    }
355
-
356
-
357
-    /**
358
-     * Gets an array for converting between QST_system and QST_IDs for system questions. Eg, if you want to know
359
-     * which system question QST_ID corresponds to the QST_system 'city', use EEM_Question::instance()->get_Question_ID_from_system_string('city');
360
-     * @param $QST_system
361
-     * @return int of QST_ID for the question that corresponds to that QST_system
362
-     */
363
-    public function get_Question_ID_from_system_string($QST_system)
364
-    {
365
-        return $this->get_var(array(array('QST_system' => $QST_system)));
366
-    }
367
-
368
-
369
-    /**
370
-     * searches the db for the question with the latest question order and returns that value.
371
-     * @access public
372
-     * @return int
373
-     */
374
-    public function get_latest_question_order()
375
-    {
376
-        $columns_to_select = array(
377
-            'max_order' => array("MAX(QST_order)", "%d")
378
-        );
379
-        $max = $this->_get_all_wpdb_results(array(), ARRAY_A, $columns_to_select);
380
-        return isset($max[0], $max[0]['max_order']) ? $max[0]['max_order'] : 0;
381
-    }
382
-
383
-    /**
384
-     * Returns an array where keys are system question QST_system values,
385
-     * and values are the highest question max the admin can set on the question
386
-     * (aka the "max max"; eg, a site admin can change the zip question to have a max
387
-     * of 5, but no larger than 12)
388
-     * @return array
389
-     */
390
-    public function system_question_maxes()
391
-    {
392
-        return array(
393
-            'fname' => 45,
394
-            'lname' => 45,
395
-            'address' => 255,
396
-            'address2' => 255,
397
-            'city' => 45,
398
-            'zip' => 12,
399
-            'email' => 255,
400
-            'phone' => 45,
401
-        );
402
-    }
403
-
404
-    /**
405
-     * Given a QST_system value, gets the question's largest allowable max input.
406
-     * @see Registration_Form_Admin_Page::system_question_maxes()
407
-     * @param string $system_question_value
408
-     * @return int|float
409
-     */
410
-    public function absolute_max_for_system_question($system_question_value)
411
-    {
412
-        $maxes = $this->system_question_maxes();
413
-        if (isset($maxes[ $system_question_value ])) {
414
-            return $maxes[ $system_question_value ];
415
-        } else {
416
-            return EE_INF;
417
-        }
418
-    }
419
-
420
-
421
-    /**
422
-     * @return array
423
-     */
424
-    public function question_descriptions()
425
-    {
426
-        return $this->_question_descriptions;
427
-    }
428
-
429
-
430
-    /**
431
-     * Returns all the question types that should have an admin-defined max input length
432
-     * @return array
433
-     */
434
-    public function questionTypesWithMaxLength()
435
-    {
436
-        return (array) $this->question_types_with_max_lengh;
437
-    }
13
+	// constant used to indicate that the question type is COUNTRY
14
+	const QST_type_country = 'COUNTRY';
15
+
16
+	// constant used to indicate that the question type is DATE
17
+	const QST_type_date = 'DATE';
18
+
19
+	// constant used to indicate that the question type is DROPDOWN
20
+	const QST_type_dropdown = 'DROPDOWN';
21
+
22
+	// constant used to indicate that the question type is CHECKBOX
23
+	const QST_type_checkbox = 'CHECKBOX';
24
+
25
+	// constant used to indicate that the question type is RADIO_BTN
26
+	const QST_type_radio = 'RADIO_BTN';
27
+
28
+	// constant used to indicate that the question type is STATE
29
+	const QST_type_state = 'STATE';
30
+
31
+	// constant used to indicate that the question type is TEXT
32
+	const QST_type_text = 'TEXT';
33
+
34
+	// constant used to indicate that the question type is TEXTAREA
35
+	const QST_type_textarea = 'TEXTAREA';
36
+
37
+	// constant used to indicate that the question type is a TEXTAREA that allows simple html
38
+	const QST_type_html_textarea = 'HTML_TEXTAREA';
39
+
40
+	// constant used to indicate that the question type is an email input
41
+	const QST_type_email = 'EMAIL';
42
+
43
+	// constant used to indicate that the question type is a US-formatted phone number
44
+	const QST_type_us_phone = 'US_PHONE';
45
+
46
+	// constant used to indicate that the question type is an integer (whole number)
47
+	const QST_type_int = 'INTEGER';
48
+
49
+	// constant used to indicate that the question type is a decimal (float)
50
+	const QST_type_decimal = 'DECIMAL';
51
+
52
+	// constant used to indicate that the question type is a valid URL
53
+	const QST_type_url = 'URL';
54
+
55
+	// constant used to indicate that the question type is a YEAR
56
+	const QST_type_year = 'YEAR';
57
+
58
+	// constant used to indicate that the question type is a multi-select
59
+	const QST_type_multi_select = 'MULTI_SELECT';
60
+
61
+	/**
62
+	 * Question types that are interchangeable, even after answers have been provided for them.
63
+	 * Top-level keys are category slugs, next level is an array of question types. If question types
64
+	 * aren't in this array, it is assumed they AREN'T interchangeable with any other question types.
65
+	 *
66
+	 * @access protected
67
+	 * @var array $_question_type_categories {
68
+	 * @type string $text
69
+	 * @type string $single -answer-enum
70
+	 * @type string $multi -answer-enum
71
+	 *                    }
72
+	 */
73
+	protected $_question_type_categories = array();
74
+
75
+	/**
76
+	 * lists all the question types which should be allowed. Ideally, this will be extensible.
77
+	 *
78
+	 * @access protected
79
+	 * @var array $_allowed_question_types
80
+	 */
81
+	protected $_allowed_question_types = array();
82
+
83
+	/**
84
+	 * brief descriptions for all the question types
85
+	 *
86
+	 * @access protected
87
+	 * @var EEM_Question $_instance
88
+	 */
89
+	protected $_question_descriptions;
90
+
91
+
92
+	/**
93
+	 * Question types that should have an admin-defined max input length
94
+	 * @var array
95
+	 */
96
+	protected $question_types_with_max_lengh;
97
+
98
+
99
+	// private instance of the Attendee object
100
+	protected static $_instance = null;
101
+
102
+
103
+	/**
104
+	 * EEM_Question constructor.
105
+	 *
106
+	 * @param null $timezone
107
+	 */
108
+	protected function __construct($timezone = null)
109
+	{
110
+		$this->singular_item = __('Question', 'event_espresso');
111
+		$this->plural_item = __('Questions', 'event_espresso');
112
+		$this->_allowed_question_types = apply_filters(
113
+			'FHEE__EEM_Question__construct__allowed_question_types',
114
+			array(
115
+				EEM_Question::QST_type_text => __('Text', 'event_espresso'),
116
+				EEM_Question::QST_type_textarea => __('Textarea', 'event_espresso'),
117
+				EEM_Question::QST_type_checkbox => __('Checkboxes', 'event_espresso'),
118
+				EEM_Question::QST_type_radio => __('Radio Buttons', 'event_espresso'),
119
+				EEM_Question::QST_type_dropdown => __('Dropdown', 'event_espresso'),
120
+				EEM_Question::QST_type_state => __('State/Province Dropdown', 'event_espresso'),
121
+				EEM_Question::QST_type_country => __('Country Dropdown', 'event_espresso'),
122
+				EEM_Question::QST_type_date => __('Date Picker', 'event_espresso'),
123
+				EEM_Question::QST_type_html_textarea => __('HTML Textarea', 'event_espresso'),
124
+				EEM_Question::QST_type_email => __('Email', 'event_espresso'),
125
+				EEM_Question::QST_type_us_phone => __('USA - Format Phone', 'event_espresso'),
126
+				EEM_Question::QST_type_decimal => __('Number', 'event_espresso'),
127
+				EEM_Question::QST_type_int => __('Whole Number', 'event_espresso'),
128
+				EEM_Question::QST_type_url => __('URL', 'event_espresso'),
129
+				EEM_Question::QST_type_year => __('Year', 'event_espresso'),
130
+				EEM_Question::QST_type_multi_select => __('Multi Select', 'event_espresso')
131
+			)
132
+		);
133
+		$this->_question_descriptions = apply_filters(
134
+			'FHEE__EEM_Question__construct__allowed_question_types',
135
+			array(
136
+				EEM_Question::QST_type_text => __('A single line text input field', 'event_espresso'),
137
+				EEM_Question::QST_type_textarea => __('A multi line text input field', 'event_espresso'),
138
+				EEM_Question::QST_type_checkbox => __('Allows multiple preset options to be selected', 'event_espresso'),
139
+				EEM_Question::QST_type_radio => __('Allows a single preset option to be selected', 'event_espresso'),
140
+				EEM_Question::QST_type_dropdown => __('A dropdown that allows a single selection', 'event_espresso'),
141
+				EEM_Question::QST_type_state => __('A dropdown that lists states/provinces', 'event_espresso'),
142
+				EEM_Question::QST_type_country => __('A dropdown that lists countries', 'event_espresso'),
143
+				EEM_Question::QST_type_date => __('A popup calendar that allows date selections', 'event_espresso'),
144
+				EEM_Question::QST_type_html_textarea => __('A multi line text input field that allows HTML', 'event_espresso'),
145
+				EEM_Question::QST_type_email => __('A text field that must contain a valid Email address', 'event_espresso'),
146
+				EEM_Question::QST_type_us_phone => __('A text field that must contain a valid US phone number', 'event_espresso'),
147
+				EEM_Question::QST_type_decimal => __('A text field that allows number values with decimals', 'event_espresso'),
148
+				EEM_Question::QST_type_int => __('A text field that only allows whole numbers (no decimals)', 'event_espresso'),
149
+				EEM_Question::QST_type_url => __('A text field that must contain a valid URL', 'event_espresso'),
150
+				EEM_Question::QST_type_year => __('A dropdown that lists the last 100 years', 'event_espresso'),
151
+				EEM_Question::QST_type_multi_select => __('A dropdown that allows multiple selections', 'event_espresso')
152
+			)
153
+		);
154
+		$this->_question_type_categories = (array) apply_filters(
155
+			'FHEE__EEM_Question__construct__question_type_categories',
156
+			array(
157
+				'text' => array(
158
+					EEM_Question::QST_type_text,
159
+					EEM_Question::QST_type_textarea,
160
+					EEM_Question::QST_type_date,
161
+					EEM_Question::QST_type_html_textarea,
162
+					EEM_Question::QST_type_email,
163
+					EEM_Question::QST_type_us_phone,
164
+					EEM_Question::QST_type_decimal,
165
+					EEM_Question::QST_type_int,
166
+					EEM_Question::QST_type_url,
167
+					EEM_Question::QST_type_year
168
+				),
169
+				'single-answer-enum' => array(
170
+					EEM_Question::QST_type_radio,
171
+					EEM_Question::QST_type_dropdown
172
+				),
173
+				'multi-answer-enum' => array(
174
+					EEM_Question::QST_type_checkbox,
175
+					EEM_Question::QST_type_multi_select
176
+				)
177
+			)
178
+		);
179
+		$this->question_types_with_max_lengh = apply_filters(
180
+			'FHEE__EEM_Question___construct__question_types_with_max_length',
181
+			array(
182
+				EEM_Question::QST_type_text,
183
+				EEM_Question::QST_type_textarea,
184
+				EEM_Question::QST_type_html_textarea
185
+			)
186
+		);
187
+
188
+		$this->_tables = array(
189
+			'Question' => new EE_Primary_Table('esp_question', 'QST_ID')
190
+		);
191
+		$this->_fields = array(
192
+			'Question' => array(
193
+				'QST_ID' => new EE_Primary_Key_Int_Field('QST_ID', __('Question ID', 'event_espresso')),
194
+				'QST_display_text' => new EE_Post_Content_Field('QST_display_text', __('Question Text', 'event_espresso'), true, ''),
195
+				'QST_admin_label' => new EE_Plain_Text_Field('QST_admin_label', __('Question Label (admin-only)', 'event_espresso'), true, ''),
196
+				'QST_system' => new EE_Plain_Text_Field('QST_system', __('Internal string ID for question', 'event_espresso'), false, ''),
197
+				'QST_type' => new EE_Enum_Text_Field('QST_type', __('Question Type', 'event_espresso'), false, 'TEXT', $this->_allowed_question_types),
198
+				'QST_required' => new EE_Boolean_Field('QST_required', __('Required Question?', 'event_espresso'), false, false),
199
+				'QST_required_text' => new EE_Simple_HTML_Field('QST_required_text', __('Text to Display if Not Provided', 'event_espresso'), true, ''),
200
+				'QST_order' => new EE_Integer_Field('QST_order', __('Question Order', 'event_espresso'), false, 0),
201
+				'QST_admin_only' => new EE_Boolean_Field('QST_admin_only', __('Admin-Only Question?', 'event_espresso'), false, false),
202
+				'QST_max' => new EE_Infinite_Integer_Field('QST_max', __('Max Size', 'event_espresso'), false, EE_INF),
203
+				'QST_wp_user' => new EE_WP_User_Field('QST_wp_user', __('Question Creator ID', 'event_espresso'), false),
204
+				'QST_deleted' => new EE_Trashed_Flag_Field('QST_deleted', __('Flag Indicating question was deleted', 'event_espresso'), false, false)
205
+			)
206
+		);
207
+		$this->_model_relations = array(
208
+			'Question_Group' => new EE_HABTM_Relation('Question_Group_Question'),
209
+			'Question_Option' => new EE_Has_Many_Relation(),
210
+			'Answer' => new EE_Has_Many_Relation(),
211
+			'WP_User' => new EE_Belongs_To_Relation(),
212
+			// for QST_order column
213
+			'Question_Group_Question' => new EE_Has_Many_Relation()
214
+		);
215
+		// this model is generally available for reading
216
+		$this->_cap_restriction_generators[ EEM_Base::caps_read ] = new EE_Restriction_Generator_Public();
217
+		$this->_cap_restriction_generators[ EEM_Base::caps_read_admin ] = new EE_Restriction_Generator_Reg_Form('QST_system');
218
+		$this->_cap_restriction_generators[ EEM_Base::caps_edit ] = new EE_Restriction_Generator_Reg_Form('QST_system');
219
+		$this->_cap_restriction_generators[ EEM_Base::caps_delete ] = new EE_Restriction_Generator_Reg_Form('QST_system');
220
+		parent::__construct($timezone);
221
+	}
222
+
223
+	/**
224
+	 * Returns the list of allowed question types, which are normally: 'TEXT','TEXTAREA','RADIO_BTN','DROPDOWN','CHECKBOX','DATE'
225
+	 * but they can be extended
226
+	 * @return string[]
227
+	 */
228
+	public function allowed_question_types()
229
+	{
230
+		return $this->_allowed_question_types;
231
+	}
232
+
233
+	/**
234
+	 * Gets all the question types in the same category
235
+	 * @param string $question_type one of EEM_Question::allowed_question_types(
236
+	 * @return string[] like EEM_Question::allowed_question_types()
237
+	 */
238
+	public function question_types_in_same_category($question_type)
239
+	{
240
+		$question_types = array($question_type);
241
+		foreach ($this->_question_type_categories as $category => $question_types_in_category) {
242
+			if (in_array($question_type, $question_types_in_category)) {
243
+				$question_types = $question_types_in_category;
244
+				break;
245
+			}
246
+		}
247
+
248
+		return array_intersect_key($this->allowed_question_types(), array_flip($question_types));
249
+	}
250
+
251
+	/**
252
+	 * Determines if the given question type is in the given question type category
253
+	 * @param string $question_type one of EEM_Question::allowed_question_types()
254
+	 * @param string $category one of the top-level keys of EEM_Question::question_type_categories()
255
+	 * @return boolean
256
+	 */
257
+	public function question_type_is_in_category($question_type, $category)
258
+	{
259
+		if (!isset($this->_question_type_categories[ $category ])) {
260
+			return false;
261
+		}
262
+		return in_array($question_type, $this->_question_type_categories[ $category ]);
263
+	}
264
+
265
+
266
+	/**
267
+	 * Returns all the question types in the given category
268
+	 * @param string $category
269
+	 * @return array|mixed
270
+	 */
271
+	public function question_types_in_category($category)
272
+	{
273
+		if (isset($this->_question_type_categories[ $category ])) {
274
+			return $this->_question_type_categories[ $category ];
275
+		}
276
+		return array();
277
+	}
278
+
279
+
280
+	/**
281
+	 * Returns all the question types that should have question options
282
+	 * @return array
283
+	 */
284
+	public function question_types_with_options()
285
+	{
286
+		return array_merge(
287
+			$this->question_types_in_category('single-answer-enum'),
288
+			$this->question_types_in_category('multi-answer-enum')
289
+		);
290
+	}
291
+
292
+	/**
293
+	 * Returns the question type categories 2d array
294
+	 * @return array see EEM_Question::_question_type_categories
295
+	 */
296
+	public function question_type_categories()
297
+	{
298
+		return $this->_question_type_categories;
299
+	}
300
+
301
+	/**
302
+	 * Returns an array of all the QST_system values that can be allowed in the system question group
303
+	 * identified by $system_question_group_id
304
+	 * @param string $system_question_group_id QSG_system
305
+	 * @return array of system question names (QST_system)
306
+	 */
307
+	public function allowed_system_questions_in_system_question_group($system_question_group_id)
308
+	{
309
+		$question_system_ids = array();
310
+		switch ($system_question_group_id) {
311
+			case EEM_Question_Group::system_personal:
312
+				$question_system_ids = array(
313
+					EEM_Attendee::system_question_fname,
314
+					EEM_Attendee::system_question_lname,
315
+					EEM_Attendee::system_question_email,
316
+					EEM_Attendee::system_question_phone
317
+				);
318
+				break;
319
+			case EEM_Question_Group::system_address:
320
+				$question_system_ids = array(
321
+					EEM_Attendee::system_question_address,
322
+					EEM_Attendee::system_question_address2,
323
+					EEM_Attendee::system_question_city,
324
+					EEM_Attendee::system_question_state,
325
+					EEM_Attendee::system_question_country,
326
+					EEM_Attendee::system_question_zip,
327
+					EEM_Attendee::system_question_phone
328
+				);
329
+				break;
330
+		}
331
+		return apply_filters('FHEE__EEM_Question__system_questions_allowed_in_system_question_group__return', $question_system_ids, $system_question_group_id);
332
+	}
333
+
334
+	/**
335
+	 * Returns an array of all the QST_system values that are required in the system question group
336
+	 * identified by $system_question_group_id
337
+	 * @param string $system_question_group_id QSG_system
338
+	 * @return array of system question names (QST_system)
339
+	 */
340
+	public function required_system_questions_in_system_question_group($system_question_group_id)
341
+	{
342
+		$question_system_ids = null;
343
+		switch ($system_question_group_id) {
344
+			case EEM_Question_Group::system_personal:
345
+				$question_system_ids = array(
346
+					EEM_Attendee::system_question_fname,
347
+					EEM_Attendee::system_question_email,
348
+				);
349
+				break;
350
+			default:
351
+				$question_system_ids = array();
352
+		}
353
+		return apply_filters('FHEE__EEM_Question__system_questions_required_in_system_question_group', $question_system_ids, $system_question_group_id);
354
+	}
355
+
356
+
357
+	/**
358
+	 * Gets an array for converting between QST_system and QST_IDs for system questions. Eg, if you want to know
359
+	 * which system question QST_ID corresponds to the QST_system 'city', use EEM_Question::instance()->get_Question_ID_from_system_string('city');
360
+	 * @param $QST_system
361
+	 * @return int of QST_ID for the question that corresponds to that QST_system
362
+	 */
363
+	public function get_Question_ID_from_system_string($QST_system)
364
+	{
365
+		return $this->get_var(array(array('QST_system' => $QST_system)));
366
+	}
367
+
368
+
369
+	/**
370
+	 * searches the db for the question with the latest question order and returns that value.
371
+	 * @access public
372
+	 * @return int
373
+	 */
374
+	public function get_latest_question_order()
375
+	{
376
+		$columns_to_select = array(
377
+			'max_order' => array("MAX(QST_order)", "%d")
378
+		);
379
+		$max = $this->_get_all_wpdb_results(array(), ARRAY_A, $columns_to_select);
380
+		return isset($max[0], $max[0]['max_order']) ? $max[0]['max_order'] : 0;
381
+	}
382
+
383
+	/**
384
+	 * Returns an array where keys are system question QST_system values,
385
+	 * and values are the highest question max the admin can set on the question
386
+	 * (aka the "max max"; eg, a site admin can change the zip question to have a max
387
+	 * of 5, but no larger than 12)
388
+	 * @return array
389
+	 */
390
+	public function system_question_maxes()
391
+	{
392
+		return array(
393
+			'fname' => 45,
394
+			'lname' => 45,
395
+			'address' => 255,
396
+			'address2' => 255,
397
+			'city' => 45,
398
+			'zip' => 12,
399
+			'email' => 255,
400
+			'phone' => 45,
401
+		);
402
+	}
403
+
404
+	/**
405
+	 * Given a QST_system value, gets the question's largest allowable max input.
406
+	 * @see Registration_Form_Admin_Page::system_question_maxes()
407
+	 * @param string $system_question_value
408
+	 * @return int|float
409
+	 */
410
+	public function absolute_max_for_system_question($system_question_value)
411
+	{
412
+		$maxes = $this->system_question_maxes();
413
+		if (isset($maxes[ $system_question_value ])) {
414
+			return $maxes[ $system_question_value ];
415
+		} else {
416
+			return EE_INF;
417
+		}
418
+	}
419
+
420
+
421
+	/**
422
+	 * @return array
423
+	 */
424
+	public function question_descriptions()
425
+	{
426
+		return $this->_question_descriptions;
427
+	}
428
+
429
+
430
+	/**
431
+	 * Returns all the question types that should have an admin-defined max input length
432
+	 * @return array
433
+	 */
434
+	public function questionTypesWithMaxLength()
435
+	{
436
+		return (array) $this->question_types_with_max_lengh;
437
+	}
438 438
 }
Please login to merge, or discard this patch.
Spacing   +10 added lines, -10 removed lines patch added patch discarded remove patch
@@ -213,10 +213,10 @@  discard block
 block discarded – undo
213 213
             'Question_Group_Question' => new EE_Has_Many_Relation()
214 214
         );
215 215
         // this model is generally available for reading
216
-        $this->_cap_restriction_generators[ EEM_Base::caps_read ] = new EE_Restriction_Generator_Public();
217
-        $this->_cap_restriction_generators[ EEM_Base::caps_read_admin ] = new EE_Restriction_Generator_Reg_Form('QST_system');
218
-        $this->_cap_restriction_generators[ EEM_Base::caps_edit ] = new EE_Restriction_Generator_Reg_Form('QST_system');
219
-        $this->_cap_restriction_generators[ EEM_Base::caps_delete ] = new EE_Restriction_Generator_Reg_Form('QST_system');
216
+        $this->_cap_restriction_generators[EEM_Base::caps_read] = new EE_Restriction_Generator_Public();
217
+        $this->_cap_restriction_generators[EEM_Base::caps_read_admin] = new EE_Restriction_Generator_Reg_Form('QST_system');
218
+        $this->_cap_restriction_generators[EEM_Base::caps_edit] = new EE_Restriction_Generator_Reg_Form('QST_system');
219
+        $this->_cap_restriction_generators[EEM_Base::caps_delete] = new EE_Restriction_Generator_Reg_Form('QST_system');
220 220
         parent::__construct($timezone);
221 221
     }
222 222
 
@@ -256,10 +256,10 @@  discard block
 block discarded – undo
256 256
      */
257 257
     public function question_type_is_in_category($question_type, $category)
258 258
     {
259
-        if (!isset($this->_question_type_categories[ $category ])) {
259
+        if ( ! isset($this->_question_type_categories[$category])) {
260 260
             return false;
261 261
         }
262
-        return in_array($question_type, $this->_question_type_categories[ $category ]);
262
+        return in_array($question_type, $this->_question_type_categories[$category]);
263 263
     }
264 264
 
265 265
 
@@ -270,8 +270,8 @@  discard block
 block discarded – undo
270 270
      */
271 271
     public function question_types_in_category($category)
272 272
     {
273
-        if (isset($this->_question_type_categories[ $category ])) {
274
-            return $this->_question_type_categories[ $category ];
273
+        if (isset($this->_question_type_categories[$category])) {
274
+            return $this->_question_type_categories[$category];
275 275
         }
276 276
         return array();
277 277
     }
@@ -410,8 +410,8 @@  discard block
 block discarded – undo
410 410
     public function absolute_max_for_system_question($system_question_value)
411 411
     {
412 412
         $maxes = $this->system_question_maxes();
413
-        if (isset($maxes[ $system_question_value ])) {
414
-            return $maxes[ $system_question_value ];
413
+        if (isset($maxes[$system_question_value])) {
414
+            return $maxes[$system_question_value];
415 415
         } else {
416 416
             return EE_INF;
417 417
         }
Please login to merge, or discard this patch.