netsurf/riscos/download.c
James Bursa 318869e90e [project @ 2004-02-26 00:44:42 by bursa]
Implement drag saving of drawfiles and clean up drag/drop/save code.

svn path=/import/netsurf/; revision=568
2004-02-26 00:44:42 +00:00

303 lines
7.4 KiB
C

/*
* This file is part of NetSurf, http://netsurf.sourceforge.net/
* Licensed under the GNU General Public License,
* http://www.opensource.org/licenses/gpl-license
* Copyright 2004 James Bursa <bursa@users.sourceforge.net>
* Copyright 2003 Rob Jackson <jacko@xms.ms>
*/
#include <assert.h>
#include <string.h>
#include "oslib/mimemap.h"
#include "oslib/osfile.h"
#include "oslib/wimp.h"
#include "oslib/wimpspriteop.h"
#include "netsurf/desktop/gui.h"
#include "netsurf/riscos/gui.h"
#include "netsurf/utils/log.h"
#include "netsurf/utils/messages.h"
#include "netsurf/utils/utils.h"
static wimp_window *download_template;
static void ro_gui_download_leaf(const char *url, char *leaf);
/**
* Load the download window template.
*/
void ro_gui_download_init(void)
{
char name[] = "download";
int context, window_size, data_size;
char *data;
/* find required buffer sizes */
context = wimp_load_template(wimp_GET_SIZE, 0, 0, wimp_NO_FONTS,
name, 0, &window_size, &data_size);
assert(context != 0);
download_template = xcalloc((unsigned int) window_size, 1);
data = xcalloc((unsigned int) data_size, 1);
/* load */
wimp_load_template(download_template, data, data + data_size,
wimp_NO_FONTS, name, 0, 0, 0);
}
/**
* Create and open a download progress window.
*/
gui_window *gui_create_download_window(struct content *content)
{
gui_window *g = xcalloc(1, sizeof(gui_window));
os_error *e;
assert(content->type == CONTENT_OTHER);
g->type = GUI_DOWNLOAD_WINDOW;
g->data.download.content = content;
/* convert MIME type to RISC OS file type */
e = xmimemaptranslate_mime_type_to_filetype(content->mime_type,
&(g->data.download.file_type));
if (e)
g->data.download.file_type = 0xffd;
/* fill in download window icons */
download_template->icons[ICON_DOWNLOAD_URL].data.indirected_text.text =
content->url;
download_template->icons[ICON_DOWNLOAD_URL].data.indirected_text.size =
strlen(content->url) + 1;
strncpy(g->status, content->status_message, 256);
download_template->icons[ICON_DOWNLOAD_STATUS].data.indirected_text.text =
g->status;
download_template->icons[ICON_DOWNLOAD_STATUS].data.indirected_text.size =
256;
sprintf(g->data.download.sprite_name, "file_%x",
g->data.download.file_type);
e = xwimpspriteop_select_sprite(g->data.download.sprite_name, 0);
if (e)
strcpy(g->data.download.sprite_name, "file_xxx");
download_template->icons[ICON_DOWNLOAD_ICON].data.indirected_sprite.id =
(osspriteop_id) g->data.download.sprite_name;
ro_gui_download_leaf(content->url, g->data.download.path);
download_template->icons[ICON_DOWNLOAD_PATH].data.indirected_text.text =
g->data.download.path;
download_template->icons[ICON_DOWNLOAD_PATH].data.indirected_text.size =
256;
/* create and open the download window */
g->window = wimp_create_window(download_template);
ro_gui_dialog_open(g->window);
g->data.download.download_status = download_INCOMPLETE;
g->next = window_list;
window_list = g;
return g;
}
/**
* Find a friendly RISC OS leaf name for a URL.
*/
void ro_gui_download_leaf(const char *url, char *leaf)
{
char *slash;
size_t len;
unsigned int i;
/* take url from last / to first non-RISC OS character, eg. '.' */
slash = strrchr(url, '/');
if (!slash) {
strcpy(leaf, "download");
return;
}
/*len = strspn(slash + 1, "0123456789" "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"abcdefghijklmnopqrstuvwxyz");*/ /* over-paranoid */
len = strlen(slash+1);
if (40 < len)
len = 40;
strncpy(leaf, slash + 1, len);
leaf[len] = 0;
/* swap "." for "/" */
for (i=0; i!=len; i++) {
if (leaf[i] == '.') {
leaf[i] = '/';
}
}
}
/**
* Refresh the status icon in the download window.
*/
void gui_download_window_update_status(gui_window *g)
{
strncpy(g->status, g->data.download.content->status_message, 256);
wimp_set_icon_state(g->window,
ICON_DOWNLOAD_STATUS, 0, 0);
}
/**
* Handle failed downloads.
*/
void gui_download_window_error(gui_window *g, const char *error)
{
g->data.download.content = 0;
/* place error message in status icon in red */
strncpy(g->status, error, 256);
wimp_set_icon_state(g->window,
ICON_DOWNLOAD_STATUS,
wimp_COLOUR_RED << wimp_ICON_FG_COLOUR_SHIFT,
wimp_ICON_FG_COLOUR);
/* grey out file and pathname icons */
wimp_set_icon_state(g->window,
ICON_DOWNLOAD_ICON, wimp_ICON_SHADED, 0);
wimp_set_icon_state(g->window,
ICON_DOWNLOAD_PATH, wimp_ICON_SHADED, 0);
g->data.download.download_status = download_ERROR;
}
/**
* Handle completed downloads.
*/
void gui_download_window_done(gui_window *g)
{
snprintf(g->status, 256, messages_get("Downloaded"),
g->data.download.content->data.other.length);
wimp_set_icon_state(g->window,
ICON_DOWNLOAD_STATUS, 0, 0);
// clear shaded path and icon icons
wimp_set_icon_state(g->window,
ICON_DOWNLOAD_ICON, 0, wimp_ICON_SHADED);
wimp_set_icon_state(g->window,
ICON_DOWNLOAD_PATH, 0, wimp_ICON_SHADED);
g->data.download.download_status = download_COMPLETE;
}
/**
* Handle clicks in a download window.
*/
void ro_download_window_click(struct gui_window *g, wimp_pointer *pointer)
{
switch (pointer->i) {
case ICON_DOWNLOAD_ABORT:
if (g->data.download.download_status ==
download_INCOMPLETE)
fetch_abort(g->data.download.content->fetch);
ro_download_window_close(g);
break;
case ICON_DOWNLOAD_ICON:
if (g->data.download.download_status ==
download_COMPLETE) {
gui_current_drag_type = GUI_DRAG_DOWNLOAD_SAVE;
current_gui = g;
ro_gui_drag_icon(pointer);
}
break;
}
}
/**
* Handle User_Drag_Box event for a drag from a download window.
*/
void ro_download_drag_end(wimp_dragged *drag)
{
wimp_pointer pointer;
wimp_message message;
wimp_get_pointer_info(&pointer);
message.your_ref = 0;
message.action = message_DATA_SAVE;
message.data.data_xfer.w = pointer.w;
message.data.data_xfer.i = pointer.i;
message.data.data_xfer.pos.x = pointer.pos.x;
message.data.data_xfer.pos.y = pointer.pos.y;
message.data.data_xfer.est_size = (int)
current_gui->data.download.content->data.other.length;
message.data.data_xfer.file_type = current_gui->data.download.file_type;
strncpy(message.data.data_xfer.file_name,
current_gui->data.download.path, 212);
message.size = 44 + ((strlen(message.data.data_xfer.file_name) + 4) &
(~3u));
wimp_send_message_to_window(wimp_USER_MESSAGE, &message,
pointer.w, pointer.i);
}
/**
* Handle Message_DataSaveAck for a drag from a download window.
*/
void ro_download_datasave_ack(wimp_message *message)
{
char *data;
char *data_end;
os_error *error;
assert(current_gui->data.download.download_status == download_COMPLETE);
data = current_gui->data.download.content->data.other.data;
data_end = data + current_gui->data.download.content->data.other.length;
error = xosfile_save_stamped(message->data.data_xfer.file_name,
current_gui->data.download.file_type,
data, data_end);
if (error) {
LOG(("0x%x: %s\n", error->errnum, error->errmess));
warn_user(error->errmess);
} else {
ro_download_window_close(current_gui);
}
}
struct gui_window * ro_lookup_download_window_from_w(wimp_w window)
{
gui_window* g;
for (g = window_list; g != NULL; g = g->next)
{
if (g->type == GUI_DOWNLOAD_WINDOW)
{
if (g->window == window)
{
return g;
}
}
}
return NULL;
}
void ro_download_window_close(struct gui_window *g)
{
// free contexts etc???
wimp_close_window(g->window);
}