I’ve built a plugin for a client so that they can download data as a CSV file. It’s been set up so that when the user clicks on a link in the menu, the CSV should just automatically download. However, it doesn’t quite work like that, and just loads the function as a page in the WordPress backend.
This is the code I have for the function:
function download_payment_csv() {
include 'lib/connection.php';
$csv_output = '';
$values = $db->query('SELECT * FROM tbPayments ORDER BY date DESC');
$i=0;
while ($rowr = mysql_fetch_row($values)) {
for ($j=0;$j<$i;$j++) {
$csv_output .= $rowr[$j].",";
}
$csv_output .= "n";
}
header("Pragma: public");
header("Expires: 0");
header("Cache-Control: must-revalidate, post-check=0, pre-check=0");
header("Cache-Control: private", false);
header("Content-Type: application/octet-stream");
header("Content-Disposition: attachment; filename="report.csv";" );
header("Content-Transfer-Encoding: binary");
echo $csv_output;
}
And as I said, it just returns a blank screen. Any help would be appreciated!
EDIT
So this is the code I’m now working with, taking bits from what’s been said already.
function download_payment_csv() {
include 'lib/connection.php';
$csv_output = '';
$values = load_payment_csv();
$fp = fopen("php://output", "w");
$file = 'test_export';
$filename = $file."_".date("Y-m-d_H-i",time());
header("Content-Type: text/csv");
header("Content-Disposition: attachment; filename=".$filename.".csv");
// Disable caching
header("Cache-Control: no-cache, no-store, must-revalidate"); // HTTP 1.1
header("Pragma: no-cache"); // HTTP 1.0
header("Expires: 0"); // Proxies
header("Content-Transfer-Encoding: UTF-8");
if(count($values) > 0) {
foreach($values as $result) {
fputcsv($fp, $result);
}
}
fclose($fp);
}
This generates a CSV, but there is a problem with it.The problem is that when viewing the page it doesn’t download it as a CSV, it just outputs the contents of the CSV in to the page. However, adding this function to the top of the plugin:
add_action('admin_init','download_payment_csv');
This then triggers a download when the menu link is clicked, which is fine. But it triggers it for every menu item in the plugin, which is wrong. It should only trigger when the download link is clicked.
/**
* Query For Top Title Row
*/
/**
* Query For All Required Data
*/
/**
* Prepare Filename And CSV File to export
*/
Putting this all in a function should do the trick
Try this:
You’ll need to change some header information
After that make use of php://output for providing the data directly to the browser, this will prevent the blank page.
for example:
php://output is a read only stream that allows you provide data directly to the requester.
EDIT: Also you should make use of $wpdb
Here is the solution I found, after having tons of trouble with the code outputting HTML code, when used on the front side of the site. The key is to exchange exit for ‘die’. This is the code I am using on my own sites.