/* * 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 */ /** \file * Save dialog and drag and drop saving (implementation). */ #include #include #include #include #include "oslib/dragasprite.h" #include "oslib/osfile.h" #include "oslib/osspriteop.h" #include "oslib/wimp.h" #include "netsurf/desktop/save_text.h" #include "netsurf/riscos/gui.h" #include "netsurf/riscos/save_complete.h" #include "netsurf/riscos/save_draw.h" #include "netsurf/riscos/thumbnail.h" #include "netsurf/utils/log.h" #include "netsurf/utils/messages.h" #include "netsurf/utils/utils.h" gui_save_type gui_current_save_type; void ro_gui_save_complete(struct content *c, char *path); /** * Handle clicks in the save dialog. */ void ro_gui_save_click(wimp_pointer *pointer) { switch (pointer->i) { case ICON_SAVE_ICON: if (pointer->buttons == wimp_DRAG_SELECT) { gui_current_drag_type = GUI_DRAG_SAVE; ro_gui_drag_icon(pointer); } break; } } /** * Start drag of icon under the pointer. */ void ro_gui_drag_icon(wimp_pointer *pointer) { char *sprite; os_box box = { pointer->pos.x - 34, pointer->pos.y - 34, pointer->pos.x + 34, pointer->pos.y + 34 }; os_error *error; if (pointer->i == -1) return; sprite = ro_gui_get_icon_string(pointer->w, pointer->i); error = xdragasprite_start(dragasprite_HPOS_CENTRE | dragasprite_VPOS_CENTRE | dragasprite_BOUND_POINTER | dragasprite_DROP_SHADOW, (osspriteop_area *) 1, sprite, &box, 0); if (error) { LOG(("xdragasprite_start: 0x%x: %s", error->errnum, error->errmess)); warn_user(error->errmess); } } /** * Handle User_Drag_Box event for a drag from the save dialog. */ void ro_gui_save_drag_end(wimp_dragged *drag) { char *name; char *dot; wimp_pointer pointer; wimp_message message; wimp_get_pointer_info(&pointer); name = ro_gui_get_icon_string(dialog_saveas, ICON_SAVE_PATH); dot = strrchr(name, '.'); if (dot) name = dot + 1; 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 = 1000; message.data.data_xfer.file_type = 0xfaf; if (gui_current_save_type == GUI_SAVE_DRAW) message.data.data_xfer.file_type = 0xaff; if (gui_current_save_type == GUI_SAVE_COMPLETE) { message.data.data_xfer.file_type = 0x2000; if (name[0] != '!') { message.data.data_xfer.file_name[0] = '!'; strncpy(message.data.data_xfer.file_name + 1, name, 211); } else { strncpy(message.data.data_xfer.file_name, name, 212); } } else strncpy(message.data.data_xfer.file_name, name, 212); message.data.data_xfer.file_name[211] = 0; 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 the save dialog. */ void ro_gui_save_datasave_ack(wimp_message *message) { char *path = message->data.data_xfer.file_name; struct content *c = current_gui->data.browser.bw->current_content; os_error *error; ro_gui_set_icon_string(dialog_saveas, ICON_SAVE_PATH, path); switch (gui_current_save_type) { case GUI_SAVE_SOURCE: if (!c) return; error = xosfile_save_stamped(path, ro_content_filetype(c), c->source_data, c->source_data + c->source_size); if (error) { LOG(("xosfile_save_stamped: 0x%x: %s", error->errnum, error->errmess)); warn_user(error->errmess); } break; case GUI_SAVE_COMPLETE: if (!c) return; ro_gui_save_complete(c, path); break; case GUI_SAVE_DRAW: if (!c) return; save_as_draw(c, path); break; case GUI_SAVE_TEXT: if (!c) return; save_as_text(c, path); xosfile_set_type(path, 0xfff); break; } wimp_create_menu(wimp_CLOSE_MENU, 0, 0); } /** * Prepare an application directory and save_complete() to it. */ #define WIDTH 64 #define HEIGHT 64 #define SPRITE_SIZE (16 + 44 + ((WIDTH / 2 + 3) & ~3) * HEIGHT / 2) void ro_gui_save_complete(struct content *c, char *path) { char buf[256], spritename[13]; FILE *fp; os_error *error; osspriteop_area *area; char *appname; /* Create dir */ error = xosfile_create_dir(path, 0); if (error) { LOG(("xosfile_create_dir: 0x%x: %s", error->errnum, error->errmess)); warn_user(error->errmess); return; } /* Save !Run file */ snprintf(buf, sizeof buf, "%s.!Run", path); fp = fopen(buf, "w"); if (!fp) { LOG(("fopen(): errno = %i", errno)); warn_user(strerror(errno)); return; } fprintf(fp, "Filer_Run .index\n"); fclose(fp); error = xosfile_set_type(buf, 0xfeb); if (error) { LOG(("xosfile_set_type: 0x%x: %s", error->errnum, error->errmess)); warn_user(error->errmess); return; } /* Create !Sprites */ snprintf(buf, sizeof buf, "%s.!Sprites", path); appname = strrchr(path, '.'); if (!appname) { LOG(("Couldn't get appname")); warn_user("Failed to acquire dirname"); return; } snprintf(spritename, sizeof spritename, "%s", appname+1); area = malloc(SPRITE_SIZE); if (!area) { LOG(("malloc failed")); warn_user("No memory for sprite"); return; } area->size = SPRITE_SIZE; area->sprite_count = 0; area->first = 16; area->used = 16; error = xosspriteop_create_sprite(osspriteop_NAME, area, spritename, false, WIDTH / 2, HEIGHT / 2, os_MODE8BPP90X90); if (error) { LOG(("Failed to create sprite")); warn_user("Failed to create iconsprite"); free(area); return; } thumbnail_create(c, area, (osspriteop_header *) ((char *) area + 16), WIDTH / 2, HEIGHT / 2); error = xosspriteop_save_sprite_file(osspriteop_NAME, area, buf); if (error) { LOG(("Failed to save iconsprite")); warn_user("Failed to save iconsprite"); free(area); return; } free(area); /* Create !Boot file */ snprintf(buf, sizeof buf, "%s.!Boot", path); fp = fopen(buf, "w"); if (!fp) { LOG(("fopen(): errno = %i", errno)); warn_user(strerror(errno)); return; } fprintf(fp, "IconSprites .!Sprites\n"); fclose(fp); error = xosfile_set_type(buf, 0xfeb); if (error) { LOG(("xosfile_set_type: 0x%x: %s", error->errnum, error->errmess)); warn_user(error->errmess); return; } save_complete(c, path); }