Test Failed
Push — issue/2900 ( f2122b )
by Ravinder
07:32
created

misc-functions.php ➔ give_get_meta()   A

Complexity

Conditions 4
Paths 2

Size

Total Lines 14
Code Lines 5

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 4
eloc 5
nc 2
nop 4
dl 0
loc 14
rs 9.2
c 0
b 0
f 0
1
<?php
2
/**
3
 * Misc Functions
4
 *
5
 * @package     Give
6
 * @subpackage  Functions
7
 * @copyright   Copyright (c) 2016, WordImpress
8
 * @license     https://opensource.org/licenses/gpl-license GNU Public License
9
 * @since       1.0
10
 */
11
12
// Exit if accessed directly.
13
if ( ! defined( 'ABSPATH' ) ) {
14
	exit;
15
}
16
17
/**
18
 * Is Test Mode Enabled.
19
 *
20
 * @since 1.0
21
 *
22
 * @return bool $ret True if return mode is enabled, false otherwise
23
 */
24
function give_is_test_mode() {
25
26
	$ret = give_is_setting_enabled( give_get_option( 'test_mode' ) );
27
28
	return (bool) apply_filters( 'give_is_test_mode', $ret );
29
30
}
31
32
/**
33
 * Get the current page URL.
34
 *
35
 * @since 1.0
36
 * @return string $current_url Current page URL.
37
 */
38
function give_get_current_page_url() {
39
40
	global $wp;
41
42
	if ( get_option( 'permalink_structure' ) ) {
43
		$base = trailingslashit( home_url( $wp->request ) );
44
	} else {
45
		$base = add_query_arg( $wp->query_string, '', trailingslashit( home_url( $wp->request ) ) );
46
		$base = remove_query_arg( array( 'post_type', 'name' ), $base );
47
	}
48
49
	$scheme      = is_ssl() ? 'https' : 'http';
50
	$current_uri = set_url_scheme( $base, $scheme );
51
52
	if ( is_front_page() ) {
53
		$current_uri = home_url( '/' );
54
	}
55
56
	/**
57
	 * Filter the current page url
58
	 *
59
	 * @since 1.0
60
	 *
61
	 * @param string $current_uri
62
	 */
63
	return apply_filters( 'give_get_current_page_url', $current_uri );
64
65
}
66
67
68
/**
69
 * Verify credit card numbers live?
70
 *
71
 * @since 1.0
72
 *
73
 * @return bool $ret True is verify credit cards is live
74
 */
75
function give_is_cc_verify_enabled() {
76
77
	$ret = true;
78
79
	/**
80
	 * Enable if use a single gateway other than PayPal or Manual. We have to assume it accepts credit cards.
81
	 * Enable if using more than one gateway if they are not both PayPal and manual, again assuming credit card usage.
82
	 */
83
	$gateways = give_get_enabled_payment_gateways();
84
85
	if ( count( $gateways ) == 1 && ! isset( $gateways['paypal'] ) && ! isset( $gateways['manual'] ) ) {
0 ignored issues
show
introduced by
Found "== 1". Use Yoda Condition checks, you must
Loading history...
86
		$ret = true;
87
	} elseif ( count( $gateways ) == 1 ) {
0 ignored issues
show
introduced by
Found "== 1". Use Yoda Condition checks, you must
Loading history...
88
		$ret = false;
89
	} elseif ( count( $gateways ) == 2 && isset( $gateways['paypal'] ) && isset( $gateways['manual'] ) ) {
0 ignored issues
show
introduced by
Found "== 2". Use Yoda Condition checks, you must
Loading history...
90
		$ret = false;
91
	}
92
93
	/**
94
	 * Fire the filter
95
	 *
96
	 * @since 1.0
97
	 *
98
	 * @param bool $ret
99
	 */
100
	return (bool) apply_filters( 'give_is_cc_verify_enabled', $ret );
101
}
102
103
/**
104
 * Retrieve timezone.
105
 *
106
 * @since 1.0
107
 * @return string $timezone The timezone ID.
108
 */
109
function give_get_timezone_id() {
110
111
	// if site timezone string exists, return it.
112
	if ( $timezone = get_option( 'timezone_string' ) ) {
113
		return $timezone;
114
	}
115
116
	// get UTC offset, if it isn't set return UTC.
117
	if ( ! ( $utc_offset = 3600 * get_option( 'gmt_offset', 0 ) ) ) {
118
		return 'UTC';
119
	}
120
121
	// attempt to guess the timezone string from the UTC offset.
122
	$timezone = timezone_name_from_abbr( '', $utc_offset );
123
124
	// last try, guess timezone string manually.
125
	if ( $timezone === false ) {
0 ignored issues
show
introduced by
Found "=== false". Use Yoda Condition checks, you must
Loading history...
126
127
		$is_dst = date( 'I' );
128
129
		foreach ( timezone_abbreviations_list() as $abbr ) {
130
			foreach ( $abbr as $city ) {
131
				if ( $city['dst'] == $is_dst && $city['offset'] == $utc_offset ) {
132
					return $city['timezone_id'];
133
				}
134
			}
135
		}
136
	}
137
138
	// Fallback.
139
	return 'UTC';
140
}
141
142
143
/**
144
 * Get User IP
145
 *
146
 * Returns the IP address of the current visitor
147
 *
148
 * @since 1.0
149
 * @return string $ip User's IP address
150
 */
151
function give_get_ip() {
152
153
	$ip = '127.0.0.1';
154
155
	if ( ! empty( $_SERVER['HTTP_CLIENT_IP'] ) ) {
156
		// check ip from share internet
157
		$ip = $_SERVER['HTTP_CLIENT_IP'];
0 ignored issues
show
introduced by
Detected usage of a non-sanitized input variable: $_SERVER
Loading history...
158
	} elseif ( ! empty( $_SERVER['HTTP_X_FORWARDED_FOR'] ) ) {
159
		// to check ip is pass from proxy
160
		$ip = $_SERVER['HTTP_X_FORWARDED_FOR'];
0 ignored issues
show
introduced by
Detected usage of a non-sanitized input variable: $_SERVER
Loading history...
161
	} elseif ( ! empty( $_SERVER['REMOTE_ADDR'] ) ) {
0 ignored issues
show
introduced by
Due to using Batcache, server side based client related logic will not work, use JS instead.
Loading history...
162
		$ip = $_SERVER['REMOTE_ADDR'];
0 ignored issues
show
introduced by
Detected usage of a non-sanitized input variable: $_SERVER
Loading history...
introduced by
Due to using Batcache, server side based client related logic will not work, use JS instead.
Loading history...
163
	}
164
165
	/**
166
	 * Filter the IP
167
	 *
168
	 * @since 1.0
169
	 */
170
	$ip = apply_filters( 'give_get_ip', $ip );
171
172
	// Filter empty values.
173
	if ( false !== strpos( $ip, ',' ) ) {
174
		$ip = give_clean( explode( ',', $ip ) );
175
		$ip = array_filter( $ip );
176
		$ip = implode( ',', $ip );
177
	} else {
178
		$ip = give_clean( $ip );
179
	}
180
181
	return $ip;
182
}
183
184
185
/**
186
 * Store Donation Data in Sessions
187
 *
188
 * Used for storing info about donation
189
 *
190
 * @since 1.0
191
 *
192
 * @param $purchase_data
193
 *
194
 * @uses  Give()->session->set()
195
 */
196
function give_set_purchase_session( $purchase_data = array() ) {
197
	Give()->session->set( 'give_purchase', $purchase_data );
198
	Give()->session->set( 'give_email', $purchase_data['user_email'] );
199
}
200
201
/**
202
 * Retrieve Donation Data from Session
203
 *
204
 * Used for retrieving info about donation
205
 * after completing a donation
206
 *
207
 * @since 1.0
208
 * @uses  Give()->session->get()
209
 * @return mixed array | false
210
 */
211
function give_get_purchase_session() {
212
	return Give()->session->get( 'give_purchase' );
213
}
214
215
/**
216
 * Retrieve Payment Key of the Receipt Access Session.
217
 *
218
 * @since 1.8.17
219
 *
220
 * @return array|string
221
 */
222
function give_get_receipt_session() {
223
	return Give()->session->get( 'receipt_access' );
224
}
225
226
/**
227
 * Retrieve Payment Key of the History Access Session.
228
 *
229
 * @since 1.8.17
230
 *
231
 * @return array|string
232
 */
233
function give_get_history_session() {
234
	return (bool) Give()->session->get( 'history_access' );
235
}
236
237
/**
238
 * Generate Item Title for Payment Gateway.
239
 *
240
 * @param array $payment_data Payment Data.
241
 *
242
 * @since 1.8.14
243
 *
244
 * @return string By default, the name of the form. Then the price level text if any is found.
245
 */
