PluginProbe ʕ •ᴥ•ʔ
GiveWP – Donation Plugin and Fundraising Platform / 2.13.3
GiveWP – Donation Plugin and Fundraising Platform v2.13.3
4.16.2 4.16.1 4.16.0 4.15.5 4.15.4 4.15.3 4.15.2 4.15.1 4.15.0 2.3.0 2.3.1 2.3.2 2.30.0 2.31.0 2.31.1 2.32.0 2.33.0 2.33.1 2.33.2 2.33.3 2.33.4 2.33.5 2.4.0 2.4.1 2.4.2 2.4.3 2.4.4 2.4.5 2.4.6 2.4.7 2.5.0 2.5.1 2.5.10 2.5.11 2.5.12 2.5.13 2.5.2 2.5.3 2.5.4 2.5.5 2.5.6 2.5.7 2.5.8 2.5.9 2.6.0 2.6.1 2.6.2 2.6.3 2.7.0 2.7.1 2.7.2 2.7.3 2.7.4 2.7.5 2.8.0 2.8.1 2.9.0 2.9.1 2.9.2 2.9.3 2.9.4 2.9.5 2.9.6 2.9.7 3.0.0 3.0.1 3.0.2 3.0.3 3.0.4 3.1.0 3.1.1 3.1.2 3.10.0 3.11.0 3.12.0 3.12.1 3.12.2 3.12.3 3.13.0 3.14.0 3.14.1 3.14.2 3.15.0 3.15.1 3.16.0 3.16.1 3.16.2 3.16.3 3.16.4 3.16.5 3.17.0 3.17.1 3.17.2 3.18.0 3.19.0 3.19.1 3.19.2 3.19.3 3.19.4 3.2.0 3.2.1 3.2.2 3.20.0 3.21.0 3.21.1 3.22.0 3.22.1 3.22.2 3.3.0 3.3.1 3.4.0 3.4.1 3.4.2 3.5.0 3.5.1 3.6.0 3.6.1 3.6.2 3.7.0 3.8.0 3.9.0 4.0.0 4.1.0 4.1.1 4.10.0 4.10.1 4.11.0 4.12.0 4.13.0 4.13.1 4.13.2 4.14.0 4.14.1 4.14.2 4.14.3 4.14.4 4.14.5 4.14.6 4.2.0 4.2.1 4.3.0 4.3.1 4.3.2 4.4.0 4.5.0 4.6.1 4.7.0 4.7.1 4.8.0 4.8.1 4.9.0 trunk 1.9.0 2.0.0 2.0.1 2.0.2 2.0.3 2.0.4 2.0.5 2.0.6 2.0.7 2.1.0 2.1.1 2.1.2 2.1.3 2.1.4 2.1.5 2.1.6 2.1.7 2.1.8 2.10.0 2.10.1 2.10.2 2.10.3 2.10.4 2.11.0 2.11.1 2.11.2 2.11.3 2.12.0 2.12.1 2.12.2 2.12.3 2.13.0 2.13.1 2.13.2 2.13.3 2.13.4 2.14.0 2.15.0 2.16.0 2.16.1 2.17.0 2.17.1 2.17.3 2.18.0 2.18.1 2.19.1 2.19.2 2.19.3 2.19.4 2.19.5 2.19.6 2.19.7 2.19.8 2.2.0 2.2.1 2.2.2 2.2.3 2.2.4 2.2.5 2.2.6 2.20.0 2.20.1 2.20.2 2.21.0 2.21.1 2.21.2 2.21.3 2.21.4 2.22.0 2.22.1 2.22.2 2.22.3 2.23.0 2.23.1 2.23.2 2.24.0 2.24.1 2.24.2 2.25.0 2.25.1 2.25.2 2.25.3 2.26.0 2.27.0 2.27.1 2.27.2 2.27.3 2.28.0 2.29.0 2.29.1 2.29.2
give / includes / admin / tools / export / give-export-donations-exporter.php
give / includes / admin / tools / export Last commit date
class-batch-export-donors.php 5 years ago class-batch-export.php 5 years ago class-core-settings-export.php 7 years ago class-export-earnings.php 5 years ago class-export.php 6 years ago class-give-export-donations.php 5 years ago export-actions.php 5 years ago export-functions.php 5 years ago give-export-donations-exporter.php 6 years ago give-export-donations-functions.php 5 years ago pdf-reports.php 5 years ago
give-export-donations-exporter.php
672 lines
1 <?php
2 /**
3 * Payments Export Class.
4 *
5 * This class handles payment export in batches.
6 *
7 * @package Give
8 * @subpackage Admin/Reports
9 * @copyright Copyright (c) 2016, GiveWP
10 * @license https://opensource.org/licenses/gpl-license GNU Public License
11 * @since 2.1
12 */
13
14 // Exit if accessed directly.
15 if ( ! defined( 'ABSPATH' ) ) {
16 exit;
17 }
18
19 /**
20 * Give_Export_Donations_CSV Class
21 *
22 * @since 2.1
23 */
24 class Give_Export_Donations_CSV extends Give_Batch_Export {
25
26 /**
27 * Our export type. Used for export-type specific filters/actions.
28 *
29 * @since 2.1
30 *
31 * @var string
32 */
33 public $export_type = 'payments';
34
35 /**
36 * Form submission data.
37 *
38 * @since 2.1
39 *
40 * @var array
41 */
42 public $data = array();
43
44 /**
45 * Form submission data.
46 *
47 * @since 2.1
48 *
49 * @var array
50 */
51 public $cols = array();
52
53 /**
54 * Form ID.
55 *
56 * @since 2.1
57 *
58 * @var string
59 */
60 public $form_id = '';
61
62 /**
63 * Form tags ids.
64 *
65 * @since 2.1
66 *
67 * @var array
68 */
69 public $tags = '';
70
71
72 /**
73 * Form categories ids.
74 *
75 * @since 2.1
76 *
77 * @var array
78 */
79 public $categories = '';
80
81 /**
82 * Set the properties specific to the export.
83 *
84 * @since 2.1
85 *
86 * @param array $request The Form Data passed into the batch processing.
87 */
88 public function set_properties( $request ) {
89
90 // Set data from form submission
91 if ( isset( $_POST['form'] ) ) {
92 $this->data = give_clean( wp_parse_args( $_POST['form'] ) );
93 }
94
95 $this->form = $this->data['forms'];
96 $this->categories = ! empty( $request['give_forms_categories'] ) ? (array) $request['give_forms_categories'] : array();
97 $this->tags = ! empty( $request['give_forms_tags'] ) ? (array) $request['give_forms_tags'] : array();
98 $this->form_id = $this->get_form_ids( $request );
99 $this->price_id = isset( $request['give_price_option'] ) && ! in_array( $this->price_id, array( 'all', '' ) ) ? absint( $request['give_price_option'] ) : null;
100 $this->start = ! empty( $request['start'] ) ? date( 'Y-m-d', strtotime( $request['start'] ) ) : '';
101 $this->end = ! empty( $request['end'] ) ? date( 'Y-m-d', strtotime( $request['end'] ) ) : '';
102 $this->status = isset( $request['status'] ) ? sanitize_text_field( $request['status'] ) : 'complete';
103
104 /**
105 * Hook to use after setting properties.
106 *
107 * @since 2.1.3
108 */
109 do_action( 'give_export_donations_form_data', $this->data );
110 }
111
112 /**
113 * Get donation form id list
114 *
115 * @since 2.1
116 *
117 * @param array $request form data that need to be exported
118 *
119 * @return array|boolean|null $form get all the donation id that need to be exported
120 */
121 public function get_form_ids( $request = array() ) {
122 $form = ! empty( $request['forms'] ) && 0 !== $request['forms'] ? absint( $request['forms'] ) : null;
123
124 $form_ids = ! empty( $request['form_ids'] ) ? sanitize_text_field( $request['form_ids'] ) : null;
125
126 if ( empty( $form ) && ! empty( $form_ids ) && ( ! empty( $this->categories ) || ! empty( $this->tags ) ) ) {
127 $form = explode( ',', $form_ids );
128 }
129
130 return $form;
131 }
132
133 /**
134 * Set the CSV columns.
135 *
136 * @access public
137 *
138 * @since 2.1
139 *
140 * @return array|bool $cols All the columns.
141 */
142 public function csv_cols() {
143
144 $columns = isset( $this->data['give_give_donations_export_option'] ) ? $this->data['give_give_donations_export_option'] : array();
145
146 // We need columns.
147 if ( empty( $columns ) ) {
148 return false;
149 }
150
151 $this->cols = $this->get_cols( $columns );
152
153 return $this->cols;
154 }
155
156
157 /**
158 * CSV file columns.
159 *
160 * @since 2.1
161 *
162 * @param array $columns
163 *
164 * @return array
165 */
166 private function get_cols( $columns ) {
167
168 $cols = array();
169
170 foreach ( $columns as $key => $value ) {
171
172 switch ( $key ) {
173 case 'donation_id':
174 $cols['donation_id'] = __( 'Donation ID', 'give' );
175 break;
176 case 'seq_id':
177 $cols['seq_id'] = __( 'Donation Number', 'give' );
178 break;
179 case 'title_prefix':
180 $cols['title_prefix'] = __( 'Title Prefix', 'give' );
181 break;
182 case 'first_name':
183 $cols['first_name'] = __( 'First Name', 'give' );
184 break;
185 case 'last_name':
186 $cols['last_name'] = __( 'Last Name', 'give' );
187 break;
188 case 'email':
189 $cols['email'] = __( 'Email Address', 'give' );
190 break;
191 case 'company':
192 $cols['company'] = __( 'Company Name', 'give' );
193 break;
194 case 'address':
195 $cols['address_line1'] = __( 'Address 1', 'give' );
196 $cols['address_line2'] = __( 'Address 2', 'give' );
197 $cols['address_city'] = __( 'City', 'give' );
198 $cols['address_state'] = __( 'State', 'give' );
199 $cols['address_zip'] = __( 'Zip', 'give' );
200 $cols['address_country'] = __( 'Country', 'give' );
201 break;
202 case 'comment':
203 $cols['comment'] = __( 'Donor Comment', 'give' );
204 break;
205 case 'donation_total':
206 $cols['donation_total'] = __( 'Donation Total', 'give' );
207 break;
208 case 'currency_code':
209 $cols['currency_code'] = __( 'Currency Code', 'give' );
210 break;
211 case 'currency_symbol':
212 $cols['currency_symbol'] = __( 'Currency Symbol', 'give' );
213 break;
214 case 'donation_status':
215 $cols['donation_status'] = __( 'Donation Status', 'give' );
216 break;
217 case 'payment_gateway':
218 $cols['payment_gateway'] = __( 'Payment Gateway', 'give' );
219 break;
220 case 'payment_mode':
221 $cols['payment_mode'] = __( 'Payment Mode', 'give' );
222 break;
223 case 'form_id':
224 $cols['form_id'] = __( 'Form ID', 'give' );
225 break;
226 case 'form_title':
227 $cols['form_title'] = __( 'Form Title', 'give' );
228 break;
229 case 'form_level_id':
230 $cols['form_level_id'] = __( 'Level ID', 'give' );
231 break;
232 case 'form_level_title':
233 $cols['form_level_title'] = __( 'Level Title', 'give' );
234 break;
235 case 'donation_date':
236 $cols['donation_date'] = __( 'Donation Date', 'give' );
237 break;
238 case 'donation_time':
239 $cols['donation_time'] = __( 'Donation Time', 'give' );
240 break;
241 case 'userid':
242 $cols['userid'] = __( 'User ID', 'give' );
243 break;
244 case 'donorid':
245 $cols['donorid'] = __( 'Donor ID', 'give' );
246 break;
247 case 'donor_ip':
248 $cols['donor_ip'] = __( 'Donor IP Address', 'give' );
249 break;
250 case 'donation_note_private':
251 $cols['donation_note_private'] = __( 'Donation Note (private)', 'give' );
252 break;
253 case 'donation_note_to_donor':
254 $cols['donation_note_to_donor'] = __( 'Donation Note (to donor)', 'give' );
255 break;
256 default:
257 $cols[ $key ] = $key;
258
259 }
260 }
261
262 /**
263 * Filter to get columns name when exporting donation
264 *
265 * @since 2.1
266 *
267 * @param array $cols columns name for CSV
268 * @param array $columns columns select by admin to export
269 */
270 return (array) apply_filters( 'give_export_donation_get_columns_name', $cols, $columns );
271 }
272
273 /**
274 * Get the donation argument
275 *
276 * @since 2.1
277 *
278 * @param array $args donation argument
279 *
280 * @return array $args donation argument
281 */
282 public function get_donation_argument( $args = array() ) {
283 $defaults = array(
284 'number' => 30,
285 'page' => $this->step,
286 'status' => $this->status,
287 );
288 // Date query.
289 if ( ! empty( $this->start ) || ! empty( $this->end ) ) {
290 if ( ! empty( $this->start ) ) {
291 $defaults['date_query'][0]['after'] = "{$this->start} 00:00:00";
292 }
293 if ( ! empty( $this->end ) ) {
294 $defaults['date_query'][0]['before'] = "{$this->end} 23:59:59";
295 }
296 }
297
298 if ( ! empty( $this->form_id ) ) {
299 $defaults['give_forms'] = is_array( $this->form_id ) ? $this->form_id : array( $this->form_id );
300 }
301
302 /**
303 * Filter to modify Payment Query arguments for exporting
304 * donations.
305 *
306 * @since 2.1.3
307 */
308 return apply_filters( 'give_export_donations_donation_query_args', wp_parse_args( $args, $defaults ) );
309 }
310
311 /**
312 * Get the Export Data.
313 *
314 * @access public
315 *
316 * @since 2.1
317 *
318 * @global object $wpdb Used to query the database using the WordPress database API.
319 *
320 * @return array $data The data for the CSV file.
321 */
322 public function get_data() {
323
324 $data = array();
325 $i = 0;
326 // Payment query.
327 $payments = give_get_payments( $this->get_donation_argument() );
328
329 if ( $payments ) {
330
331 foreach ( $payments as $payment ) {
332
333 $columns = $this->csv_cols();
334 $payment = new Give_Payment( $payment->ID );
335 $payment_meta = $payment->payment_meta;
336 $address = $payment->address;
337
338 // Set columns.
339 if ( ! empty( $columns['donation_id'] ) ) {
340 $data[ $i ]['donation_id'] = $payment->ID;
341 }
342
343 if ( ! empty( $columns['seq_id'] ) ) {
344 $data[ $i ]['seq_id'] = Give()->seq_donation_number->get_serial_code( $payment->ID );
345 }
346
347 if ( ! empty( $columns['title_prefix'] ) ) {
348 $data[ $i ]['title_prefix'] = ! empty( $payment->title_prefix ) ? $payment->title_prefix : '';
349 }
350
351 if ( ! empty( $columns['first_name'] ) ) {
352 $data[ $i ]['first_name'] = isset( $payment->first_name ) ? $payment->first_name : '';
353 }
354
355 if ( ! empty( $columns['last_name'] ) ) {
356 $data[ $i ]['last_name'] = isset( $payment->last_name ) ? $payment->last_name : '';
357 }
358
359 if ( ! empty( $columns['email'] ) ) {
360 $data[ $i ]['email'] = $payment->email;
361 }
362
363 if ( ! empty( $columns['company'] ) ) {
364 $data[ $i ]['company'] = empty( $payment_meta['_give_donation_company'] ) ? '' : str_replace( "\'", "'", $payment_meta['_give_donation_company'] );
365 }
366
367 if ( ! empty( $columns['address_line1'] ) ) {
368 $data[ $i ]['address_line1'] = isset( $address['line1'] ) ? $address['line1'] : '';
369 $data[ $i ]['address_line2'] = isset( $address['line2'] ) ? $address['line2'] : '';
370 $data[ $i ]['address_city'] = isset( $address['city'] ) ? $address['city'] : '';
371 $data[ $i ]['address_state'] = isset( $address['state'] ) ? $address['state'] : '';
372 $data[ $i ]['address_zip'] = isset( $address['zip'] ) ? $address['zip'] : '';
373 $data[ $i ]['address_country'] = isset( $address['country'] ) ? $address['country'] : '';
374 }
375
376 if ( ! empty( $columns['comment'] ) ) {
377 $comment = give_get_donor_donation_comment( $payment->ID, $payment->donor_id );
378 $data[ $i ]['comment'] = ! empty( $comment ) ? $comment->comment_content : '';
379 }
380
381 if ( ! empty( $columns['donation_total'] ) ) {
382 $data[ $i ]['donation_total'] = give_format_amount( give_donation_amount( $payment->ID ) );
383 }
384
385 if ( ! empty( $columns['currency_code'] ) ) {
386 $data[ $i ]['currency_code'] = empty( $payment_meta['_give_payment_currency'] ) ? give_get_currency() : $payment_meta['_give_payment_currency'];
387 }
388
389 if ( ! empty( $columns['currency_symbol'] ) ) {
390 $currency_code = $data[ $i ]['currency_code'];
391 $data[ $i ]['currency_symbol'] = give_currency_symbol( $currency_code, true );
392 }
393
394 if ( ! empty( $columns['donation_status'] ) ) {
395 $data[ $i ]['donation_status'] = give_get_payment_status( $payment, true );
396 }
397
398 if ( ! empty( $columns['payment_gateway'] ) ) {
399 $data[ $i ]['payment_gateway'] = $payment->gateway;
400 }
401
402 if ( ! empty( $columns['payment_mode'] ) ) {
403 $data[ $i ]['payment_mode'] = $payment->mode;
404 }
405
406 if ( ! empty( $columns['form_id'] ) ) {
407 $data[ $i ]['form_id'] = $payment->form_id;
408 }
409
410 if ( ! empty( $columns['form_title'] ) ) {
411 $data[ $i ]['form_title'] = get_the_title( $payment->form_id );
412 }
413
414 if ( ! empty( $columns['form_level_id'] ) ) {
415 $data[ $i ]['form_level_id'] = $payment->price_id;
416 }
417
418 if ( ! empty( $columns['form_level_title'] ) ) {
419 $var_prices = give_has_variable_prices( $payment->form_id );
420 if ( empty( $var_prices ) ) {
421 $data[ $i ]['form_level_title'] = '';
422 } else {
423 if ( 'custom' === $payment->price_id ) {
424 $custom_amount_text = give_get_meta( $payment->form_id, '_give_custom_amount_text', true );
425
426 if ( empty( $custom_amount_text ) ) {
427 $custom_amount_text = esc_html__( 'Custom Amount', 'give' );
428 }
429 $data[ $i ]['form_level_title'] = $custom_amount_text;
430 } else {
431 $data[ $i ]['form_level_title'] = give_get_price_option_name( $payment->form_id, $payment->price_id );
432 }
433 }
434 }
435
436 if ( ! empty( $columns['donation_date'] ) ) {
437 $payment_date = strtotime( $payment->date );
438 $data[ $i ]['donation_date'] = date( give_date_format(), $payment_date );
439 }
440
441 if ( ! empty( $columns['donation_time'] ) ) {
442 $payment_date = strtotime( $payment->date );
443 $data[ $i ]['donation_time'] = date_i18n( 'H', $payment_date ) . ':' . date( 'i', $payment_date );
444 }
445
446 if ( ! empty( $columns['userid'] ) ) {
447 $data[ $i ]['userid'] = $payment->user_id;
448 }
449
450 if ( ! empty( $columns['donorid'] ) ) {
451 $data[ $i ]['donorid'] = $payment->customer_id;
452 }
453
454 if ( ! empty( $columns['donor_ip'] ) ) {
455 $data[ $i ]['donor_ip'] = give_get_payment_user_ip( $payment->ID );
456 }
457
458 if ( ! empty( $columns['donation_note_private'] ) ) {
459 $comments = Give()->comment->db->get_comments(
460 array(
461 'comment_parent' => $payment->ID,
462 'comment_type' => 'donation',
463 'meta_query' => array(
464 'relation' => 'OR',
465 array(
466 'key' => 'note_type',
467 'compare' => 'NOT EXISTS',
468 ),
469 array(
470 'key' => 'note_type',
471 'value' => 'donor',
472 'compare' => '!=',
473 ),
474 ),
475 )
476 );
477
478 $comment_html = array();
479
480 if ( ! empty( $comments ) ) {
481 foreach ( $comments as $comment ) {
482 $comment_html[] = sprintf(
483 '%s - %s',
484 date( 'Y-m-d', strtotime( $comment->comment_date ) ),
485 $comment->comment_content
486 );
487 }
488 }
489
490 $data[ $i ]['donation_note_private'] = implode( "\n", $comment_html );
491 }
492
493 if ( ! empty( $columns['donation_note_to_donor'] ) ) {
494 $comments = Give()->comment->db->get_comments(
495 array(
496 'comment_parent' => $payment->ID,
497 'comment_type' => 'donation',
498 'meta_query' => array(
499 array(
500 'key' => 'note_type',
501 'value' => 'donor',
502 ),
503 ),
504 )
505 );
506
507 $comment_html = array();
508
509 if ( ! empty( $comments ) ) {
510 foreach ( $comments as $comment ) {
511 $comment_html[] = sprintf(
512 '%s - %s',
513 date( 'Y-m-d', strtotime( $comment->comment_date ) ),
514 $comment->comment_content
515 );
516 }
517 }
518 $data[ $i ]['donation_note_to_donor'] = implode( "\n", $comment_html );
519 }
520
521 // Add custom field data.
522 // First we remove the standard included keys from above.
523 $remove_keys = array(
524 'donation_id',
525 'seq_id',
526 'first_name',
527 'last_name',
528 'email',
529 'address_line1',
530 'address_line2',
531 'address_city',
532 'address_state',
533 'address_zip',
534 'address_country',
535 'donation_total',
536 'payment_gateway',
537 'payment_mode',
538 'form_id',
539 'form_title',
540 'form_level_id',
541 'form_level_title',
542 'donation_date',
543 'donation_time',
544 'userid',
545 'donorid',
546 'donor_ip',
547 );
548
549 // Removing above keys...
550 foreach ( $remove_keys as $key ) {
551 unset( $columns[ $key ] );
552 }
553
554 // Now loop through remaining meta fields.
555 foreach ( $columns as $col ) {
556 $field_data = get_post_meta( $payment->ID, $col, true );
557 $data[ $i ][ $col ] = $field_data;
558 unset( $columns[ $col ] );
559 }
560
561 /**
562 * Filter to modify Donation CSV data when exporting donation
563 *
564 * @since 2.1
565 *
566 * @param array Donation data
567 * @param Give_Payment $payment Instance of Give_Payment
568 * @param array $columns Donation data $columns that are not being merge
569 * @param Give_Export_Donations_CSV $this Instance of Give_Export_Donations_CSV
570 *
571 * @return array Donation data
572 */
573 $data[ $i ] = apply_filters( 'give_export_donation_data', $data[ $i ], $payment, $columns, $this );
574
575 $new_data = array();
576 $old_data = $data[ $i ];
577
578 // sorting the columns bas on row
579 foreach ( $this->csv_cols() as $key => $value ) {
580 if ( array_key_exists( $key, $old_data ) ) {
581 $new_data[ $key ] = $old_data[ $key ];
582 }
583 }
584
585 $data[ $i ] = $new_data;
586
587 // Increment iterator.
588 $i ++;
589
590 }
591
592 $data = apply_filters( 'give_export_get_data', $data );
593 $data = apply_filters( "give_export_get_data_{$this->export_type}", $data );
594
595 return $data;
596
597 }
598
599 return array();
600
601 }
602
603 /**
604 * Return the calculated completion percentage.
605 *
606 * @since 2.1
607 *
608 * @return int
609 */
610 public function get_percentage_complete() {
611 $args = $this->get_donation_argument(
612 array(
613 'number' => - 1,
614 'output' => '',
615 )
616 );
617 if ( isset( $args['page'] ) ) {
618 unset( $args['page'] );
619 }
620 $query = give_get_payments( $args );
621 $total = count( $query );
622 $percentage = 100;
623 if ( $total > 0 ) {
624 $percentage = ( ( 30 * $this->step ) / $total ) * 100;
625 }
626 if ( $percentage > 100 ) {
627 $percentage = 100;
628 }
629
630 return $percentage;
631 }
632
633 /**
634 * Print the CSV rows for the current step.
635 *
636 * @access public
637 *
638 * @since 2.1
639 *
640 * @return string|false
641 */
642 public function print_csv_rows() {
643
644 $row_data = '';
645 $data = $this->get_data();
646 $cols = $this->get_csv_cols();
647
648 if ( $data ) {
649
650 // Output each row
651 foreach ( $data as $row ) {
652 $i = 1;
653 foreach ( $row as $col_id => $column ) {
654 // Make sure the column is valid
655 if ( array_key_exists( $col_id, $cols ) ) {
656 $row_data .= '"' . preg_replace( '/"/', "'", $column ) . '"';
657 $row_data .= $i == count( $cols ) ? '' : ',';
658 $i ++;
659 }
660 }
661 $row_data .= "\r\n";
662 }
663
664 $this->stash_step_data( $row_data );
665
666 return $row_data;
667 }
668
669 return false;
670 }
671 }
672