Completed
Push — issues/2504 ( bfffe3 )
by Ravinder
525:17 queued 518:31
created

misc-functions.php ➔ give_doing_it_wrong()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 5

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
nc 1
nop 3
dl 0
loc 5
rs 10
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, ',' ) ) {
0 ignored issues
show
introduced by
Space after opening control structure is required
Loading history...
introduced by
No space before opening parenthesis is prohibited
Loading history...
174
		$ip = give_clean( explode( ',', $ip ) );
175
		$ip = array_filter( $ip );
176
		$ip = implode( ',', $ip );
177
	} else{
0 ignored issues
show
introduced by
Space after opening control structure is required
Loading history...
introduced by
No space before opening parenthesis is prohibited
Loading history...
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 (
265
			'custom' !== $price_id
266
			&& ! empty( $item_price_level_text )
267
		) {
268
			// Matches a donation level - append level text.
269
			$item_name .= ' - ' . $item_price_level_text;
270
		}
271
	}
272
273
	/**
274
	 * Filter the Item Title of Payment Gateway.
275
	 *
276
	 * @param string $item_name    Item Title of Payment Gateway.
277
	 * @param int    $form_id      Donation Form ID.
278
	 * @param array  $payment_data Payment Data.
279
	 *
280
	 * @since 1.8.14
281
	 *
282
	 * @return string
283
	 */
284
	return apply_filters( 'give_payment_gateway_item_title', $item_name, $form_id, $payment_data );
285
}
286
287
/**
288
 * Get Donation Summary
289
 *
290
 * Creates a donation summary for payment gateways from the donation data before the payment is created in the database.
291
 *
292
 * @since       1.8.12
293
 *
294
 * @param array $donation_data
295
 * @param bool  $name_and_email
296
 * @param int   $length
297
 *
298
 * @return string
299
 */
300
function give_payment_gateway_donation_summary( $donation_data, $name_and_email = true, $length = 255 ) {
301
	$form_id = isset( $donation_data['post_data']['give-form-id'] ) ? $donation_data['post_data']['give-form-id'] : '';
302
303
	// Form title.
304
	$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' ) ) );
305
306
	// Form multilevel if applicable.
307
	if ( isset( $donation_data['post_data']['give-price-id'] ) ) {
308
		$summary .= ': ' . give_get_price_option_name( $form_id, $donation_data['post_data']['give-price-id'] );
309
	}
310
311
	// Add Donor's name + email if requested.
312
	if ( $name_and_email ) {
313
314
		// First name
315 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...
316
			$summary .= ' - ' . $donation_data['user_info']['first_name'];
317
		}
318
319 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...
320
			$summary .= ' ' . $donation_data['user_info']['last_name'];
321
		}
322
323
		$summary .= ' (' . $donation_data['user_email'] . ')';
324
	}
325
326
	// Cut the length
327
	$summary = substr( $summary, 0, $length );
328
329
	return apply_filters( 'give_payment_gateway_donation_summary', $summary );
330
}
331
332
333
/**
334
 * Get user host
335
 *
336
 * Returns the webhost this site is using if possible
337
 *
338
 * @since 1.0
339
 * @return string $host if detected, false otherwise
340
 */