246
function give_payment_gateway_item_title( $payment_data ) {
247
248
	$form_id   = intval( $payment_data['post_data']['give-form-id'] );
249
	$item_name = isset( $payment_data['post_data']['give-form-title'] ) ? $payment_data['post_data']['give-form-title'] : '';
250
	$price_id  = isset( $payment_data['post_data']['give-price-id'] ) ? $payment_data['post_data']['give-price-id'] : '';
251
0 ignored issues
show
Coding Style introduced by
Functions must not contain multiple empty lines in a row; found 2 empty lines
Loading history...
252
253
	// Verify has variable prices.
254
	if ( give_has_variable_prices( $form_id ) && ! empty( $price_id ) ) {
255
256
		$item_price_level_text = give_get_price_option_name( $form_id, $price_id, 0, false );
257
258
		/**
259
		 * Output donation level text if:
260
		 *
261
		 * 1. It's not a custom amount
262
		 * 2. The level field has actual text and isn't the amount (which is already displayed on the receipt).
263
		 */
264
		if ( 'custom' !== $price_id && ! empty( $item_price_level_text ) ) {
265
			// Matches a donation level - append level text.
266
			$item_name .= ' - ' . $item_price_level_text;
267
		}
268
	}
269
270
	/**
271
	 * Filter the Item Title of Payment Gateway.
272
	 *
273
	 * @param string $item_name    Item Title of Payment Gateway.
274
	 * @param int    $form_id      Donation Form ID.
275
	 * @param array  $payment_data Payment Data.
276
	 *
277
	 * @since 1.8.14
278
	 *
279
	 * @return string
280
	 */
281
	return apply_filters( 'give_payment_gateway_item_title', $item_name, $form_id, $payment_data );
282
}
283
284
/**
285
 * Get Donation Summary
286
 *
287
 * Creates a donation summary for payment gateways from the donation data before the payment is created in the database.
288
 *
289
 * @since       1.8.12
290
 *
291
 * @param array $donation_data
292
 * @param bool  $name_and_email
293
 * @param int   $length
294
 *
295
 * @return string
296
 */
297
function give_payment_gateway_donation_summary( $donation_data, $name_and_email = true, $length = 255 ) {
298
299
	$form_id  = isset( $donation_data['post_data']['give-form-id'] ) ? $donation_data['post_data']['give-form-id'] : '';
300
	$price_id = isset( $donation_data['post_data']['give-price-id'] ) ? $donation_data['post_data']['give-price-id'] : '';
301
302
	// Form title.
303
	$summary = ( ! empty( $donation_data['post_data']['give-form-title'] ) ? $donation_data['post_data']['give-form-title'] : ( ! empty( $form_id ) ? wp_sprintf( __( 'Donation Form ID: %d', 'give' ), $form_id ) : __( 'Untitled donation form', 'give' ) ) );
304
305
	// Form multilevel if applicable.
306
	if ( ! empty( $price_id ) && 'custom' !== $price_id ) {
307
		$summary .= ': ' . give_get_price_option_name( $form_id, $donation_data['post_data']['give-price-id'] );
308
	}
309
310
	// Add Donor's name + email if requested.
311
	if ( $name_and_email ) {
312
313
		// First name
314 View Code Duplication
		if ( isset( $donation_data['user_info']['first_name'] ) && ! empty( $donation_data['user_info']['first_name'] ) ) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
315
			$summary .= ' - ' . $donation_data['user_info']['first_name'];
316
		}
317
318 View Code Duplication
		if ( isset( $donation_data['user_info']['last_name'] ) && ! empty( $donation_data['user_info']['last_name'] ) ) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
319
			$summary .= ' ' . $donation_data['user_info']['last_name'];
320
		}
321
322
		$summary .= ' (' . $donation_data['user_email'] . ')';
323
	}
324
325
	// Cut the length
326
	$summary = substr( $summary, 0, $length );
327
328
	return apply_filters( 'give_payment_gateway_donation_summary', $summary );
329
}
330
331
332
/**
333
 * Get user host
334
 *
335
 * Returns the webhost this site is using if possible
336
 *
337
 * @since 1.0
338
 * @return string $host if detected, false otherwise
339
 */
