mirror of
https://github.com/netsurf-browser/netsurf
synced 2025-01-11 13:29:21 +03:00
c105738fa3
This changes the LOG macro to be varadic removing the need for all callsites to have double bracketing and allows for future improvement on how we use the logging macros. The callsites were changed with coccinelle and the changes checked by hand. Compile tested for several frontends but not all. A formatting annotation has also been added which allows the compiler to check the parameters and types passed to the logging.
454 lines
12 KiB
C
454 lines
12 KiB
C
/*
|
|
* Copyright 2010 Ole Loots <ole@monochrom.net>
|
|
*
|
|
* This file is part of NetSurf, http://www.netsurf-browser.org/
|
|
*
|
|
* NetSurf is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; version 2 of the License.
|
|
*
|
|
* NetSurf is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <stdbool.h>
|
|
|
|
#include "utils/log.h"
|
|
#include "utils/messages.h"
|
|
#include "utils/utils.h"
|
|
#include "utils/nsoption.h"
|
|
#include "content/urldb.h"
|
|
#include "content/fetch.h"
|
|
#include "desktop/save_complete.h"
|
|
#include "desktop/textinput.h"
|
|
#include "desktop/download.h"
|
|
#include "desktop/browser.h"
|
|
#include "desktop/gui_download.h"
|
|
|
|
#include "atari/gui.h"
|
|
#include "atari/misc.h"
|
|
#include "atari/res/netsurf.rsh"
|
|
#include "atari/download.h"
|
|
#include "atari/osspec.h"
|
|
|
|
extern struct gui_window * input_window;
|
|
extern GRECT desk_area;
|
|
|
|
static void gui_download_window_destroy( struct gui_download_window * gdw );
|
|
static void on_abort_click(struct gui_download_window *dw);
|
|
static void on_cbrdy_click(struct gui_download_window *dw);
|
|
static void on_close(struct gui_download_window * dw);
|
|
static void on_redraw(struct gui_download_window *dw, GRECT *clip);
|
|
|
|
static void toolbar_redraw_cb(GUIWIN *win, uint16_t msg, GRECT *clip)
|
|
{
|
|
struct gui_download_window *data;
|
|
|
|
if (msg != WM_REDRAW) {
|
|
data = gemtk_wm_get_user_data(win);
|
|
|
|
assert(data);
|
|
|
|
on_redraw(data, clip);
|
|
}
|
|
}
|
|
|
|
static short on_aes_event(GUIWIN *win, EVMULT_OUT *ev_out, short msg[8])
|
|
{
|
|
short retval = 0;
|
|
struct gui_download_window *data;
|
|
|
|
GRECT clip;
|
|
|
|
data = gemtk_wm_get_user_data(win);
|
|
|
|
if ((ev_out->emo_events & MU_MESAG) != 0) {
|
|
// handle message
|
|
//printf("download win msg: %d\n", msg[0]);
|
|
switch (msg[0]) {
|
|
|
|
case WM_REDRAW:
|
|
clip.g_x = msg[4];
|
|
clip.g_y = msg[5];
|
|
clip.g_w = msg[6];
|
|
clip.g_h = msg[7];
|
|
on_redraw(data, &clip);
|
|
break;
|
|
|
|
case WM_CLOSED:
|
|
// TODO: this needs to iterate through all gui windows and
|
|
// check if the rootwin is this window...
|
|
on_close(data);
|
|
break;
|
|
|
|
case WM_TOOLBAR:
|
|
switch(msg[4]){
|
|
|
|
case DOWNLOAD_BT_ABORT:
|
|
on_abort_click(data);
|
|
break;
|
|
|
|
case DOWNLOAD_CB_CLOSE_RDY:
|
|
on_cbrdy_click(data);
|
|
break;
|
|
|
|
default: break;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if ((ev_out->emo_events & MU_KEYBD) != 0) {
|
|
|
|
|
|
}
|
|
if ((ev_out->emo_events & MU_BUTTON) != 0) {
|
|
|
|
}
|
|
|
|
return(retval);
|
|
}
|
|
|
|
static void on_redraw(struct gui_download_window *dw, GRECT *clip)
|
|
{
|
|
OBJECT *tree = dw->tree;
|
|
GRECT work, visible;
|
|
uint32_t p = 0;
|
|
|
|
gemtk_wm_get_grect(dw->guiwin, GEMTK_WM_AREA_TOOLBAR, &work);
|
|
tree->ob_x = work.g_x;
|
|
tree->ob_y = work.g_y;
|
|
|
|
if(!rc_intersect(clip, &work)){
|
|
return;
|
|
}
|
|
|
|
/*
|
|
Update the AES Object to reflect current state of download:
|
|
*/
|
|
((TEDINFO *)get_obspec(tree, DOWNLOAD_FILENAME))->te_ptext = dw->lbl_file;
|
|
((TEDINFO *)get_obspec(tree, DOWNLOAD_LBL_BYTES))->te_ptext = dw->lbl_done;
|
|
((TEDINFO *)get_obspec(tree, DOWNLOAD_LBL_PERCENT))->te_ptext = dw->lbl_percent;
|
|
((TEDINFO *)get_obspec(tree, DOWNLOAD_LBL_SPEED))->te_ptext = dw->lbl_speed;
|
|
|
|
if (dw->size_total > 0 ) {
|
|
p = ((double)dw->size_downloaded / (double)dw->size_total * 100);
|
|
}
|
|
tree[DOWNLOAD_PROGRESS_DONE].ob_width = MAX( MIN( p*(DOWNLOAD_BAR_MAX/100),
|
|
DOWNLOAD_BAR_MAX ), 1);
|
|
if (dw->close_on_finish) {
|
|
tree[DOWNLOAD_CB_CLOSE_RDY].ob_state |= (OS_SELECTED | OS_CROSSED);
|
|
} else {
|
|
tree[DOWNLOAD_CB_CLOSE_RDY].ob_state &= ~(OS_SELECTED | OS_CROSSED);
|
|
}
|
|
tree[DOWNLOAD_BT_ABORT].ob_state &= ~OS_SELECTED;
|
|
|
|
/*Walk the AES rectangle list and redraw the visible areas of the window: */
|
|
wind_get_grect(dw->aes_handle, WF_FIRSTXYWH, &visible);
|
|
while (visible.g_x && visible.g_y) {
|
|
if (rc_intersect(&work, &visible)) {
|
|
objc_draw_grect(tree, 0, 8, &visible);
|
|
}
|
|
wind_get_grect(dw->aes_handle, WF_NEXTXYWH, &visible);
|
|
}
|
|
}
|
|
|
|
static void on_abort_click(struct gui_download_window *dw)
|
|
{
|
|
if( dw->status == NSATARI_DOWNLOAD_COMPLETE
|
|
|| dw->status == NSATARI_DOWNLOAD_ERROR ) {
|
|
gemtk_wm_send_msg(dw->guiwin, WM_CLOSED, 0,0,0,0);
|
|
}
|
|
else if( dw->status != NSATARI_DOWNLOAD_CANCELED ){
|
|
dw->abort = true;
|
|
}
|
|
}
|
|
|
|
static void on_cbrdy_click(struct gui_download_window *dw)
|
|
{
|
|
dw->close_on_finish = !dw->close_on_finish;
|
|
if (dw->close_on_finish && dw->status == NSATARI_DOWNLOAD_COMPLETE) {
|
|
gemtk_wm_send_msg(dw->guiwin, WM_CLOSED, 0,0,0,0);
|
|
}
|
|
gemtk_wm_exec_redraw(dw->guiwin, NULL);
|
|
evnt_timer(250);
|
|
}
|
|
|
|
static void on_close(struct gui_download_window * dw)
|
|
{
|
|
gui_download_window_destroy(dw);
|
|
}
|
|
|
|
static void gui_download_window_destroy( struct gui_download_window * gdw)
|
|
{
|
|
|
|
|
|
LOG("");
|
|
if (gdw->status == NSATARI_DOWNLOAD_WORKING) {
|
|
download_context_abort(gdw->ctx);
|
|
}
|
|
|
|
download_context_destroy(gdw->ctx);
|
|
|
|
if (gdw->destination) {
|
|
free( gdw->destination );
|
|
}
|
|
if (gdw->fd != NULL) {
|
|
fclose(gdw->fd);
|
|
gdw->fd = NULL;
|
|
}
|
|
if (gdw->fbuf != NULL) {
|
|
free( gdw->fbuf );
|
|
}
|
|
gemtk_wm_remove(gdw->guiwin);
|
|
wind_close(gdw->aes_handle);
|
|
wind_delete(gdw->aes_handle);
|
|
free(gdw);
|
|
}
|
|
|
|
static char * select_filepath( const char * path, const char * filename )
|
|
{
|
|
char tmp[PATH_MAX];
|
|
char res_path[PATH_MAX];
|
|
char res_file[PATH_MAX];
|
|
char * ret = NULL;
|
|
|
|
strncpy(res_path, path, PATH_MAX);
|
|
strncpy(res_file, filename, PATH_MAX);
|
|
res_file[PATH_MAX-1] = 0;
|
|
res_path[PATH_MAX-1] = 0;
|
|
|
|
if(select_file(res_path, res_file, (char*)"*",
|
|
(char*)messages_get("SaveAsNS"), NULL)) {
|
|
snprintf(tmp, PATH_MAX, "%s%s", res_path, res_file);
|
|
ret = malloc(strlen(tmp)+1);
|
|
strcpy(ret, tmp);
|
|
}
|
|
|
|
printf("download file: %s\n", ret);
|
|
return(ret);
|
|
}
|
|
|
|
static struct gui_download_window *
|
|
gui_download_window_create(download_context *ctx, struct gui_window *parent)
|
|
{
|
|
const char *filename;
|
|
char *destination;
|
|
char gdos_path[PATH_MAX];
|
|
struct gui_download_window * gdw;
|
|
int dlgres = 0;
|
|
OBJECT * tree = gemtk_obj_get_tree(DOWNLOAD);
|
|
char alert[200];
|
|
|
|
|
|
LOG("Creating download window for gui window: %p", parent);
|
|
|
|
/* TODO: Implement real form and use messages file strings! */
|
|
|
|
if (tree == NULL){
|
|
die("Couldn't find AES Object tree for download window!");
|
|
return(NULL);
|
|
}
|
|
|
|
filename = download_context_get_filename((const download_context*)ctx);
|
|
snprintf(alert, 200, "[2][Accept download?|%.*s][Yes|Save as...|No]",
|
|
40,filename);
|
|
dlgres = form_alert(2, alert);
|
|
if( dlgres == 3){
|
|
return( NULL );
|
|
}
|
|
else if( dlgres == 2 ){
|
|
gemdos_realpath(nsoption_charp(downloads_path), gdos_path);
|
|
char * tmp = select_filepath( gdos_path, filename );
|
|
if( tmp == NULL )
|
|
return( NULL );
|
|
destination = tmp;
|
|
} else {
|
|
int dstsize=0;
|
|
gemdos_realpath(nsoption_charp(downloads_path), gdos_path);
|
|
dstsize = strlen(gdos_path) + strlen(filename) + 2;
|
|
destination = malloc( dstsize );
|
|
snprintf(destination, dstsize, "%s/%s", gdos_path, filename);
|
|
}
|
|
|
|
gdw = calloc(1, sizeof(struct gui_download_window));
|
|
if( gdw == NULL ){
|
|
warn_user(NULL, "Out of memory!");
|
|
free( destination );
|
|
return( NULL );
|
|
}
|
|
|
|
gdw->ctx = ctx;
|
|
gdw->abort = false;
|
|
gdw->start = clock() / CLOCKS_PER_SEC;
|
|
gdw->lastrdw = 0;
|
|
gdw->status = NSATARI_DOWNLOAD_WORKING;
|
|
gdw->parent = parent;
|
|
gdw->fbufsize = MAX(BUFSIZ, 48000);
|
|
gdw->size_downloaded = 0;
|
|
gdw->size_total = download_context_get_total_length(ctx);
|
|
gdw->destination = destination;
|
|
gdw->tree = tree;
|
|
|
|
gdw->fd = fopen(gdw->destination, "wb");
|
|
if( gdw->fd == NULL ){
|
|
char spare[200];
|
|
snprintf(spare, 200, "Couldn't open %s for writing!", gdw->destination);
|
|
gemtk_msg_box_show(GEMTK_MSG_BOX_ALERT, spare);
|
|
gui_download_window_destroy(gdw);
|
|
return( NULL );
|
|
}
|
|
|
|
gdw->fbuf = malloc( gdw->fbufsize+1 );
|
|
if( gdw->fbuf != NULL ){
|
|
setvbuf( gdw->fd, gdw->fbuf, _IOFBF, gdw->fbufsize );
|
|
}
|
|
|
|
gdw->aes_handle = wind_create_grect(CLOSER | NAME | MOVER, &desk_area);
|
|
wind_set_str(gdw->aes_handle, WF_NAME, "Download");
|
|
unsigned long gwflags = GEMTK_WM_FLAG_DEFAULTS;
|
|
gdw->guiwin = gemtk_wm_add(gdw->aes_handle, gwflags, on_aes_event);
|
|
if( gdw->guiwin == NULL || gdw->fd == NULL ){
|
|
die("could not create guiwin");
|
|
gui_download_window_destroy(gdw);
|
|
return( NULL );
|
|
}
|
|
gemtk_wm_set_user_data(gdw->guiwin, gdw);
|
|
gemtk_wm_set_toolbar(gdw->guiwin, tree, 0, 0);
|
|
gemtk_wm_set_toolbar_redraw_func(gdw->guiwin, toolbar_redraw_cb);
|
|
|
|
strncpy((char*)&gdw->lbl_file, filename, MAX_SLEN_LBL_FILE-1);
|
|
LOG("created download: %s (total size: %d)", gdw->destination, gdw->size_total);
|
|
|
|
GRECT work, curr;
|
|
work.g_x = 0;
|
|
work.g_y = 0;
|
|
work.g_w = tree->ob_width;
|
|
work.g_h = tree->ob_height;
|
|
|
|
wind_calc_grect(WC_BORDER, CLOSER | MOVER | NAME, &work, &curr);
|
|
|
|
curr.g_x = (desk_area.g_w / 2) - (curr.g_w / 2);
|
|
curr.g_y = (desk_area.g_h / 2) - (curr.g_h / 2);
|
|
|
|
wind_open_grect(gdw->aes_handle, &curr);
|
|
gdw->lastrdw = clock() / (CLOCKS_PER_SEC >> 3);
|
|
|
|
return(gdw);
|
|
}
|
|
|
|
|
|
static nserror gui_download_window_data(struct gui_download_window *dw,
|
|
const char *data, unsigned int size)
|
|
{
|
|
|
|
uint32_t clck = clock();
|
|
uint32_t tnow = clck / (CLOCKS_PER_SEC>>3);
|
|
uint32_t sdiff = (clck / (CLOCKS_PER_SEC)) - dw->start;
|
|
|
|
LOG("");
|
|
|
|
if(dw->abort == true){
|
|
dw->status = NSATARI_DOWNLOAD_CANCELED;
|
|
dw->abort = false;
|
|
download_context_abort(dw->ctx);
|
|
gemtk_wm_exec_redraw(dw->guiwin, NULL);
|
|
return(NSERROR_OK);
|
|
}
|
|
|
|
/* save data */
|
|
fwrite( data , size, sizeof(unsigned char),dw->fd );
|
|
dw->size_downloaded += size;
|
|
|
|
/* Update GUI */
|
|
if ((tnow - dw->lastrdw) > 1) {
|
|
float speed;
|
|
|
|
dw->lastrdw = tnow;
|
|
speed = dw->size_downloaded / sdiff;
|
|
|
|
if( dw->size_total > 0 ){
|
|
uint32_t p = 0;
|
|
p = ((double)dw->size_downloaded / (double)dw->size_total * 100);
|
|
snprintf( (char*)&dw->lbl_percent, MAX_SLEN_LBL_PERCENT,
|
|
"%"PRIu32"%s", p, "%"
|
|
);
|
|
} else {
|
|
snprintf( (char*)&dw->lbl_percent, MAX_SLEN_LBL_PERCENT,
|
|
"%s", "?%"
|
|
);
|
|
}
|
|
snprintf( (char*)&dw->lbl_speed, MAX_SLEN_LBL_SPEED, "%s/s",
|
|
human_friendly_bytesize(speed)
|
|
);
|
|
snprintf( (char*)&dw->lbl_done, MAX_SLEN_LBL_DONE, "%s / %s",
|
|
human_friendly_bytesize(dw->size_downloaded),
|
|
(dw->size_total>0) ? human_friendly_bytesize(dw->size_total) : "?"
|
|
);
|
|
|
|
gemtk_wm_exec_redraw(dw->guiwin, NULL);
|
|
}
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
static void gui_download_window_error(struct gui_download_window *dw,
|
|
const char *error_msg)
|
|
{
|
|
LOG("%s", error_msg);
|
|
strncpy((char*)&dw->lbl_file, error_msg, MAX_SLEN_LBL_FILE-1);
|
|
dw->status = NSATARI_DOWNLOAD_ERROR;
|
|
gemtk_wm_exec_redraw(dw->guiwin, NULL);
|
|
atari_window_set_status(input_window, messages_get("Done") );
|
|
// TODO: change abort to close
|
|
}
|
|
|
|
static void gui_download_window_done(struct gui_download_window *dw)
|
|
{
|
|
LOG("");
|
|
|
|
// TODO: change abort to close
|
|
dw->status = NSATARI_DOWNLOAD_COMPLETE;
|
|
|
|
if( dw->fd != NULL ) {
|
|
fclose( dw->fd );
|
|
dw->fd = NULL;
|
|
}
|
|
|
|
if (dw->close_on_finish) {
|
|
gemtk_wm_send_msg(dw->guiwin, WM_CLOSED, 0, 0, 0, 0);
|
|
} else {
|
|
snprintf( (char*)&dw->lbl_percent, MAX_SLEN_LBL_PERCENT,
|
|
"%u%s", 100, "%"
|
|
);
|
|
snprintf( (char*)&dw->lbl_done, MAX_SLEN_LBL_DONE, "%s / %s",
|
|
human_friendly_bytesize(dw->size_downloaded),
|
|
(dw->size_total>0) ? human_friendly_bytesize(dw->size_total) : human_friendly_bytesize(dw->size_downloaded)
|
|
);
|
|
gemtk_wm_exec_redraw(dw->guiwin, NULL);
|
|
}
|
|
atari_window_set_status(input_window, messages_get("Done") );
|
|
}
|
|
|
|
static struct gui_download_table download_table = {
|
|
.create = gui_download_window_create,
|
|
.data = gui_download_window_data,
|
|
.error = gui_download_window_error,
|
|
.done = gui_download_window_done,
|
|
};
|
|
|
|
struct gui_download_table *atari_download_table = &download_table;
|