341
function give_get_host() {
342
	$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...
343
344
	if ( defined( 'WPE_APIKEY' ) ) {
345
		$host = 'WP Engine';
346
	} elseif ( defined( 'PAGELYBIN' ) ) {
347
		$host = 'Pagely';
348
	} elseif ( DB_HOST == 'localhost:/tmp/mysql5.sock' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
349
		$host = 'ICDSoft';
350
	} elseif ( DB_HOST == 'mysqlv5' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
351
		$host = 'NetworkSolutions';
352
	} elseif ( strpos( DB_HOST, 'ipagemysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
353
		$host = 'iPage';
354
	} elseif ( strpos( DB_HOST, 'ipowermysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
355
		$host = 'IPower';
356
	} elseif ( strpos( DB_HOST, '.gridserver.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
357
		$host = 'MediaTemple Grid';
358
	} elseif ( strpos( DB_HOST, '.pair.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
359
		$host = 'pair Networks';
360
	} elseif ( strpos( DB_HOST, '.stabletransit.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
361
		$host = 'Rackspace Cloud';
362
	} elseif ( strpos( DB_HOST, '.sysfix.eu' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
363
		$host = 'SysFix.eu Power Hosting';
364
	} elseif ( strpos( $_SERVER['SERVER_NAME'], 'Flywheel' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
365
		$host = 'Flywheel';
366
	} else {
367
		// Adding a general fallback for data gathering
368
		$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...
369
	}
370
371
	return $host;
372
}
373
374
375
/**
376
 * Check site host
377
 *
378
 * @since 1.0
379
 *
380
 * @param bool /string $host The host to check
381
 *
382
 * @return bool true if host matches, false if not
383
 */
384
function give_is_host( $host = false ) {
385
386
	$return = false;
387
388
	if ( $host ) {
389
		$host = str_replace( ' ', '', strtolower( $host ) );
390
391
		switch ( $host ) {
392
			case 'wpengine':
393
				if ( defined( 'WPE_APIKEY' ) ) {
394
					$return = true;
395
				}
396
				break;
397
			case 'pagely':
398
				if ( defined( 'PAGELYBIN' ) ) {
399
					$return = true;
400
				}
401
				break;
402
			case 'icdsoft':
403
				if ( DB_HOST == 'localhost:/tmp/mysql5.sock' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
404
					$return = true;
405
				}
406
				break;
407
			case 'networksolutions':
408
				if ( DB_HOST == 'mysqlv5' ) {
0 ignored issues
show
introduced by
Found "== '". Use Yoda Condition checks, you must
Loading history...
409
					$return = true;
410
				}
411
				break;
412
			case 'ipage':
413
				if ( strpos( DB_HOST, 'ipagemysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
414
					$return = true;
415
				}
416
				break;
417
			case 'ipower':
418
				if ( strpos( DB_HOST, 'ipowermysql.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
419
					$return = true;
420
				}
421
				break;
422
			case 'mediatemplegrid':
423
				if ( strpos( DB_HOST, '.gridserver.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
424
					$return = true;
425
				}
426
				break;
427
			case 'pairnetworks':
428
				if ( strpos( DB_HOST, '.pair.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
429
					$return = true;
430
				}
431
				break;
432
			case 'rackspacecloud':
433
				if ( strpos( DB_HOST, '.stabletransit.com' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
434
					$return = true;
435
				}
436
				break;
437
			case 'sysfix.eu':
438
			case 'sysfix.eupowerhosting':
439
				if ( strpos( DB_HOST, '.sysfix.eu' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
440
					$return = true;
441
				}
442
				break;
443
			case 'flywheel':
444
				if ( strpos( $_SERVER['SERVER_NAME'], 'Flywheel' ) !== false ) {
0 ignored issues
show
introduced by
Found "!== false". Use Yoda Condition checks, you must
Loading history...
445
					$return = true;
446
				}
447
				break;
448
			default:
449
				$return = false;
450
		}// End switch().
451
	}// End if().
452
453
	return $return;
454
}
455
456
/**
457
 * Marks a function as deprecated and informs when it has been used.
458
 *
459
 * There is a hook give_deprecated_function_run that will be called that can be used
460
 * to get the backtrace up to what file and function called the deprecated
461
 * function.
462
 *
463
 * The current behavior is to trigger a user error if WP_DEBUG is true.
464
 *
465
 * This function is to be used in every function that is deprecated.
466
 *
467
 * @uses do_action() Calls 'give_deprecated_function_run' and passes the function name, what to use instead,
468
 *   and the version the function was deprecated in.
469
 * @uses apply_filters() Calls 'give_deprecated_function_trigger_error' and expects boolean value of true to do
470
 *   trigger or false to not trigger error.
471
 *
472
 * @param string $function    The function that was called.
473
 * @param string $version     The plugin version that deprecated the function.
474
 * @param string $replacement Optional. The function that should have been called.
475
 * @param array  $backtrace   Optional. Contains stack backtrace of deprecated function.
476
 */
477
function _give_deprecated_function( $function, $version, $replacement = null, $backtrace = null ) {
478
479
	/**
480
	 * Fires while give deprecated function call occurs.
481
	 *
482
	 * Allow you to hook to deprecated function call.
483
	 *
484
	 * @since 1.0
485
	 *
486
	 * @param string $function    The function that was called.
487
	 * @param string $replacement Optional. The function that should have been called.
488
	 * @param string $version     The plugin version that deprecated the function.
489
	 */
490
	do_action( 'give_deprecated_function_run', $function, $replacement, $version );
491
492
	$show_errors = current_user_can( 'manage_options' );
493
494
	// Allow plugin to filter the output error trigger.
495
	if ( WP_DEBUG && apply_filters( 'give_deprecated_function_trigger_error', $show_errors ) ) {
496
		if ( ! is_null( $replacement ) ) {
497
			trigger_error( sprintf( __( '%1$s is <strong>deprecated</strong> since Give version %2$s! Use %3$s instead.', 'give' ), $function, $version, $replacement ) );
498
			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...
499
			// Alternatively we could dump this to a file.
500
		} else {
501
			trigger_error( sprintf( __( '%1$s is <strong>deprecated</strong> since Give version %2$s with no alternative available.', 'give' ), $function, $version ) );
502
			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...
503
			// Alternatively we could dump this to a file.
504
		}
505
	}
506
}
507
508
/**
509
 * Give Get Admin ID
510
 *
511
 * Helper function to return the ID of the post for admin usage
512
 *
513
 * @return string $post_id
514
 */
515
function give_get_admin_post_id() {
516
	$post_id = isset( $_REQUEST['post'] )
0 ignored issues
show
introduced by
Detected access of super global var $_REQUEST, probably need manual inspection.
Loading history...
517
		? absint( $_REQUEST['post'] )
0 ignored issues
show
introduced by
Detected access of super global var $_REQUEST, probably need manual inspection.
Loading history...
518
		: null;
519
520
	$post_id = ! empty( $post_id )
521
		? $post_id
522
		: ( 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...
523
524
	$post_id = ! empty( $post_id )
525
		? $post_id
526
		: ( 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...
527
528
	return $post_id;
529
}
530
531
/**
532
 * Get PHP Arg Separator Output
533
 *
534
 * @since 1.0
535
 * @return string Arg separator output
536
 */
537
function give_get_php_arg_separator_output() {
538
	return ini_get( 'arg_separator.output' );
539
}
540
541
542
/**
543
 * Month Num To Name
544
 *
545
 * Takes a month number and returns the name three letter name of it.
546
 *
547
 * @since 1.0
548
 *
549
 * @param int $n
550
 *
551
 * @return string Short month name
552
 */
553
function give_month_num_to_name( $n ) {
554
	$timestamp = mktime( 0, 0, 0, $n, 1, 2005 );
555
556
	return date_i18n( 'M', $timestamp );
557
}
558
559
560
/**
561
 * Checks whether function is disabled.
562
 *
563
 * @since 1.0
564
 *
565
 * @param string $function Name of the function.
566
 *
567
 * @return bool Whether or not function is disabled.
568
 */
569
function give_is_func_disabled( $function ) {
570
	$disabled = explode( ',', ini_get( 'disable_functions' ) );
571
572
	return in_array( $function, $disabled );
573
}
574
575
576
/**
577
 * Give Newsletter
578
 *
579
 * Returns the main Give newsletter form
580
 */
581
function give_get_newsletter() {
582
	?>
583
584
	<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>
585
586
	<div class="give-newsletter-form-wrap">
587
588
		<form action="//givewp.us3.list-manage.com/subscribe/post?u=3ccb75d68bda4381e2f45794c&amp;id=12a081aa13"
589
		      method="post" id="mc-embedded-subscribe-form" name="mc-embedded-subscribe-form" class="validate"
590
		      target="_blank" novalidate>
591
			<div class="give-newsletter-confirmation">
592
				<p><?php esc_html_e( 'Thanks for Subscribing!', 'give' ); ?> :)</p>
593
			</div>
594
595
			<table class="form-table give-newsletter-form">
596
				<tr valign="middle">
597
					<td>
598
						<label for="mce-EMAIL"
599
						       class="screen-reader-text"><?php esc_html_e( 'Email Address (required)', 'give' ); ?></label>
600
						<input type="email" name="EMAIL" id="mce-EMAIL"
601
						       placeholder="<?php esc_attr_e( 'Email Address (required)', 'give' ); ?>"
602
						       class="required email" value="">
603
					</td>
604
					<td>
605
						<label for="mce-FNAME"
606
						       class="screen-reader-text"><?php esc_html_e( 'First Name', 'give' ); ?></label>
607
						<input type="text" name="FNAME" id="mce-FNAME"
608
						       placeholder="<?php esc_attr_e( 'First Name', 'give' ); ?>" class="" value="">
609
					</td>
610
					<td>
611
						<label for="mce-LNAME"
612
						       class="screen-reader-text"><?php esc_html_e( 'Last Name', 'give' ); ?></label>
613
						<input type="text" name="LNAME" id="mce-LNAME"
614
						       placeholder="<?php esc_attr_e( 'Last Name', 'give' ); ?>" class="" value="">
615
					</td>
616
					<td>
617
						<input type="submit" name="subscribe" id="mc-embedded-subscribe" class="button"
618
						       value="<?php esc_attr_e( 'Subscribe', 'give' ); ?>">
619
					</td>
620
				</tr>
621
			</table>
622
		</form>
623
624
		<div style="position: absolute; left: -5000px;">
625
			<input type="text" name="b_3ccb75d68bda4381e2f45794c_12a081aa13" tabindex="-1" value="">
626
		</div>
627
628
	</div>
629
630
	<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...
631
	<script type='text/javascript'>(function( $ ) {
632
			window.fnames = new Array();
633
			window.ftypes = new Array();
634
			fnames[ 0 ] = 'EMAIL';
635
			ftypes[ 0 ] = 'email';
636
			fnames[ 1 ] = 'FNAME';
637
			ftypes[ 1 ] = 'text';
638
			fnames[ 2 ] = 'LNAME';
639
			ftypes[ 2 ] = 'text';
640
641
			//Successful submission
642
			$( 'form[name="mc-embedded-subscribe-form"]' ).on( 'submit', function() {
643
644
				var email_field = $( this ).find( '#mce-EMAIL' ).val();
645
				if ( ! email_field ) {
646
					return false;
647
				}
648
				$( this ).find( '.give-newsletter-confirmation' ).show().delay( 5000 ).slideUp();
649
				$( this ).find( '.give-newsletter-form' ).hide();
650
651
			} );
652
653
		}( jQuery ));
654
		var $mcj = jQuery.noConflict( true );
655
656
657
	</script>
658
	<!--End mc_embed_signup-->
659
660
	<?php
661
}
662
663
664
/**
665
 * Create SVG library function
666
 *
667
 * @param string $icon
668
 *
669
 * @return string
670
 */
671
function give_svg_icons( $icon ) {
672
673
	// Store your SVGs in an associative array
674
	$svgs = array(
675
		'microphone'    => '',
676
		'alert'         => '',
677
		'placemark'     => '',
678
		'give_grey'     => '',
679
		'give_cpt_icon' => '',
680
	);
681
682
	// Return the chosen icon's SVG string
683
	return $svgs[ $icon ];
684
}
685
686
/**
687
 * Modify Admin Nav Menu Label
688
 *
689
 * @since 1.3
690
 *
691
 * @param object $post_type The current object to add a menu items meta box for.
692
 *
693
 * @return mixed
694
 */
695
function modify_nav_menu_meta_box_object( $post_type ) {
696
	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...
697
		$post_type->labels->name = esc_html__( 'Donation Forms', 'give' );
698
	}
699
700
	return $post_type;
701
}
702
703
add_filter( 'nav_menu_meta_box_object', 'modify_nav_menu_meta_box_object' );
704
705
/**
706
 * Show Donation Forms Post Type in Appearance > Menus by default on fresh install.
707
 *
708
 * @since 1.8.14
709
 *
710
 * @todo  Remove this, when WordPress Core ticket is resolved (https://core.trac.wordpress.org/ticket/16828).
711
 *
712
 * @return bool
713
 */
714
function give_donation_metabox_menu() {
715
716
	// Get Current Screen.
717
	$screen = get_current_screen();
718
719
	// Proceed, if current screen is navigation menus.
720
	if (
721
		'nav-menus' === $screen->id &&
722
		give_is_setting_enabled( give_get_option( 'forms_singular' ) ) &&
723
		! get_user_option( 'give_is_donation_forms_menu_updated' )
724
	) {
725
726
		// Return false, if it fails to retrieve hidden meta box list and is not admin.
727
		if (
728
			! is_admin() ||
729
			( ! $hidden_meta_boxes = get_user_option( 'metaboxhidden_nav-menus' ) )
730
		) {
731
			return false;
732
		}
733
734
		// Return false, In case, we don't find 'Donation Form' in hidden meta box list.
735
		if ( ! in_array( 'add-post-type-give_forms', $hidden_meta_boxes, true ) ) {
736
			return false;
737
		}
738
739
		// Exclude 'Donation Form' value from hidden meta box's list.
740
		$hidden_meta_boxes = array_diff( $hidden_meta_boxes, array( 'add-post-type-give_forms' ) );
741
742
		// Get current user ID.
743
		$user = wp_get_current_user();
744
745
		update_user_option( $user->ID, 'metaboxhidden_nav-menus', $hidden_meta_boxes, true );
746
		update_user_option( $user->ID, 'give_is_donation_forms_menu_updated', true, true );
747
	}
748
}
749
750
add_action( 'current_screen', 'give_donation_metabox_menu' );
751
752
/**
753
 * Array_column backup usage
754
 *
755
 * This file is part of the array_column library.
756
 *
757
 * @since      : 1.3.0.1
758
 *
759
 * @copyright  Copyright (c) Ben Ramsey (http://benramsey.com)
760
 * @license    https://opensource.org/licenses/MIT MIT
761
 */
762
763
if ( ! function_exists( 'array_column' ) ) {
764
	/**
765
	 * Returns the values from a single column of the input array, identified by
766
	 * the $columnKey.
767
	 *
768
	 * Optionally, you may provide an $indexKey to index the values in the returned
769
	 * array by the values from the $indexKey column in the input array.
770
	 *
771
	 * @param array      $input     A multi-dimensional array (record set) from which to pull
772
	 *                              a column of values.
773
	 * @param int|string $columnKey The column of values to return. This value may be the
774
	 *                              integer key of the column you wish to retrieve, or it
775
	 *                              may be the string key name for an associative array.
776
	 * @param mixed      $indexKey  (Optional.) The column to use as the index/keys for
777
	 *                              the returned array. This value may be the integer key
778
	 *                              of the column, or it may be the string key name.
779
	 *
780
	 * @return array
781
	 */
782
	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...
783
		// Using func_get_args() in order to check for proper number of
784
		// parameters and trigger errors exactly as the built-in array_column()
785
		// does in PHP 5.5.
786
		$argc   = func_num_args();
787
		$params = func_get_args();
788
789
		if ( $argc < 2 ) {
790
			trigger_error( sprintf( esc_html__( 'array_column() expects at least 2 parameters, %s given.', 'give' ), $argc ), E_USER_WARNING );
791
792
			return null;
793
		}
794
795
		if ( ! is_array( $params[0] ) ) {
796
			trigger_error( sprintf( esc_html__( 'array_column() expects parameter 1 to be array, %s given.', 'give' ), gettype( $params[0] ) ), E_USER_WARNING );
797
798
			return null;
799
		}
800
801 View Code Duplication
		if ( ! is_int( $params[1] )
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...
802
		     && ! is_float( $params[1] )
803
		     && ! is_string( $params[1] )
804
		     && $params[1] !== null
805
		     && ! ( is_object( $params[1] ) && method_exists( $params[1], '__toString' ) )
806
		) {
807
			trigger_error( esc_html__( 'array_column(): The column key should be either a string or an integer.', 'give' ), E_USER_WARNING );
808
809
			return false;
810
		}
811
812 View Code Duplication
		if ( isset( $params[2] )
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...
813
		     && ! is_int( $params[2] )
814
		     && ! is_float( $params[2] )
815
		     && ! is_string( $params[2] )
816
		     && ! ( is_object( $params[2] ) && method_exists( $params[2], '__toString' ) )
817
		) {
818
			trigger_error( esc_html__( 'array_column(): The index key should be either a string or an integer.', 'give' ), E_USER_WARNING );
819
820
			return false;
821
		}
822
823
		$paramsInput     = $params[0];
824
		$paramsColumnKey = ( $params[1] !== null ) ? (string) $params[1] : null;
825
826
		$paramsIndexKey = null;
827
		if ( isset( $params[2] ) ) {
828
			if ( is_float( $params[2] ) || is_int( $params[2] ) ) {
829
				$paramsIndexKey = (int) $params[2];
830
			} else {
831
				$paramsIndexKey = (string) $params[2];
832
			}
833
		}
834
835
		$resultArray = array();
836
837
		foreach ( $paramsInput as $row ) {
838
			$key    = $value = null;
839
			$keySet = $valueSet = false;
840
841
			if ( $paramsIndexKey !== null && array_key_exists( $paramsIndexKey, $row ) ) {
842
				$keySet = true;
843
				$key    = (string) $row[ $paramsIndexKey ];
844
			}
845
846
			if ( $paramsColumnKey === null ) {
847
				$valueSet = true;
848
				$value    = $row;
849
			} elseif ( is_array( $row ) && array_key_exists( $paramsColumnKey, $row ) ) {
850
				$valueSet = true;
851
				$value    = $row[ $paramsColumnKey ];
852
			}
853
854
			if ( $valueSet ) {
855
				if ( $keySet ) {
856
					$resultArray[ $key ] = $value;
857
				} else {
858
					$resultArray[] = $value;
859
				}
860
			}
861
		}
862
863
		return $resultArray;
864
	}
865
}// End if().
866
867
/**
868
 * Determines the receipt visibility status.
869
 *
870
 * @since 1.3.2
871
 *
872
 * @param string $payment_key
873
 *
874
 * @return bool Whether the receipt is visible or not.
875
 */
876
function give_can_view_receipt( $payment_key = '' ) {
877
878
	$return = false;
879
880
	if ( empty( $payment_key ) ) {
881
		return $return;
882
	}
883
884
	global $give_receipt_args;
885
886
	$give_receipt_args['id'] = give_get_purchase_id_by_key( $payment_key );
887
888
	$user_id = (int) give_get_payment_user_id( $give_receipt_args['id'] );
889
890
	$payment_meta = give_get_payment_meta( $give_receipt_args['id'] );
891
892
	if ( is_user_logged_in() ) {
893
		if ( $user_id === (int) get_current_user_id() ) {
894
			$return = true;
895
		} elseif ( wp_get_current_user()->user_email === give_get_payment_user_email( $give_receipt_args['id'] ) ) {
896
			$return = true;
897
		} elseif ( current_user_can( 'view_give_sensitive_data' ) ) {
898
			$return = true;
899
		}
900
	}
901
902
	// Check whether it is purchase session?
903
	$purchase_session = give_get_purchase_session();
904
	if ( ! empty( $purchase_session ) && ! is_user_logged_in() ) {
905
		if ( $purchase_session['purchase_key'] === $payment_meta['key'] ) {
906
			$return = true;
907
		}
908
	}
909
910
	// Check whether it is receipt access session?
911
	$receipt_session = give_get_receipt_session();
912
	if ( ! empty( $receipt_session ) && ! is_user_logged_in() ) {
913
		if ( $receipt_session === $payment_meta['key'] ) {
914
			$return = true;
915
		}
916
	}
917
918
	// Check whether it is history access session?
919
	if ( true === give_get_history_session() ) {
920
		$return = true;
921
	}
922
923
	return (bool) apply_filters( 'give_can_view_receipt', $return, $payment_key );
924
925
}
926
927
/**
928
 * Fallback for cal_days_in_month
929
 *
930
 * Fallback in case the calendar extension is not loaded in PHP; Only supports Gregorian calendar
931
 */
932
if ( ! function_exists( 'cal_days_in_month' ) ) {
933
	/**
934
	 * cal_days_in_month
935
	 *
936
	 * @param int $calendar
937
	 * @param int $month
938
	 * @param int $year
939
	 *
940
	 * @return bool|string
941
	 */
942
	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...
943
		return date( 't', mktime( 0, 0, 0, $month, 1, $year ) );
944
	}
945
}
946
947
/**
948
 * Get plugin info including status, type, and license validation.
949
 *
950
 * This is an enhanced version of get_plugins() that returns the status
951
 * (`active` or `inactive`) of all plugins, type of plugin (`add-on` or `other`
952
 * and license validation for Give add-ons (`true` or `false`). Does not include
953
 * MU plugins.
954
 *
955
 * @since 1.8.0
956
 *
957
 * @return array Plugin info plus status, type, and license validation if
958
 *               available.
959
 */
960
function give_get_plugins() {
961
	$plugins             = get_plugins();
962
	$active_plugin_paths = (array) get_option( 'active_plugins', array() );
963
964
	if ( is_multisite() ) {
965
		$network_activated_plugin_paths = array_keys( get_site_option( 'active_sitewide_plugins', array() ) );
966
		$active_plugin_paths            = array_merge( $active_plugin_paths, $network_activated_plugin_paths );
967
	}
968
969
	foreach ( $plugins as $plugin_path => $plugin_data ) {
970
		// Is plugin active?
971
		if ( in_array( $plugin_path, $active_plugin_paths ) ) {
972
			$plugins[ $plugin_path ]['Status'] = 'active';
973
		} else {
974
			$plugins[ $plugin_path ]['Status'] = 'inactive';
975
		}
976
977
		$dirname = strtolower( dirname( $plugin_path ) );
978
979
		// Is plugin a Give add-on by WordImpress?
980
		if ( strstr( $dirname, 'give-' ) && strstr( $plugin_data['AuthorURI'], 'wordimpress.com' ) ) {
981
			// Plugin is a Give-addon.
982
			$plugins[ $plugin_path ]['Type'] = 'add-on';
983
984
			// Get license info from database.
985
			$plugin_name    = str_replace( 'Give - ', '', $plugin_data['Name'] );
986
			$db_option      = 'give_' . preg_replace( '/[^a-zA-Z0-9_\s]/', '', str_replace( ' ', '_', strtolower( $plugin_name ) ) ) . '_license_active';
987
			$license_active = get_option( $db_option );
988
989
			// Does a valid license exist?
990
			if ( ! empty( $license_active ) && 'valid' === $license_active->license ) {
991
				$plugins[ $plugin_path ]['License'] = true;
992
			} else {
993
				$plugins[ $plugin_path ]['License'] = false;
994
			}
995
		} else {
996
			// Plugin is not a Give add-on.
997
			$plugins[ $plugin_path ]['Type'] = 'other';
998
		}
999
	}
1000
1001
	return $plugins;
1002
}
1003
1004
1005
/**
1006
 * Check if terms enabled or not for form.
1007
 *
1008
 * @since 1.8
1009
 *
1010
 * @param $form_id
1011
 *
1012
 * @return bool
1013
 */
1014
function give_is_terms_enabled( $form_id ) {
1015
	$form_option = give_get_meta( $form_id, '_give_terms_option', true );
1016
1017
	if (
1018
		give_is_setting_enabled( $form_option, 'global' )
1019
		&& give_is_setting_enabled( give_get_option( 'terms' ) )
1020
	) {
1021
		return true;
1022
1023
	} elseif ( give_is_setting_enabled( $form_option ) ) {
1024
		return true;
1025
1026
	} else {
1027
		return false;
1028
	}
1029
}
1030
1031
1032
/**
1033
 * Delete donation stats cache.
1034
 *
1035
 * @todo  Resolve stats cache key naming issue. Currently it is difficult to regenerate cache key.
1036
 *
1037
 * @since 1.8.7
1038
 *
1039
 * @param string|array $date_range Date for stats.
1040
 *                                 Date value should be in today, yesterday, this_week, last_week, this_month,
1041
 *                                 last_month, this_quarter, last_quarter, this_year, last_year. For date value other,
1042
 *                                 all cache will be removed.
1043
 *
1044
 * @param array        $args
1045
 *
1046
 * @return WP_Error|bool
1047
 */
1048
function give_delete_donation_stats( $date_range = '', $args = array() ) {
1049
1050
	// Delete all cache.
1051
	$status = Give_Cache::delete( Give_Cache::get_options_like( 'give_stats' ) );
1052
1053
	/**
1054
	 * Fire the action when donation stats delete.
1055
	 *
1056
	 * @since 1.8.7
1057
	 *
1058
	 * @param string|array $date_range
1059
	 * @param array        $args
1060
	 */
1061
	do_action( 'give_delete_donation_stats', $status, $date_range, $args );
1062
1063
	return $status;
1064
}
1065
1066
/**
1067
 * Check if admin creating new donation form or not.
1068
 *
1069
 * @since 2.0
1070
 * @return bool
1071
 */
1072
function give_is_add_new_form_page() {
1073
	$status = false;
1074
1075
	if ( false !== strpos( $_SERVER['REQUEST_URI'], '/wp-admin/post-new.php?post_type=give_forms' ) ) {
1076
		$status = true;
1077
	}
1078
1079
	return $status;
1080
}
1081
1082
/**
1083
 * Get Form/Payment meta.
1084
 *
1085
 * @since 1.8.8
1086
 *
1087
 * @param int    $id
1088
 * @param string $meta_key
1089
 * @param bool   $single
1090
 * @param bool   $default
1091
 *
1092
 * @return mixed
1093
 */
1094
function give_get_meta( $id, $meta_key = '', $single = false, $default = false ) {
1095
	/**
1096
	 * Filter the meta value
1097
	 *
1098
	 * @since 1.8.8
1099
	 */
1100
	$meta_value = apply_filters(
1101
		'give_get_meta',
1102
		get_post_meta( $id, $meta_key, $single ),
1103
		$id,
1104
		$meta_key,
1105
		$default
1106
	);
1107
1108
	if (
1109
		( empty( $meta_key ) || empty( $meta_value ) )
1110
		&& $default
1111
	) {
1112
		$meta_value = $default;
1113
	}
1114
1115
	return $meta_value;
1116
}
1117
1118
/**
1119
 * Update Form/Payment meta.
1120
 *
1121
 * @since 1.8.8
1122
 *
1123
 * @param int    $id
1124
 * @param string $meta_key
1125
 * @param mixed  $meta_value
1126
 * @param mixed  $prev_value
1127
 *
1128
 * @return mixed
1129
 */
1130
function give_update_meta( $id, $meta_key, $meta_value, $prev_value = '' ) {
1131
	$status = update_post_meta( $id, $meta_key, $meta_value, $prev_value );
1132
1133
	/**
1134
	 * Filter the meta value update status
1135
	 *
1136
	 * @since 1.8.8
1137
	 */
1138
	return apply_filters( 'give_update_meta', $status, $id, $meta_key, $meta_value );
1139
}
1140
1141
/**
1142
 * Delete Form/Payment meta.
1143
 *
1144
 * @since 1.8.8
1145
 *
1146
 * @param int    $id
1147
 * @param string $meta_key
1148
 * @param string $meta_value
1149
 *
1150
 * @return mixed
1151
 */
1152
function give_delete_meta( $id, $meta_key, $meta_value = '' ) {
1153
	$status = delete_post_meta( $id, $meta_key, $meta_value );
1154
1155
	/**
1156
	 * Filter the meta value delete status
1157
	 *
1158
	 * @since 1.8.8
1159
	 */
1160
	return apply_filters( 'give_delete_meta', $status, $id, $meta_key, $meta_value );
1161
}
1162
1163
/**
1164
 * Check if the upgrade routine has been run for a specific action
1165
 *
1166
 * @since  1.0
1167
 *
1168
 * @param  string $upgrade_action The upgrade action to check completion for
1169
 *
1170
 * @return bool                   If the action has been added to the completed actions array
1171
 */
1172
function give_has_upgrade_completed( $upgrade_action = '' ) {
1173
	// Bailout.
1174
	if ( empty( $upgrade_action ) ) {
1175
		return false;
1176
	}
1177
1178
	// Fresh install?
1179
	// If fresh install then all upgrades will be consider as completed.
1180
	$is_fresh_install = ! get_option( 'give_version' );
1181
	if ( $is_fresh_install ) {
1182
		return true;
1183
	}
1184
0 ignored issues
show
Coding Style introduced by
Functions must not contain multiple empty lines in a row; found 2 empty lines
Loading history...
1185
1186
	$completed_upgrades = give_get_completed_upgrades();
1187
1188
	return in_array( $upgrade_action, $completed_upgrades );
1189
1190
}
1191
1192
/**
1193
 * For use when doing 'stepped' upgrade routines, to see if we need to start somewhere in the middle
1194
 *
1195
 * @since 1.8
1196
 *
1197
 * @return mixed   When nothing to resume returns false, otherwise starts the upgrade where it left off
1198
 */
1199
function give_maybe_resume_upgrade() {
1200
	$doing_upgrade = get_option( 'give_doing_upgrade', false );
1201
	if ( empty( $doing_upgrade ) ) {
1202
		return false;
1203
	}
1204
1205
	return $doing_upgrade;
1206
}
1207
1208
/**
1209
 * Adds an upgrade action to the completed upgrades array
1210
 *
1211
 * @since  1.0
1212
 *
1213
 * @param  string $upgrade_action The action to add to the completed upgrades array
1214
 *
1215
 * @return bool                   If the function was successfully added
1216
 */
1217
function give_set_upgrade_complete( $upgrade_action = '' ) {
1218
1219
	if ( empty( $upgrade_action ) ) {
1220
		return false;
1221
	}
1222
1223
	$completed_upgrades   = give_get_completed_upgrades();
1224
	$completed_upgrades[] = $upgrade_action;
1225
1226
	// Remove any blanks, and only show uniques.
1227
	$completed_upgrades = array_unique( array_values( $completed_upgrades ) );
1228
1229
	/**
1230
	 * Fire the action when any upgrade set to complete.
1231
	 *
1232
	 * @since 1.8.12
1233
	 */
1234
	do_action( 'give_set_upgrade_completed', $upgrade_action, $completed_upgrades );
1235
1236
	return update_option( 'give_completed_upgrades', $completed_upgrades );
1237
}
1238
1239
/**
1240
 * Get's the array of completed upgrade actions
1241
 *
1242
 * @since  1.0
1243
 * @return array The array of completed upgrades
1244
 */
1245
function give_get_completed_upgrades() {
1246
	return (array) get_option( 'give_completed_upgrades' );
1247
}
1248
1249
/**
1250
 * In 2.0 we updated table for log, payment and form.
1251
 *
1252
 * Note: internal purpose only.
1253
 *
1254
 * @since 2.0
1255
 * @global wpdb  $wpdb
1256
 *
1257
 * @param string $type Context for table
1258
 *
1259
 * @return null|array
1260
 */
1261
function __give_v20_bc_table_details( $type ) {
1262
	global $wpdb;
1263
	$table = array();
1264
1265
	// Bailout.
1266
	if ( empty( $type ) ) {
1267
		return null;
1268
	}
1269
1270
	switch ( $type ) {
1271
		case 'form':
1272
			$table['name']         = $wpdb->formmeta;
1273
			$table['column']['id'] = 'form_id';
1274
1275
			break;
1276
1277
		case 'payment':
1278
			$table['name']         = $wpdb->paymentmeta;
1279
			$table['column']['id'] = 'payment_id';
1280
	}
1281
1282
	// Backward compatibility.
1283
	if ( ! give_has_upgrade_completed( 'v20_move_metadata_into_new_table' ) ) {
1284
		$table['name']         = $wpdb->postmeta;
1285
		$table['column']['id'] = 'post_id';
1286
	}
1287
0 ignored issues
show
Coding Style introduced by
Functions must not contain multiple empty lines in a row; found 2 empty lines
Loading history...
1288
1289
	return $table;
1290
}
1291
1292
/**
1293
 * Remove the Give transaction pages from WP search results.
1294
 *
1295
 * @since 1.8.13
1296
 *
1297
 * @param WP_Query $query
1298
 */
1299
function give_remove_pages_from_search( $query ) {
1300
1301
	if ( ! $query->is_admin && $query->is_search && $query->is_main_query() ) {
1302
1303
		$transaction_failed = give_get_option( 'failure_page', 0 );
1304
		$success_page       = give_get_option( 'success_page', 0 );
1305
1306
		$args               = apply_filters(
1307
			'give_remove_pages_from_search', array(
1308
			$transaction_failed,
1309
			$success_page,
1310
		), $query
0 ignored issues
show
Coding Style introduced by
This line of the multi-line function call does not seem to be indented correctly. Expected 12 spaces, but found 8.
Loading history...
1311
		);
1312
		$query->set( 'post__not_in', $args );
1313
	}
1314
}
1315
1316
add_action( 'pre_get_posts', 'give_remove_pages_from_search', 10, 1 );
1317
1318
/**
1319
 * Inserts a new key/value before a key in the array.
1320
 *
1321
 * @since 1.8.13
1322
 *
1323
 * @param string       $key       The key to insert before.
1324
 * @param array        $array     An array to insert in to.
1325
 * @param string       $new_key   The key to insert.
1326
 * @param array|string $new_value An value to insert.
1327
 *
1328
 * @return array The new array if the key exists, the passed array otherwise.
1329
 *
1330
 * @see   array_insert_before()
1331
 */
1332 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...
1333
	if ( array_key_exists( $key, $array ) ) {
1334
		$new = array();
1335
		foreach ( $array as $k => $value ) {
1336
			if ( $k === $key ) {
1337
				$new[ $new_key ] = $new_value;
1338
			}
1339
			$new[ $k ] = $value;
1340
		}
1341
1342
		return $new;
1343
	}
1344
1345
	return $array;
1346
}
1347
1348
/**
1349
 * Inserts a new key/value after a key in the array.
1350
 *
1351
 * @since 1.8.13
1352
 *
1353
 * @param string       $key       The key to insert after.
1354
 * @param array        $array     An array to insert in to.
1355
 * @param string       $new_key   The key to insert.
1356
 * @param array|string $new_value An value to insert.
1357
 *
1358
 * @return array The new array if the key exists, the passed array otherwise.
1359
 *
1360
 * @see   array_insert_before()
1361
 */
1362 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...
1363
	if ( array_key_exists( $key, $array ) ) {
1364
		$new = array();
1365
		foreach ( $array as $k => $value ) {
1366
			$new[ $k ] = $value;
1367
			if ( $k === $key ) {
1368
				$new[ $new_key ] = $new_value;
1369
			}
1370
		}
1371
1372
		return $new;
1373
	}
1374
1375
	return $array;
1376
}
1377
1378
/**
1379
 * Pluck a certain field out of each object in a list.
1380
 *
1381
 * This has the same functionality and prototype of
1382
 * array_column() (PHP 5.5) but also supports objects.
1383
 *
1384
 * @since 1.8.13
1385
 *
1386
 * @param array      $list      List of objects or arrays
1387
 * @param int|string $field     Field from the object to place instead of the entire object
1388
 * @param int|string $index_key Optional. Field from the object to use as keys for the new array.
1389
 *                              Default null.
1390
 *
1391
 * @return array Array of found values. If `$index_key` is set, an array of found values with keys
1392
 *               corresponding to `$index_key`. If `$index_key` is null, array keys from the original
1393
 *               `$list` will be preserved in the results.
1394
 */
1395
function give_list_pluck( $list, $field, $index_key = null ) {
1396
1397
	if ( ! $index_key ) {
1398
		/*
1399
		 * This is simple. Could at some point wrap array_column()
1400
		 * if we knew we had an array of arrays.
1401
		 */
1402
		foreach ( $list as $key => $value ) {
1403
			if ( is_object( $value ) ) {
1404
				if ( isset( $value->$field ) ) {
1405
					$list[ $key ] = $value->$field;
1406
				}
1407
			} else {
1408
				if ( isset( $value[ $field ] ) ) {
1409
					$list[ $key ] = $value[ $field ];
1410
				}
1411
			}
1412
		}
1413
1414
		return $list;
1415
	}
1416
1417
	/*
1418
	 * When index_key is not set for a particular item, push the value
1419
	 * to the end of the stack. This is how array_column() behaves.
1420
	 */
1421
	$newlist = array();
1422
	foreach ( $list as $value ) {
1423
		if ( is_object( $value ) ) {
1424
			if ( isset( $value->$index_key ) ) {
1425
				$newlist[ $value->$index_key ] = $value->$field;
1426
			} else {
1427
				$newlist[] = $value->$field;
1428
			}
1429
		} else {
1430
			if ( isset( $value[ $index_key ] ) ) {
1431
				$newlist[ $value[ $index_key ] ] = $value[ $field ];
1432
			} else {
1433
				$newlist[] = $value[ $field ];
1434
			}
1435
		}
1436
	}
1437
1438
	$list = $newlist;
1439
1440
	return $list;
1441
}
1442
1443
/**
1444
 * Add meta data field to a donor.
1445
 *
1446
 * @since 1.8.13
1447
 *
1448
 * @param int    $donor_id   Donor ID.
1449
 * @param string $meta_key   Metadata name.
1450
 * @param mixed  $meta_value Metadata value. Must be serializable if non-scalar.
1451
 * @param bool   $unique     Optional. Whether the same key should not be added.
1452
 *                           Default false.
1453
 *
1454
 * @return int|false Meta ID on success, false on failure.
1455
 */
1456
function add_donor_meta( $donor_id, $meta_key, $meta_value, $unique = false ) {
1457
	return add_metadata( 'give_customer', $donor_id, $meta_key, $meta_value, $unique );
1458
}
1459
1460
/**
1461
 * Remove metadata matching criteria from a Donor meta.
1462
 *
1463
 * You can match based on the key, or key and value. Removing based on key and
1464
 * value, will keep from removing duplicate metadata with the same key. It also
1465
 * allows removing all metadata matching key, if needed.
1466
 *
1467
 * @since 1.8.13
1468
 *
1469
 * @param int    $donor_id   Donor ID
1470
 * @param string $meta_key   Metadata name.
1471
 * @param mixed  $meta_value Optional. Metadata value.
1472
 *
1473
 * @return bool True on success, false on failure.
1474
 */
1475
function delete_donor_meta( $donor_id, $meta_key, $meta_value = '' ) {
1476
	return delete_metadata( 'give_customer', $donor_id, $meta_key, $meta_value );
1477
}
1478
1479
/**
1480
 * Retrieve donor meta field for a donor meta table.
1481
 *
1482
 * @since 1.8.13
1483
 *
1484
 * @param int    $donor_id Donor ID.
1485
 * @param string $key      Optional. The meta key to retrieve. By default, returns data for all keys.
1486
 * @param bool   $single   Whether to return a single value.
1487
 *
1488
 * @return mixed Will be an array if $single is false. Will be value of meta data field if $single
1489
 *  is true.
1490
 */
1491
function get_donor_meta( $donor_id, $key = '', $single = false ) {
1492
	return get_metadata( 'give_customer', $donor_id, $key, $single );
1493
}
1494
1495
/**
1496
 * Update customer meta field based on Donor ID.
1497
 *
1498
 * If the meta field for the donor does not exist, it will be added.
1499
 *
1500
 * @since 1.8.13
1501
 *
1502
 * @param int    $donor_id   Donor ID.
1503
 * @param string $meta_key   Metadata key.
1504
 * @param mixed  $meta_value Metadata value.
1505
 * @param mixed  $prev_value Optional. Previous value to check before removing.
1506
 *
1507
 * @return int|bool Meta ID if the key didn't exist, true on successful update, false on failure.
1508
 */
1509
function update_donor_meta( $donor_id, $meta_key, $meta_value, $prev_value = '' ) {
1510
	return update_metadata( 'give_customer', $donor_id, $meta_key, $meta_value, $prev_value );
1511
}
1512
1513
1514
/**
1515
 * Give recalculate income and donation of the donation from ID
1516
 *
1517
 * @since 1.8.13
1518
 *
1519
 * @param int $form_id Form id of which recalculation needs to be done.
1520
 *
1521
 * @return void
1522
 */
1523
function give_recount_form_income_donation( $form_id = 0 ) {
1524
	// Check if form id is not empty.
1525
	if ( ! empty( $form_id ) ) {
1526
		/**
1527
		 * Filter to modify payment status.
1528
		 *
1529
		 * @since 1.8.13
1530
		 */
1531
		$accepted_statuses = apply_filters( 'give_recount_accepted_statuses', array( 'publish' ) );
1532
1533
		/**
1534
		 * Filter to modify args of payment query before recalculating the form total
1535
		 *
1536
		 * @since 1.8.13
1537
		 */
1538
		$args = apply_filters(
1539
			'give_recount_form_stats_args', array(
1540
				'give_forms'     => $form_id,
1541
				'status'         => $accepted_statuses,
1542
				'posts_per_page' => - 1,
1543
				'fields'         => 'ids',
1544
			)
1545
		);
1546
1547
		$totals = array(
1548
			'sales'    => 0,
1549
			'earnings' => 0,
1550
		);
1551
1552
		$payments = new Give_Payments_Query( $args );
1553
		$payments = $payments->get_payments();
1554
1555
		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...
1556 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...
1557
				// Ensure acceptible status only
1558
				if ( ! in_array( $payment->post_status, $accepted_statuses ) ) {
1559
					continue;
1560
				}
1561
1562
				// Ensure only payments for this form are counted
1563
				if ( $payment->form_id != $form_id ) {
1564
					continue;
1565
				}
1566
1567
				$totals['sales'] ++;
1568
				$totals['earnings'] += $payment->total;
1569
1570
			}
1571
		}
1572
		give_update_meta( $form_id, '_give_form_sales', $totals['sales'] );
1573
		give_update_meta( $form_id, '_give_form_earnings', give_sanitize_amount_for_db( $totals['earnings'] ) );
1574
	}// End if().
1575
}
1576
1577
1578
/**
1579
 * Get attribute string
1580
 *
1581
 * @since 1.8.17
1582
 *
1583
 * @param array $attributes
1584
 *
1585
 * @return string
1586
 */
1587
function give_get_attribute_str( $attributes ) {
1588
	$attribute_str = '';
1589
1590
	if ( empty( $attributes ) ) {
1591
		return $attribute_str;
1592
	}
1593
1594
	foreach ( $attributes as $tag => $value ) {
1595
		$attribute_str .= " {$tag}=\"{$value}\"";
1596
	}
1597
1598
	return trim( $attribute_str );
1599
}
1600
1601
/**
1602
 * Get the upload dir path
1603
 *
1604
 * @since 1.8.17
1605
 *
1606
 * @return string $wp_upload_dir;
1607
 */
1608
function give_get_wp_upload_dir() {
1609
	$wp_upload_dir = wp_upload_dir();
1610
1611
	return ( ! empty( $wp_upload_dir['path'] ) ? $wp_upload_dir['path'] : false );
1612
}
1613
1614
/**
1615
 * Get the data from uploaded JSON file
1616
 *
1617
 * @since 1.8.17
1618
 *
1619
 * @param string $file_name filename of the json file that is being uploaded
1620
 *
1621
 * @return string/bool $file_contents File content
0 ignored issues
show
Documentation introduced by
The doc-type string/bool could not be parsed: Unknown type name "string/bool" at position 0. (view supported doc-types)

This check marks PHPDoc comments that could not be parsed by our parser. To see which comment annotations we can parse, please refer to our documentation on supported doc-types.

Loading history...
1622
 */
1623
function give_get_core_settings_json( $file_name ) {
1624
	$upload_dir = give_get_wp_upload_dir();
1625
	$file_path  = $upload_dir . '/' . $file_name;
1626
1627
	if ( is_wp_error( $file_path ) || empty( $file_path ) ) {
1628
		Give_Admin_Settings::add_error( 'give-import-csv', __( 'Please upload or provide a valid JSON file.', 'give' ) );
1629
	}
1630
1631
	$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...
1632
1633
	return $file_contents;
1634
}
1635
1636
/**
1637
 * Get number of donation to show when user is not login.
1638
 *
1639
 * @since 1.8.17
1640
 *
1641
 * @return int $country The two letter country code for the site's base country
1642
 */
1643
function give_get_limit_display_donations() {
1644
	return give_get_option( 'limit_display_donations', 1 );
1645
}
1646
1647
/**
1648
 * Add footer to the table when donor is view the donation history page with out login
1649
 *
1650
 * @since 1.8.17
1651
 */
1652
function give_donation_history_table_end() {
1653
	$email = Give()->session->get( 'give_email' );
1654
	?>
1655
	<tfoot>
1656
	<tr>
1657
		<td colspan="9999">
1658
			<div class="give-security-wrap">
1659
				<div class="give-security-column give-security-description-wrap">
1660
					<?php
1661
					echo sprintf(
0 ignored issues
show
introduced by
Expected a sanitizing function (see Codex for 'Data Validation'), but instead saw 'sprintf'
Loading history...
1662
						__( 'For security reasons, please confirm your email address (%s) to view your complete donation history.', 'give' ),
1663
						$email
1664
					);
1665
					?>
1666
				</div>
1667
				<div class="give-security-column give-security-button-wrap">
1668
					<a href="#" data-email="<?php echo $email; ?>" id="give-confirm-email-btn" class="give-confirm-email-btn give-btn">
0 ignored issues
show
introduced by
Expected next thing to be a escaping function, not '$email'
Loading history...
1669
						<?php _e( 'Confirm Email', 'give' ); ?>
1670
					</a>
1671
					<span><?php _e( 'Email Sent!', 'give' ); ?></span>
1672
				</div>
1673
			</div>
1674
		</td>
1675
	</tr>
1676
	</tfoot>
1677
	<?php
1678
}
1679
1680
1681
/**
1682
 * Wrapper for _doing_it_wrong.
1683
 *
1684
 * @since  1.8.18
1685
 * @param  string $function
1686
 * @param  string $message
1687
 * @param  string $version
1688
 *
1689
 * @return void
1690
 */
1691
function give_doing_it_wrong( $function, $message, $version ) {
1692
	$message .= "\nBacktrace:" . wp_debug_backtrace_summary();
1693
1694
	_doing_it_wrong( $function, $message , $version );
1695
}