340
function give_get_host() {
341
	$host = false;
0 ignored issues
show
Unused Code introduced by
$host is not used, you could remove the assignment.

This check looks for variable assignements that are either overwritten by other assignments or where the variable is not used subsequently.

$myVar = 'Value';
$higher = false;

if (rand(1, 6) > 3) {
    $higher = true;
} else {
    $higher = false;
}

Both the $myVar assignment in line 1 and the $higher assignment in line 2 are dead. The first because $myVar is never used and the second because $higher is always overwritten for every possible time line.

Loading history...
342
343
	if ( defined( 'WPE_APIKEY' ) ) {
344
		$host = 'WP Engine';
345
	} elseif ( defined( 'PAGELYBIN' ) ) {
346
		$host = 'Pagely';
347
	} elseif ( DB_HOST == 'localhost:/tmp/mysql5.sock' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
348
		$host = 'ICDSoft';
349
	} elseif ( DB_HOST == 'mysqlv5' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
350
		$host = 'NetworkSolutions';
351
	} elseif ( strpos( DB_HOST, 'ipagemysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
352
		$host = 'iPage';
353
	} elseif ( strpos( DB_HOST, 'ipowermysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
354
		$host = 'IPower';
355
	} elseif ( strpos( DB_HOST, '.gridserver.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
356
		$host = 'MediaTemple Grid';
357
	} elseif ( strpos( DB_HOST, '.pair.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
358
		$host = 'pair Networks';
359
	} elseif ( strpos( DB_HOST, '.stabletransit.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
360
		$host = 'Rackspace Cloud';
361
	} elseif ( strpos( DB_HOST, '.sysfix.eu' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
362
		$host = 'SysFix.eu Power Hosting';
363
	} elseif ( strpos( $_SERVER['SERVER_NAME'], 'Flywheel' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
364
		$host = 'Flywheel';
365
	} else {
366
		// Adding a general fallback for data gathering
367
		$host = 'DBH: ' . DB_HOST . ', SRV: ' . $_SERVER['SERVER_NAME'];
0 ignored issues
show
introduced by
Detected usage of a non-sanitized input variable: $_SERVER
Loading history...
368
	}
369
370
	return $host;
371
}
372
373
374
/**
375
 * Check site host
376
 *
377
 * @since 1.0
378
 *
379
 * @param bool /string $host The host to check
380
 *
381
 * @return bool true if host matches, false if not
382
 */
383
function give_is_host( $host = false ) {
384
385
	$return = false;
386
387
	if ( $host ) {
388
		$host = str_replace( ' ', '', strtolower( $host ) );
389
390
		switch ( $host ) {
391
			case 'wpengine':
392
				if ( defined( 'WPE_APIKEY' ) ) {
393
					$return = true;
394
				}
395
				break;
396
			case 'pagely':
397
				if ( defined( 'PAGELYBIN' ) ) {
398
					$return = true;
399
				}
400
				break;
401
			case 'icdsoft':
402
				if ( DB_HOST == 'localhost:/tmp/mysql5.sock' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
403
					$return = true;
404
				}
405
				break;
406
			case 'networksolutions':
407
				if ( DB_HOST == 'mysqlv5' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
408
					$return = true;
409
				}
410
				break;
411
			case 'ipage':
412
				if ( strpos( DB_HOST, 'ipagemysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
413
					$return = true;
414
				}
415
				break;
416
			case 'ipower':
417
				if ( strpos( DB_HOST, 'ipowermysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
418
					$return = true;
419
				}
420
				break;
421
			case 'mediatemplegrid':
422
				if ( strpos( DB_HOST, '.gridserver.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
423
					$return = true;
424
				}
425
				break;
426
			case 'pairnetworks':
427
				if ( strpos( DB_HOST, '.pair.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
428
					$return = true;
429
				}
430
				break;
431
			case 'rackspacecloud':
432
				if ( strpos( DB_HOST, '.stabletransit.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
433
					$return = true;
434
				}
435
				break;
436
			case 'sysfix.eu':
437
			case 'sysfix.eupowerhosting':
438
				if ( strpos( DB_HOST, '.sysfix.eu' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
439
					$return = true;
440
				}
441
				break;
442
			case 'flywheel':
443
				if ( strpos( $_SERVER['SERVER_NAME'], 'Flywheel' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
444
					$return = true;
445
				}
446
				break;
447
			default:
448
				$return = false;
449
		}// End switch().
450
	}// End if().
451
452
	return $return;
453
}
454
455
/**
456
 * Marks a function as deprecated and informs when it has been used.
457
 *
458
 * There is a hook give_deprecated_function_run that will be called that can be used
459
 * to get the backtrace up to what file and function called the deprecated
460
 * function.
461
 *
462
 * The current behavior is to trigger a user error if WP_DEBUG is true.
463
 *
464
 * This function is to be used in every function that is deprecated.
465
 *
466
 * @uses do_action() Calls 'give_deprecated_function_run' and passes the function name, what to use instead,
467
 *   and the version the function was deprecated in.
468
 * @uses apply_filters() Calls 'give_deprecated_function_trigger_error' and expects boolean value of true to do
469
 *   trigger or false to not trigger error.
470
 *
471
 * @param string $function    The function that was called.
472
 * @param string $version     The plugin version that deprecated the function.
473
 * @param string $replacement Optional. The function that should have been called.
474
 * @param array  $backtrace   Optional. Contains stack backtrace of deprecated function.
475
 */
476
function _give_deprecated_function( $function, $version, $replacement = null, $backtrace = null ) {
477
478
	/**
479
	 * Fires while give deprecated function call occurs.
480
	 *
481
	 * Allow you to hook to deprecated function call.
482
	 *
483
	 * @since 1.0
484
	 *
485
	 * @param string $function    The function that was called.
486
	 * @param string $replacement Optional. The function that should have been called.
487
	 * @param string $version     The plugin version that deprecated the function.
488
	 */
489
	do_action( 'give_deprecated_function_run', $function, $replacement, $version );
490
491
	$show_errors = current_user_can( 'manage_options' );
492
493
	// Allow plugin to filter the output error trigger.
494
	if ( WP_DEBUG && apply_filters( 'give_deprecated_function_trigger_error', $show_errors ) ) {
495
		if ( ! is_null( $replacement ) ) {
496
			trigger_error( sprintf( __( '%1$s is <strong>deprecated</strong> since Give version %2$s! Use %3$s instead.', 'give' ), $function, $version, $replacement ) );
497
			trigger_error( print_r( $backtrace, 1 ) ); // Limited to previous 1028 characters, but since we only need to move back 1 in stack that should be fine.
0 ignored issues
show
introduced by
The use of function print_r() is discouraged
Loading history...
498
			// Alternatively we could dump this to a file.
499
		} else {
500
			trigger_error( sprintf( __( '%1$s is <strong>deprecated</strong> since Give version %2$s with no alternative available.', 'give' ), $function, $version ) );
501
			trigger_error( print_r( $backtrace, 1 ) );// Limited to previous 1028 characters, but since we only need to move back 1 in stack that should be fine.
0 ignored issues
show
introduced by
The use of function print_r() is discouraged
Loading history...
502
			// Alternatively we could dump this to a file.
503
		}
504
	}
505
}
506
507
/**
508
 * Give Get Admin ID
509
 *
510
 * Helper function to return the ID of the post for admin usage
511
 *
512
 * @return string $post_id
513
 */
514
function give_get_admin_post_id() {
515
	$post_id = isset( $_REQUEST['post'] ) ? absint( $_REQUEST['post'] ) : null;
0 ignored issues
show
introduced by
Detected access of super global var $_REQUEST, probably need manual inspection.
Loading history...
516
517
	$post_id = ! empty( $post_id ) ? $post_id : ( isset( $_REQUEST['post_id'] ) ? absint( $_REQUEST['post_id'] ) : null );
0 ignored issues
show
introduced by
Detected access of super global var $_REQUEST, probably need manual inspection.
Loading history...
518
519
	$post_id = ! empty( $post_id ) ? $post_id : ( isset( $_REQUEST['post_ID'] ) ? absint( $_REQUEST['post_ID'] ) : null );
0 ignored issues
show
introduced by
Detected access of super global var $_REQUEST, probably need manual inspection.
Loading history...
520
521
	return $post_id;
522
}
523
524
/**
525
 * Get PHP Arg Separator Output
526
 *
527
 * @since 1.0
528
 * @return string Arg separator output
529
 */
530
function give_get_php_arg_separator_output() {
531
	return ini_get( 'arg_separator.output' );
532
}
533
534
535
/**
536
 * Month Num To Name
537
 *
538
 * Takes a month number and returns the name three letter name of it.
539
 *
540
 * @since 1.0
541
 *
542
 * @param int $n
543
 *
544
 * @return string Short month name
545
 */
546
function give_month_num_to_name( $n ) {
547
	$timestamp = mktime( 0, 0, 0, $n, 1, 2005 );
548
549
	return date_i18n( 'M', $timestamp );
550
}
551
552
553
/**
554
 * Checks whether function is disabled.
555
 *
556
 * @since 1.0
557
 *
558
 * @param string $function Name of the function.
559
 *
560
 * @return bool Whether or not function is disabled.
561
 */
562
function give_is_func_disabled( $function ) {
563
	$disabled = explode( ',', ini_get( 'disable_functions' ) );
564
565
	return in_array( $function, $disabled );
566
}
567
568
569
/**
570
 * Give Newsletter
571
 *
572
 * Returns the main Give newsletter form
573
 */
574
function give_get_newsletter() {
575
	?>
576
577
	<p class="newsletter-intro"><?php esc_html_e( 'Be sure to sign up for the Give newsletter below to stay informed of important updates and news.', 'give' ); ?></p>
578
579
	<div class="give-newsletter-form-wrap">
580
581
		<form action="//givewp.us3.list-manage.com/subscribe/post?u=3ccb75d68bda4381e2f45794c&amp;id=12a081aa13"
582
		      method="post" id="mc-embedded-subscribe-form" name="mc-embedded-subscribe-form" class="validate"
583
		      target="_blank" novalidate>
584
			<div class="give-newsletter-confirmation">
585
				<p><?php esc_html_e( 'Thanks for Subscribing!', 'give' ); ?> :)</p>
586
			</div>
587
588
			<table class="form-table give-newsletter-form">
589
				<tr valign="middle">
590
					<td>
591
						<label for="mce-EMAIL"
592
						       class="screen-reader-text"><?php esc_html_e( 'Email Address (required)', 'give' ); ?></label>
593
						<input type="email" name="EMAIL" id="mce-EMAIL"
594
						       placeholder="<?php esc_attr_e( 'Email Address (required)', 'give' ); ?>"
595
						       class="required email" value="">
596
					</td>
597
					<td>
598
						<label for="mce-FNAME"
599
						       class="screen-reader-text"><?php esc_html_e( 'First Name', 'give' ); ?></label>
600
						<input type="text" name="FNAME" id="mce-FNAME"
601
						       placeholder="<?php esc_attr_e( 'First Name', 'give' ); ?>" class="" value="">
602
					</td>
603
					<td>
604
						<label for="mce-LNAME"
605
						       class="screen-reader-text"><?php esc_html_e( 'Last Name', 'give' ); ?></label>
606
						<input type="text" name="LNAME" id="mce-LNAME"
607
						       placeholder="<?php esc_attr_e( 'Last Name', 'give' ); ?>" class="" value="">
608
					</td>
609
					<td>
610
						<input type="submit" name="subscribe" id="mc-embedded-subscribe" class="button"
611
						       value="<?php esc_attr_e( 'Subscribe', 'give' ); ?>">
612
					</td>
613
				</tr>
614
			</table>
615
		</form>
616
617
		<div style="position: absolute; left: -5000px;">
618
			<input type="text" name="b_3ccb75d68bda4381e2f45794c_12a081aa13" tabindex="-1" value="">
619
		</div>
620
621
	</div>
622
623
	<script type='text/javascript' src='//s3.amazonaws.com/downloads.mailchimp.com/js/mc-validate.js'></script>
0 ignored issues
show
introduced by
Scripts must be registered/enqueued via wp_enqueue_script
Loading history...
624
	<script type='text/javascript'>(function ($) {
625
			window.fnames = new Array();
626
			window.ftypes = new Array();
627
			fnames[0] = 'EMAIL';
628
			ftypes[0] = 'email';
629
			fnames[1] = 'FNAME';
630
			ftypes[1] = 'text';
631
			fnames[2] = 'LNAME';
632
			ftypes[2] = 'text';
633
634
			//Successful submission
635
			$('form[name="mc-embedded-subscribe-form"]').on('submit', function () {
636
637
				var email_field = $(this).find('#mce-EMAIL').val();
638
				if (!email_field) {
639
					return false;
640
				}
641
				$(this).find('.give-newsletter-confirmation').show().delay(5000).slideUp();
642
				$(this).find('.give-newsletter-form').hide();
643
644
			});
645
646
		}(jQuery));
647
		var $mcj = jQuery.noConflict(true);
648
649
650
	</script>
651
	<!--End mc_embed_signup-->
652
653
	<?php
654
}
655
656
657
/**
658
 * Create SVG library function
659
 *
660
 * @param string $icon
661
 *
662
 * @return string
663
 */
664
function give_svg_icons( $icon ) {
665
666
	// Store your SVGs in an associative array
667
	$svgs = array(
668
		'microphone'    => '',
669
		'alert'         => '',
670
		'placemark'     => '',
671
		'give_grey'     => '',
672
		'give_cpt_icon' => '',
673
	);
674
675
	// Return the chosen icon's SVG string
676
	return $svgs[ $icon ];
677
}
678
679
/**
680
 * Modify Admin Nav Menu Label
681
 *
682
 * @since 1.3
683
 *
684
 * @param object $post_type The current object to add a menu items meta box for.
685
 *
686
 * @return mixed
687
 */
688
function modify_nav_menu_meta_box_object( $post_type ) {
689
	if ( isset( $post_type->name ) && $post_type->name == 'give_forms' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
690
		$post_type->labels->name = esc_html__( 'Donation Forms', 'give' );
691
	}
692
693
	return $post_type;
694
}
695
696
add_filter( 'nav_menu_meta_box_object', 'modify_nav_menu_meta_box_object' );
697
698
/**
699
 * Show Donation Forms Post Type in Appearance > Menus by default on fresh install.
700
 *
701
 * @since 1.8.14
702
 *
703
 * @todo  Remove this, when WordPress Core ticket is resolved (https://core.trac.wordpress.org/ticket/16828).
704
 *
705
 * @return bool
706
 */
707
function give_donation_metabox_menu() {
708
709
	// Get Current Screen.
710
	$screen = get_current_screen();
711
712
	// Proceed, if current screen is navigation menus.
713
	if ( 'nav-menus' === $screen->id && give_is_setting_enabled( give_get_option( 'forms_singular' ) ) && ! get_user_option( 'give_is_donation_forms_menu_updated' ) ) {
714
715
		// Return false, if it fails to retrieve hidden meta box list and is not admin.
716
		if ( ! is_admin() || ( ! $hidden_meta_boxes = get_user_option( 'metaboxhidden_nav-menus' ) ) ) {
717
			return false;
718
		}
719
720
		// Return false, In case, we don't find 'Donation Form' in hidden meta box list.
721
		if ( ! in_array( 'add-post-type-give_forms', $hidden_meta_boxes, true ) ) {
722
			return false;
723
		}
724
725
		// Exclude 'Donation Form' value from hidden meta box's list.
726
		$hidden_meta_boxes = array_diff( $hidden_meta_boxes, array( 'add-post-type-give_forms' ) );
727
728
		// Get current user ID.
729
		$user = wp_get_current_user();
730
731
		update_user_option( $user->ID, 'metaboxhidden_nav-menus', $hidden_meta_boxes, true );
732
		update_user_option( $user->ID, 'give_is_donation_forms_menu_updated', true, true );
733
	}
734
}
735
736
add_action( 'current_screen', 'give_donation_metabox_menu' );
737
738
/**
739
 * Array_column backup usage
740
 *
741
 * This file is part of the array_column library.
742
 *
743
 * @since      : 1.3.0.1
744
 *
745
 * @copyright  Copyright (c) Ben Ramsey (http://benramsey.com)
746
 * @license    https://opensource.org/licenses/MIT MIT
747
 */
748
749
if ( ! function_exists( 'array_column' ) ) {
750
	/**
751
	 * Returns the values from a single column of the input array, identified by
752
	 * the $columnKey.
753
	 *
754
	 * Optionally, you may provide an $indexKey to index the values in the returned
755
	 * array by the values from the $indexKey column in the input array.
756
	 *
757
	 * @param array      $input     A multi-dimensional array (record set) from which to pull
758
	 *                              a column of values.
759
	 * @param int|string $columnKey The column of values to return. This value may be the
760
	 *                              integer key of the column you wish to retrieve, or it
761
	 *                              may be the string key name for an associative array.
762
	 * @param mixed      $indexKey  (Optional.) The column to use as the index/keys for
763
	 *                              the returned array. This value may be the integer key
764
	 *                              of the column, or it may be the string key name.
765
	 *
766
	 * @return array
767
	 */
768
	function array_column( $input = null, $columnKey = null, $indexKey = null ) {
0 ignored issues
show
Unused Code introduced by
The parameter $input is not used and could be removed.

This check looks from parameters that have been defined for a function or method, but which are not used in the method body.

Loading history...
Unused Code introduced by
The parameter $columnKey is not used and could be removed.

This check looks from parameters that have been defined for a function or method, but which are not used in the method body.

Loading history...
Unused Code introduced by
The parameter $indexKey is not used and could be removed.

This check looks from parameters that have been defined for a function or method, but which are not used in the method body.

Loading history...
769
		// Using func_get_args() in order to check for proper number of
770
		// parameters and trigger errors exactly as the built-in array_column()
771
		// does in PHP 5.5.
772
		$argc   = func_num_args();
773
		$params = func_get_args();
774
775
		if ( $argc < 2 ) {
776
			trigger_error( sprintf( esc_html__( 'array_column() expects at least 2 parameters, %s given.', 'give' ), $argc ), E_USER_WARNING );
777
778
			return null;
779
		}
780
781
		if ( ! is_array( $params[0] ) ) {
782
			trigger_error( sprintf( esc_html__( 'array_column() expects parameter 1 to be array, %s given.', 'give' ), gettype( $params[0] ) ), E_USER_WARNING );
783
784
			return null;
785
		}
786
787 View Code Duplication
		if ( ! is_int( $params[1] ) && ! is_float( $params[1] ) && ! is_string( $params[1] ) && $params[1] !== null && ! ( is_object( $params[1] ) && method_exists( $params[1], '__toString' ) ) ) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
788
			trigger_error( esc_html__( 'array_column(): The column key should be either a string or an integer.', 'give' ), E_USER_WARNING );
789
790
			return false;
791
		}
792
793 View Code Duplication
		if ( isset( $params[2] ) && ! is_int( $params[2] ) && ! is_float( $params[2] ) && ! is_string( $params[2] ) && ! ( is_object( $params[2] ) && method_exists( $params[2], '__toString' ) ) ) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
794
			trigger_error( esc_html__( 'array_column(): The index key should be either a string or an integer.', 'give' ), E_USER_WARNING );
795
796
			return false;
797
		}
798
799
		$paramsInput     = $params[0];
800
		$paramsColumnKey = ( $params[1] !== null ) ? (string) $params[1] : null;
801
802
		$paramsIndexKey = null;
803
		if ( isset( $params[2] ) ) {
804
			if ( is_float( $params[2] ) || is_int( $params[2] ) ) {
805
				$paramsIndexKey = (int) $params[2];
806
			} else {
807
				$paramsIndexKey = (string) $params[2];
808
			}
809
		}
810
811
		$resultArray = array();
812
813
		foreach ( $paramsInput as $row ) {
814
			$key    = $value = null;
815
			$keySet = $valueSet = false;
816
817
			if ( $paramsIndexKey !== null && array_key_exists( $paramsIndexKey, $row ) ) {
818
				$keySet = true;
819
				$key    = (string) $row[ $paramsIndexKey ];
820
			}
821
822
			if ( $paramsColumnKey === null ) {
823
				$valueSet = true;
824
				$value    = $row;
825
			} elseif ( is_array( $row ) && array_key_exists( $paramsColumnKey, $row ) ) {
826
				$valueSet = true;
827
				$value    = $row[ $paramsColumnKey ];
828
			}
829
830
			if ( $valueSet ) {
831
				if ( $keySet ) {
832
					$resultArray[ $key ] = $value;
833
				} else {
834
					$resultArray[] = $value;
835
				}
836
			}
837
		}
838
839
		return $resultArray;
840
	}
841
}// End if().
842
843
/**
844
 * Determines the receipt visibility status.
845
 *
846
 * @since 1.3.2
847
 *
848
 * @param string $payment_key
849
 *
850
 * @return bool Whether the receipt is visible or not.
851
 */
852
function give_can_view_receipt( $payment_key = '' ) {
853
854
	$return = false;
855
856
	if ( empty( $payment_key ) ) {
857
		return $return;
858
	}
859
860
	global $give_receipt_args;
861
862
	$give_receipt_args['id'] = give_get_donation_id_by_key( $payment_key );
863
864
	$user_id = (int) give_get_payment_user_id( $give_receipt_args['id'] );
865
866
	$payment_meta = give_get_payment_meta( $give_receipt_args['id'] );
867
868
	if ( is_user_logged_in() ) {
869
		if ( $user_id === (int) get_current_user_id() ) {
870
			$return = true;
871
		} elseif ( wp_get_current_user()->user_email === give_get_payment_user_email( $give_receipt_args['id'] ) ) {
872
			$return = true;
873
		} elseif ( current_user_can( 'view_give_sensitive_data' ) ) {
874
			$return = true;
875
		}
876
	}
877
878
	// Check whether it is purchase session?
879
	$purchase_session = give_get_purchase_session();
880
	if ( ! empty( $purchase_session ) && ! is_user_logged_in() ) {
881
		if ( $purchase_session['purchase_key'] === $payment_meta['key'] ) {
882
			$return = true;
883
		}
884
	}
885
886
	// Check whether it is receipt access session?
887
	$receipt_session = give_get_receipt_session();
888
	if ( ! empty( $receipt_session ) && ! is_user_logged_in() ) {
889
		if ( $receipt_session === $payment_meta['key'] ) {
890
			$return = true;
891
		}
892
	}
893
894
	// Check whether it is history access session?
895
	if ( true === give_get_history_session() ) {
896
		$return = true;
897
	}
898
899
	return (bool) apply_filters( 'give_can_view_receipt', $return, $payment_key );
900
901
}
902
903
/**
904
 * Fallback for cal_days_in_month
905
 *
906
 * Fallback in case the calendar extension is not loaded in PHP; Only supports Gregorian calendar
907
 */
908
if ( ! function_exists( 'cal_days_in_month' ) ) {
909
	/**
910
	 * cal_days_in_month
911
	 *
912
	 * @param int $calendar
913
	 * @param int $month
914
	 * @param int $year
915
	 *
916
	 * @return bool|string
917
	 */
918
	function cal_days_in_month( $calendar, $month, $year ) {
0 ignored issues
show
Unused Code introduced by
The parameter $calendar is not used and could be removed.

This check looks from parameters that have been defined for a function or method, but which are not used in the method body.

Loading history...
919
		return date( 't', mktime( 0, 0, 0, $month, 1, $year ) );
920
	}
921
}
922
923
/**
924
 * Get plugin info including status, type, and license validation.
925
 *
926
 * This is an enhanced version of get_plugins() that returns the status
927
 * (`active` or `inactive`) of all plugins, type of plugin (`add-on` or `other`
928
 * and license validation for Give add-ons (`true` or `false`). Does not include
929
 * MU plugins.
930
 *
931
 * @since 1.8.0
932
 *
933
 * @return array Plugin info plus status, type, and license validation if
934
 *               available.
935
 */
936
function give_get_plugins() {
937
	$plugins             = get_plugins();
938
	$active_plugin_paths = (array) get_option( 'active_plugins', array() );
939
940
	if ( is_multisite() ) {
941
		$network_activated_plugin_paths = array_keys( get_site_option( 'active_sitewide_plugins', array() ) );
942
		$active_plugin_paths            = array_merge( $active_plugin_paths, $network_activated_plugin_paths );
943
	}
944
945
	foreach ( $plugins as $plugin_path => $plugin_data ) {
946
		// Is plugin active?
947
		if ( in_array( $plugin_path, $active_plugin_paths ) ) {
948
			$plugins[ $plugin_path ]['Status'] = 'active';
949
		} else {
950
			$plugins[ $plugin_path ]['Status'] = 'inactive';
951
		}
952
953
		$dirname = strtolower( dirname( $plugin_path ) );
954
955
		// Is plugin a Give add-on by WordImpress?
956
		if ( strstr( $dirname, 'give-' ) && strstr( $plugin_data['AuthorURI'], 'wordimpress.com' ) ) {
957
			// Plugin is a Give-addon.
958
			$plugins[ $plugin_path ]['Type'] = 'add-on';
959
960
			// Get license info from database.
961
			$plugin_name    = str_replace( 'Give - ', '', $plugin_data['Name'] );
962
			$db_option      = 'give_' . preg_replace( '/[^a-zA-Z0-9_\s]/', '', str_replace( ' ', '_', strtolower( $plugin_name ) ) ) . '_license_active';
963
			$license_active = get_option( $db_option );
964
965
			// Does a valid license exist?
966
			if ( ! empty( $license_active ) && 'valid' === $license_active->license ) {
967
				$plugins[ $plugin_path ]['License'] = true;
968
			} else {
969
				$plugins[ $plugin_path ]['License'] = false;
970
			}
971
		} else {
972
			// Plugin is not a Give add-on.
973
			$plugins[ $plugin_path ]['Type'] = 'other';
974
		}
975
	}
976
977
	return $plugins;
978
}
979
980
/**
981
 * Check if terms enabled or not for form.
982
 *
983
 * @since 1.8
984
 *
985
 * @param $form_id
986
 *
987
 * @return bool
988
 */
989
function give_is_terms_enabled( $form_id ) {
990
	$form_option = give_get_meta( $form_id, '_give_terms_option', true );
991
992
	if ( give_is_setting_enabled( $form_option, 'global' ) && give_is_setting_enabled( give_get_option( 'terms' ) ) ) {
993
		return true;
994
995
	} elseif ( give_is_setting_enabled( $form_option ) ) {
996
		return true;
997
998
	} else {
999
		return false;
1000
	}
1001
}
1002
1003
/**
1004
 * Delete donation stats cache.
1005
 *
1006
 * @todo  Resolve stats cache key naming issue. Currently it is difficult to regenerate cache key.
1007
 *
1008
 * @since 1.8.7
1009
 *
1010
 * @param string|array $date_range Date for stats.
1011
 *                                 Date value should be in today, yesterday, this_week, last_week, this_month,
1012
 *                                 last_month, this_quarter, last_quarter, this_year, last_year. For date value other,
1013
 *                                 all cache will be removed.
1014
 *
1015
 * @param array        $args
1016
 *
1017
 * @return WP_Error|bool
1018
 */
1019
function give_delete_donation_stats( $date_range = '', $args = array() ) {
1020
1021
	// Delete all cache.
1022
	$status = Give_Cache::delete( Give_Cache::get_options_like( 'give_stats' ) );
1023
1024
	/**
1025
	 * Fire the action when donation stats delete.
1026
	 *
1027
	 * @since 1.8.7
1028
	 *
1029
	 * @param string|array $date_range
1030
	 * @param array        $args
1031
	 */
1032
	do_action( 'give_delete_donation_stats', $status, $date_range, $args );
1033
1034
	return $status;
1035
}
1036
1037
/**
1038
 * Check if admin creating new donation form or not.
1039
 *
1040
 * @since 2.0
1041
 * @return bool
1042
 */
1043
function give_is_add_new_form_page() {
1044
	$status = false;
1045
1046
	if ( false !== strpos( $_SERVER['REQUEST_URI'], '/wp-admin/post-new.php?post_type=give_forms' ) ) {
1047
		$status = true;
1048
	}
1049
1050
	return $status;
1051
}
1052
1053
/**
1054
 * Get Form/Payment meta.
1055
 *
1056
 * Note: This function will help you to get meta for payment and form.
1057
 *       If you want to get meta for donors then use get_meta of Give_Donor and
1058
 *       If you want to get meta for logs then use get_meta of Give_Logging->logmeta_db.
1059
 *
1060
 * @since 1.8.8
1061
 *
1062
 * @param int    $id
1063
 * @param string $meta_key
1064
 * @param bool   $single
1065
 * @param bool   $default
1066
 *
1067
 * @return mixed
1068
 */
1069
function give_get_meta( $id, $meta_key = '', $single = false, $default = false ) {
1070
	/**
1071
	 * Filter the meta value
1072
	 *
1073
	 * @since 1.8.8
1074
	 */
1075
	$meta_value = apply_filters( 'give_get_meta', get_post_meta( $id, $meta_key, $single ), $id, $meta_key, $default );
1076
1077
	if ( ( empty( $meta_key ) || empty( $meta_value ) ) && $default ) {
1078
		$meta_value = $default;
1079
	}
1080
1081
	return $meta_value;
1082
}
1083
1084
/**
1085
 * Update Form/Payment meta.
1086
 *
1087
 * @since 1.8.8
1088
 *
1089
 * @param int    $id
1090
 * @param string $meta_key
1091
 * @param mixed  $meta_value
1092
 * @param mixed  $prev_value
1093
 *
1094
 * @return mixed
1095
 */
1096
function give_update_meta( $id, $meta_key, $meta_value, $prev_value = '' ) {
1097
	$status = update_post_meta( $id, $meta_key, $meta_value, $prev_value );
1098
1099
	/**
1100
	 * Filter the meta value update status
1101
	 *
1102
	 * @since 1.8.8
1103
	 */
1104
	return apply_filters( 'give_update_meta', $status, $id, $meta_key, $meta_value );
1105
}
1106
1107
/**
1108
 * Delete Form/Payment meta.
1109
 *
1110
 * @since 1.8.8
1111
 *
1112
 * @param int    $id
1113
 * @param string $meta_key
1114
 * @param string $meta_value
1115
 *
1116
 * @return mixed
1117
 */
1118
function give_delete_meta( $id, $meta_key, $meta_value = '' ) {
1119
	$status = delete_post_meta( $id, $meta_key, $meta_value );
1120
1121
	/**
1122
	 * Filter the meta value delete status
1123
	 *
1124
	 * @since 1.8.8
1125
	 */
1126
	return apply_filters( 'give_delete_meta', $status, $id, $meta_key, $meta_value );
1127
}
1128
1129
/**
1130
 * Check if the upgrade routine has been run for a specific action
1131
 *
1132
 * @since  1.0
1133
 *
1134
 * @param  string $upgrade_action The upgrade action to check completion for
1135
 *
1136
 * @return bool                   If the action has been added to the completed actions array
1137
 */
1138
function give_has_upgrade_completed( $upgrade_action = '' ) {
1139
	// Bailout.
1140
	if ( empty( $upgrade_action ) ) {
1141
		return false;
1142
	}
1143
1144
	// Fresh install?
1145
	// If fresh install then all upgrades will be consider as completed.
1146
	$is_fresh_install = ! get_option( 'give_version' );
1147
	if ( $is_fresh_install ) {
1148
		return true;
1149
	}
1150
0 ignored issues
show
Coding Style introduced by
Functions must not contain multiple empty lines in a row; found 2 empty lines
Loading history...
1151
1152
	$completed_upgrades = give_get_completed_upgrades();
1153
1154
	return in_array( $upgrade_action, $completed_upgrades );
1155
1156
}
1157
1158
/**
1159
 * For use when doing 'stepped' upgrade routines, to see if we need to start somewhere in the middle
1160
 *
1161
 * @since 1.8
1162
 *
1163
 * @return mixed   When nothing to resume returns false, otherwise starts the upgrade where it left off
1164
 */
1165
function give_maybe_resume_upgrade() {
1166
	$doing_upgrade = get_option( 'give_doing_upgrade', false );
1167
	if ( empty( $doing_upgrade ) ) {
1168
		return false;
1169
	}
1170
1171
	return $doing_upgrade;
1172
}
1173
1174
/**
1175
 * Adds an upgrade action to the completed upgrades array
1176
 *
1177
 * @since  1.0
1178
 *
1179
 * @param  string $upgrade_action The action to add to the completed upgrades array
1180
 *
1181
 * @return bool                   If the function was successfully added
1182
 */
1183
function give_set_upgrade_complete( $upgrade_action = '' ) {
1184
1185
	if ( empty( $upgrade_action ) ) {
1186
		return false;
1187
	}
1188
1189
	$completed_upgrades   = give_get_completed_upgrades();
1190
	$completed_upgrades[] = $upgrade_action;
1191
1192
	// Remove any blanks, and only show uniques.
1193
	$completed_upgrades = array_unique( array_values( $completed_upgrades ) );
1194
1195
	/**
1196
	 * Fire the action when any upgrade set to complete.
1197
	 *
1198
	 * @since 1.8.12
1199
	 */
1200
	do_action( 'give_set_upgrade_completed', $upgrade_action, $completed_upgrades );
1201
1202
	return update_option( 'give_completed_upgrades', $completed_upgrades, 'no' );
1203
}
1204
1205
/**
1206
 * Get's the array of completed upgrade actions
1207
 *
1208
 * @since  1.0
1209
 * @return array The array of completed upgrades
1210
 */
1211
function give_get_completed_upgrades() {
1212
	return (array) get_option( 'give_completed_upgrades' );
1213
}
1214
1215
/**
1216
 * In 2.0 we updated table for log, payment and form.
1217
 *
1218
 * Note: internal purpose only.
1219
 *
1220
 * @since 2.0
1221
 * @global wpdb  $wpdb
1222
 *
1223
 * @param string $type Context for table
1224
 *
1225
 * @return null|array
1226
 */
1227
function __give_v20_bc_table_details( $type ) {
1228
	global $wpdb;
1229
	$table = array();
1230
1231
	// Bailout.
1232
	if ( empty( $type ) ) {
1233
		return null;
1234
	}
1235
1236
	switch ( $type ) {
1237
		case 'form':
1238
			$table['name']         = $wpdb->formmeta;
1239
			$table['column']['id'] = 'form_id';
1240
1241
			break;
1242
1243
		case 'payment':
1244
			$table['name']         = $wpdb->paymentmeta;
1245
			$table['column']['id'] = 'payment_id';
1246
	}
1247
1248
	// Backward compatibility.
1249
	if ( ! give_has_upgrade_completed( 'v20_move_metadata_into_new_table' ) ) {
1250
		$table['name']         = $wpdb->postmeta;
1251
		$table['column']['id'] = 'post_id';
1252
	}
1253
0 ignored issues
show
Coding Style introduced by
Functions must not contain multiple empty lines in a row; found 2 empty lines
Loading history...
1254
1255
	return $table;
1256
}
1257
1258
/**
1259
 * Remove the Give transaction pages from WP search results.
1260
 *
1261
 * @since 1.8.13
1262
 *
1263
 * @param WP_Query $query
1264
 */
1265
function give_remove_pages_from_search( $query ) {
1266
1267
	if ( ! $query->is_admin && $query->is_search && $query->is_main_query() ) {
1268
1269
		$transaction_failed = give_get_option( 'failure_page', 0 );
1270
		$success_page       = give_get_option( 'success_page', 0 );
1271
1272
		$args = apply_filters( 'give_remove_pages_from_search', array(
1273
			$transaction_failed,
1274
			$success_page,
1275
		), $query );
1276
		$query->set( 'post__not_in', $args );
1277
	}
1278
}
1279
1280
add_action( 'pre_get_posts', 'give_remove_pages_from_search', 10, 1 );
1281
1282
/**
1283
 * Inserts a new key/value before a key in the array.
1284
 *
1285
 * @since 1.8.13
1286
 *
1287
 * @param string       $key       The key to insert before.
1288
 * @param array        $array     An array to insert in to.
1289
 * @param string       $new_key   The key to insert.
1290
 * @param array|string $new_value An value to insert.
1291
 *
1292
 * @return array The new array if the key exists, the passed array otherwise.
1293
 *
1294
 * @see   array_insert_before()
1295
 */
1296 View Code Duplication
function give_array_insert_before( $key, array &$array, $new_key, $new_value ) {
0 ignored issues
show
Duplication introduced by
This function seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
1297
	if ( array_key_exists( $key, $array ) ) {
1298
		$new = array();
1299
		foreach ( $array as $k => $value ) {
1300
			if ( $k === $key ) {
1301
				$new[ $new_key ] = $new_value;
1302
			}
1303
			$new[ $k ] = $value;
1304
		}
1305
1306
		return $new;
1307
	}
1308
1309
	return $array;
1310
}
1311
1312
/**
1313
 * Inserts a new key/value after a key in the array.
1314
 *
1315
 * @since 1.8.13
1316
 *
1317
 * @param string       $key       The key to insert after.
1318
 * @param array        $array     An array to insert in to.
1319
 * @param string       $new_key   The key to insert.
1320
 * @param array|string $new_value An value to insert.
1321
 *
1322
 * @return array The new array if the key exists, the passed array otherwise.
1323
 *
1324
 * @see   array_insert_before()
1325
 */
1326 View Code Duplication
function give_array_insert_after( $key, array &$array, $new_key, $new_value ) {
0 ignored issues
show
Duplication introduced by
This function seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
1327
	if ( array_key_exists( $key, $array ) ) {
1328
		$new = array();
1329
		foreach ( $array as $k => $value ) {
1330
			$new[ $k ] = $value;
1331
			if ( $k === $key ) {
1332
				$new[ $new_key ] = $new_value;
1333
			}
1334
		}
1335
1336
		return $new;
1337
	}
1338
1339
	return $array;
1340
}
1341
1342
/**
1343
 * Pluck a certain field out of each object in a list.
1344
 *
1345
 * This has the same functionality and prototype of
1346
 * array_column() (PHP 5.5) but also supports objects.
1347
 *
1348
 * @since 1.8.13
1349
 *
1350
 * @param array      $list      List of objects or arrays
1351
 * @param int|string $field     Field from the object to place instead of the entire object
1352
 * @param int|string $index_key Optional. Field from the object to use as keys for the new array.
1353
 *                              Default null.
1354
 *
1355
 * @return array Array of found values. If `$index_key` is set, an array of found values with keys
1356
 *               corresponding to `$index_key`. If `$index_key` is null, array keys from the original
1357
 *               `$list` will be preserved in the results.
1358
 */
1359
function give_list_pluck( $list, $field, $index_key = null ) {
1360
1361
	if ( ! $index_key ) {
1362
		/**
1363
		 * This is simple. Could at some point wrap array_column()
1364
		 * if we knew we had an array of arrays.
1365
		 */
1366
		foreach ( $list as $key => $value ) {
1367
			if ( is_object( $value ) ) {
1368
				if ( isset( $value->$field ) ) {
1369
					$list[ $key ] = $value->$field;
1370
				}
1371
			} else {
1372
				if ( isset( $value[ $field ] ) ) {
1373
					$list[ $key ] = $value[ $field ];
1374
				}
1375
			}
1376
		}
1377
1378
		return $list;
1379
	}
1380
1381
	/*
1382
	 * When index_key is not set for a particular item, push the value
1383
	 * to the end of the stack. This is how array_column() behaves.
1384
	 */
1385
	$newlist = array();
1386
	foreach ( $list as $value ) {
1387
		if ( is_object( $value ) ) {
1388
			if ( isset( $value->$index_key ) ) {
1389
				$newlist[ $value->$index_key ] = $value->$field;
1390
			} else {
1391
				$newlist[] = $value->$field;
1392
			}
1393
		} else {
1394
			if ( isset( $value[ $index_key ] ) ) {
1395
				$newlist[ $value[ $index_key ] ] = $value[ $field ];
1396
			} else {
1397
				$newlist[] = $value[ $field ];
1398
			}
1399
		}
1400
	}
1401
1402
	$list = $newlist;
1403
1404
	return $list;
1405
}
1406
1407
/**
1408
 * Add meta data field to a donor.
1409
 *
1410
 * @since 1.8.13
1411
 *
1412
 * @param int    $donor_id   Donor ID.
1413
 * @param string $meta_key   Metadata name.
1414
 * @param mixed  $meta_value Metadata value. Must be serializable if non-scalar.
1415
 * @param bool   $unique     Optional. Whether the same key should not be added.
1416
 *                           Default false.
1417
 *
1418
 * @return int|false Meta ID on success, false on failure.
1419
 */
1420
function add_donor_meta( $donor_id, $meta_key, $meta_value, $unique = false ) {
1421
	return add_metadata( 'give_customer', $donor_id, $meta_key, $meta_value, $unique );
1422
}
1423
1424
/**
1425
 * Remove metadata matching criteria from a Donor meta.
1426
 *
1427
 * You can match based on the key, or key and value. Removing based on key and
1428
 * value, will keep from removing duplicate metadata with the same key. It also
1429
 * allows removing all metadata matching key, if needed.
1430
 *
1431
 * @since 1.8.13
1432
 *
1433
 * @param int    $donor_id   Donor ID
1434
 * @param string $meta_key   Metadata name.
1435
 * @param mixed  $meta_value Optional. Metadata value.
1436
 *
1437
 * @return bool True on success, false on failure.
1438
 */
1439
function delete_donor_meta( $donor_id, $meta_key, $meta_value = '' ) {
1440
	return delete_metadata( 'give_customer', $donor_id, $meta_key, $meta_value );
1441
}
1442
1443
/**
1444
 * Retrieve donor meta field for a donor meta table.
1445
 *
1446
 * @since 1.8.13
1447
 *
1448
 * @param int    $donor_id Donor ID.
1449
 * @param string $key      Optional. The meta key to retrieve. By default, returns data for all keys.
1450
 * @param bool   $single   Whether to return a single value.
1451
 *
1452
 * @return mixed Will be an array if $single is false. Will be value of meta data field if $single
1453
 *  is true.
1454
 */
1455
function get_donor_meta( $donor_id, $key = '', $single = false ) {
1456
	return get_metadata( 'give_customer', $donor_id, $key, $single );
1457
}
1458
1459
/**
1460
 * Update customer meta field based on Donor ID.
1461
 *
1462
 * If the meta field for the donor does not exist, it will be added.
1463
 *
1464
 * @since 1.8.13
1465
 *
1466
 * @param int    $donor_id   Donor ID.
1467
 * @param string $meta_key   Metadata key.
1468
 * @param mixed  $meta_value Metadata value.
1469
 * @param mixed  $prev_value Optional. Previous value to check before removing.
1470
 *
1471
 * @return int|bool Meta ID if the key didn't exist, true on successful update, false on failure.
1472
 */
1473
function update_donor_meta( $donor_id, $meta_key, $meta_value, $prev_value = '' ) {
1474
	return update_metadata( 'give_customer', $donor_id, $meta_key, $meta_value, $prev_value );
1475
}
1476
1477
1478
/**
1479
 * Give recalculate income and donation of the donation from ID
1480
 *
1481
 * @since 1.8.13
1482
 *
1483
 * @param int $form_id Form id of which recalculation needs to be done.
1484
 *
1485
 * @return void
1486
 */
1487
function give_recount_form_income_donation( $form_id = 0 ) {
1488
	// Check if form id is not empty.
1489
	if ( ! empty( $form_id ) ) {
1490
		/**
1491
		 * Filter to modify payment status.
1492
		 *
1493
		 * @since 1.8.13
1494
		 */
1495
		$accepted_statuses = apply_filters( 'give_recount_accepted_statuses', array( 'publish' ) );
1496
1497
		/**
1498
		 * Filter to modify args of payment query before recalculating the form total
1499
		 *
1500
		 * @since 1.8.13
1501
		 */
1502
		$args = apply_filters( 'give_recount_form_stats_args', array(
1503
				'give_forms'     => $form_id,
1504
				'status'         => $accepted_statuses,
1505
				'posts_per_page' => - 1,
1506
				'fields'         => 'ids',
1507
			) );
0 ignored issues
show
Coding Style introduced by
This line of the multi-line function call does not seem to be indented correctly. Expected 8 spaces, but found 12.
Loading history...
1508
1509
		$totals = array(
1510
			'sales'    => 0,
1511
			'earnings' => 0,
1512
		);
1513
1514
		$payments = new Give_Payments_Query( $args );
1515
		$payments = $payments->get_payments();
1516
1517
		if ( $payments ) {
0 ignored issues
show
Bug Best Practice introduced by
The expression $payments of type array is implicitly converted to a boolean; are you sure this is intended? If so, consider using ! empty($expr) instead to make it clear that you intend to check for an array without elements.

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

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

Loading history...
1518 View Code Duplication
			foreach ( $payments as $payment ) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
1519
				// Ensure acceptible status only
1520
				if ( ! in_array( $payment->post_status, $accepted_statuses ) ) {
1521
					continue;
1522
				}
1523
1524
				// Ensure only payments for this form are counted
1525
				if ( $payment->form_id != $form_id ) {
1526
					continue;
1527
				}
1528
1529
				$totals['sales'] ++;
1530
				$totals['earnings'] += $payment->total;
1531
1532
			}
1533
		}
1534
		give_update_meta( $form_id, '_give_form_sales', $totals['sales'] );
1535
		give_update_meta( $form_id, '_give_form_earnings', give_sanitize_amount_for_db( $totals['earnings'] ) );
1536
	}// End if().
1537
}
1538
1539
1540
/**
1541
 * Get attribute string
1542
 *
1543
 * @since 1.8.17
1544
 *
1545
 * @param array $attributes
1546
 *
1547
 * @return string
1548
 */
1549
function give_get_attribute_str( $attributes ) {
1550
	$attribute_str = '';
1551
1552
	if ( empty( $attributes ) ) {
1553
		return $attribute_str;
1554
	}
1555
1556
	foreach ( $attributes as $tag => $value ) {
1557
		$attribute_str .= " {$tag}=\"{$value}\"";
1558
	}
1559
1560
	return trim( $attribute_str );
1561
}
1562
1563
/**
1564
 * Get the upload dir path
1565
 *
1566
 * @since 1.8.17
1567
 *
1568
 * @return string $wp_upload_dir;
1569
 */
1570
function give_get_wp_upload_dir() {
1571
	$wp_upload_dir = wp_upload_dir();
1572
1573
	return ( ! empty( $wp_upload_dir['path'] ) ? $wp_upload_dir['path'] : false );
1574
}
1575
1576
/**
1577
 * Get the data from uploaded JSON file
1578
 *
1579
 * @since 1.8.17
1580
 *
1581
 * @param string $file_name filename of the json file that is being uploaded
1582
 *
1583
 * @return string|bool $file_contents File content
1584
 */
1585
function give_get_core_settings_json( $file_name ) {
1586
	$upload_dir = give_get_wp_upload_dir();
1587
	$file_path  = $upload_dir . '/' . $file_name;
1588
1589
	if ( is_wp_error( $file_path ) || empty( $file_path ) ) {
1590
		Give_Admin_Settings::add_error( 'give-import-csv', __( 'Please upload or provide a valid JSON file.', 'give' ) );
1591
	}
1592
1593
	$file_contents = file_get_contents( $file_path );
0 ignored issues
show
introduced by
file_get_contents is highly discouraged, please use wpcom_vip_file_get_contents() instead.
Loading history...
1594
1595
	return $file_contents;
1596
}
1597
1598
/**
1599
 * Get number of donation to show when user is not login.
1600
 *
1601
 * @since 1.8.17
1602
 *
1603
 * @return int $country The two letter country code for the site's base country
1604
 */
1605
function give_get_limit_display_donations() {
1606
	return give_get_option( 'limit_display_donations', 1 );
1607
}
1608
1609
/**
1610
 * Add footer to the table when donor is view the donation history page with out login
1611
 *
1612
 * @since 1.8.17
1613
 */
1614
function give_donation_history_table_end() {
1615
	$email = Give()->session->get( 'give_email' );
1616
	?>
1617
	<tfoot>
1618
	<tr>
1619
		<td colspan="9999">
1620
			<div class="give-security-wrap">
1621
				<div class="give-security-column give-security-description-wrap">
1622
					<?php
1623
					echo sprintf( __( 'For security reasons, please confirm your email address (%s) to view your complete donation history.', 'give' ), $email );
0 ignored issues
show
introduced by
Expected a sanitizing function (see Codex for 'Data Validation'), but instead saw 'sprintf'
Loading history...
1624
					?>
1625
				</div>
1626
				<div class="give-security-column give-security-button-wrap">
1627
					<a href="#" data-email="<?php echo $email; ?>" id="give-confirm-email-btn"
0 ignored issues
show
introduced by
Expected next thing to be a escaping function, not '$email'
Loading history...
1628
					   class="give-confirm-email-btn give-btn">
1629
						<?php _e( 'Confirm Email', 'give' ); ?>
1630
					</a>
1631
					<span><?php _e( 'Email Sent!', 'give' ); ?></span>
1632
				</div>
1633
			</div>
1634
		</td>
1635
	</tr>
1636
	</tfoot>
1637
	<?php
1638
}
1639
1640
1641
/**
1642
 * Wrapper for _doing_it_wrong.
1643
 *
1644
 * @since  1.8.18
1645
 *
1646
 * @param  string $function
1647
 * @param  string $message
1648
 * @param  string $version
1649
 *
1650
 * @return void
1651
 */
1652
function give_doing_it_wrong( $function, $message, $version ) {
1653
	$message .= "\nBacktrace:" . wp_debug_backtrace_summary();
1654
1655
	_doing_it_wrong( $function, $message, $version );
1656
}
1657
1658
1659
/**
1660
 * Remove limit from running php script complete.
1661
 *
1662
 * @since 1.8.18
1663
 */
1664
function give_ignore_user_abort() {
1665
	ignore_user_abort( true );
1666
1667
	if ( ! give_is_func_disabled( 'set_time_limit' ) && ! ini_get( 'safe_mode' ) ) {
1668
		set_time_limit( 0 );
1669
	}
1670
}
1671
1672
/**
1673
 * Get post type count.
1674
 *
1675
 * @since 2.0.2
1676
 *
1677
 * @param string $post_type
1678
 * @param array  $args
1679
 *
1680
 * @return int
1681
 */
1682
function give_get_total_post_type_count( $post_type = '', $args = array() ) {
0 ignored issues
show
Unused Code introduced by
The parameter $args is not used and could be removed.

This check looks from parameters that have been defined for a function or method, but which are not used in the method body.

Loading history...
1683
	global $wpdb;
1684
	$where = '';
1685
1686
	if ( ! $post_type ) {
1687
		return 0;
1688
	}
1689
1690
	// Bulit where query
1691
	if ( ! empty( $post_type ) ) {
1692
		$where .= ' WHERE';
1693
1694
		if ( is_array( $post_type ) ) {
1695
			$where .= " post_type='" . implode( "' OR post_type='", $post_type ) . "'";
1696
		} else {
1697
			$where .= " post_type='{$post_type}'";
1698
		}
1699
	}
1700
1701
	$result = $wpdb->get_var( "SELECT count(ID) FROM {$wpdb->posts}{$where}" );
0 ignored issues
show
introduced by
Usage of a direct database call is discouraged.
Loading history...
introduced by
Usage of a direct database call without caching is prohibited. Use wp_cache_get / wp_cache_set.
Loading history...
1702
1703
	return absint( $result );
1704
}
1705
1706
/**
1707
 * Define a constant if it is not already defined.
1708
 *
1709
 * @since  2.0.5
1710
 *
1711
 * @param string $name  Constant name.
1712
 * @param string $value Value.
1713
 *
1714
 * @credit WooCommerce
1715
 */
1716
function give_maybe_define_constant( $name, $value ) {
1717
	if ( ! defined( $name ) ) {
1718
		define( $name, $value );
1719
	}
1720
}
1721
1722
/**
1723
 * Decode time short tag in string
1724
 *
1725
 * @since 2.1.0
1726
 *
1727
 * @param string $string
1728
 * @param int    $timestamp
1729
 *
1730
 * @return string
1731
 */
1732
function give_time_do_tags( $string, $timestamp = 0 ) {
1733
	$current_time = ! empty( $timestamp ) ? $timestamp : current_time( 'timestamp' );
1734
1735
	$formatted_string = str_replace( array(
1736
		'{D}',
1737
		'{DD}',
1738
		'{M}',
1739
		'{MM}',
1740
		'{YY}',
1741
		'{YYYY}',
1742
		'{H}',
1743
		'{HH}',
1744
		'{N}',
1745
		'{S}'
0 ignored issues
show
introduced by
Comma required after last value in array declaration
Loading history...
1746
	), array(
0 ignored issues
show
Coding Style introduced by
This line of the multi-line function call does not seem to be indented correctly. Expected 8 spaces, but found 4.
Loading history...
1747
			date( 'j', $current_time ),
1748
			date( 'd', $current_time ),
1749
			date( 'n', $current_time ),
1750
			date( 'm', $current_time ),
1751
			date( 'Y', $current_time ),
1752
			date( 'Y', $current_time ),
1753
			date( 'G', $current_time ),
1754
			date( 'H', $current_time ),
1755
			date( 's', $current_time )
0 ignored issues
show
introduced by
Comma required after last value in array declaration
Loading history...
1756
		), $string );
0 ignored issues
show
Coding Style introduced by
This line of the multi-line function call does not seem to be indented correctly. Expected 4 spaces, but found 8.
Loading history...
1757
1758
	/**
1759
	 * Filter the parsed string.
1760
	 *
1761
	 * @since 2.1.0
1762
	 */
1763
	return apply_filters( 'give_time_do_tags', $formatted_string, $string, $timestamp );
1764
}
1765
1766
1767
/**
1768
 * Check if Company field enabled or not for form or globally.
1769
 *
1770
 * @since 2.1
1771
 *
1772
 * @param $form_id
1773
 *
1774
 * @return bool
1775
 */
1776
function give_is_company_field_enabled( $form_id ) {
1777
	$form_setting_val   = give_get_meta( $form_id, '_give_company_field', true );
1778
	$global_setting_val = give_get_option( 'company_field' );
1779
1780
	if ( ! empty( $form_setting_val ) ) {
1781
		if ( give_is_setting_enabled( $form_setting_val, array( 'required', 'optional' ) ) ) {
0 ignored issues
show
Documentation introduced by
array('required', 'optional') is of type array<integer,string,{"0":"string","1":"string"}>, but the function expects a string|null.

It seems like the type of the argument is not accepted by the function/method which you are calling.

In some cases, in particular if PHP’s automatic type-juggling kicks in this might be fine. In other cases, however this might be a bug.

We suggest to add an explicit type cast like in the following example:

function acceptsInteger($int) { }

$x = '123'; // string "123"

// Instead of
acceptsInteger($x);

// we recommend to use
acceptsInteger((integer) $x);
Loading history...
1782
			return true;
1783
		} elseif ( 'global' === $form_setting_val && give_is_setting_enabled( $global_setting_val, array(
0 ignored issues
show
Documentation introduced by
array('required', 'optional') is of type array<integer,string,{"0":"string","1":"string"}>, but the function expects a string|null.

It seems like the type of the argument is not accepted by the function/method which you are calling.

In some cases, in particular if PHP’s automatic type-juggling kicks in this might be fine. In other cases, however this might be a bug.

We suggest to add an explicit type cast like in the following example:

function acceptsInteger($int) { }

$x = '123'; // string "123"

// Instead of
acceptsInteger($x);

// we recommend to use
acceptsInteger((integer) $x);
Loading history...
1784
				'required',
1785
				'optional'
0 ignored issues
show
introduced by
Comma required after last value in array declaration
Loading history...
1786
			) ) ) {
0 ignored issues
show
Coding Style introduced by
This line of the multi-line function call does not seem to be indented correctly. Expected 8 spaces, but found 12.
Loading history...
1787
			return true;
1788
		} else {
1789
			return false;
1790
		}
0 ignored issues
show
introduced by
Blank line found after control structure
Loading history...
1791
1792
	} elseif ( give_is_setting_enabled( $global_setting_val, array( 'required', 'optional' ) ) ) {
0 ignored issues
show
Documentation introduced by
array('required', 'optional') is of type array<integer,string,{"0":"string","1":"string"}>, but the function expects a string|null.

It seems like the type of the argument is not accepted by the function/method which you are calling.

In some cases, in particular if PHP’s automatic type-juggling kicks in this might be fine. In other cases, however this might be a bug.

We suggest to add an explicit type cast like in the following example:

function acceptsInteger($int) { }

$x = '123'; // string "123"

// Instead of
acceptsInteger($x);

// we recommend to use
acceptsInteger((integer) $x);
Loading history...
1793
		return true;
1794
1795
	} else {
1796
		return false;
1797
	}
1798
}
1799