mirror of
https://github.com/netsurf-browser/netsurf
synced 2024-12-18 02:02:36 +03:00
b88a81b9d9
Allow images to be unloaded to disk or compressed in memory. Provide thumbnails in all tree windows (hotlist, history). Optimise the application initialisation times. Part 1 of 2. svn path=/import/netsurf/; revision=1761
532 lines
13 KiB
C
532 lines
13 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 2005 Richard Wilson <info@tinct.net>
|
|
*/
|
|
|
|
/** \file
|
|
* Global history (implementation).
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
#include "oslib/wimp.h"
|
|
#include "oslib/wimpspriteop.h"
|
|
#include "netsurf/content/url_store.h"
|
|
#include "netsurf/desktop/tree.h"
|
|
#include "netsurf/riscos/global_history.h"
|
|
#include "netsurf/riscos/gui.h"
|
|
#include "netsurf/riscos/menus.h"
|
|
#include "netsurf/riscos/theme.h"
|
|
#include "netsurf/riscos/treeview.h"
|
|
#include "netsurf/utils/messages.h"
|
|
#include "netsurf/utils/log.h"
|
|
#include "netsurf/utils/url.h"
|
|
#include "netsurf/utils/utils.h"
|
|
|
|
#define MAXIMUM_URL_LENGTH 1024
|
|
#define MAXIMUM_BASE_NODES 16
|
|
|
|
#define GLOBAL_HISTORY_RECENT_READ "Choices:WWW.NetSurf.Recent"
|
|
#define GLOBAL_HISTORY_RECENT_WRITE "<Choices$Write>.WWW.NetSurf.Recent"
|
|
#define GLOBAL_HISTORY_READ "Choices:WWW.NetSurf.History"
|
|
#define GLOBAL_HISTORY_WRITE "<Choices$Write>.WWW.NetSurf.History"
|
|
|
|
|
|
static struct node *global_history_base_node[MAXIMUM_BASE_NODES];
|
|
static int global_history_base_node_time[MAXIMUM_BASE_NODES];
|
|
static int global_history_base_node_count = 0;
|
|
|
|
static char *global_history_recent_url[GLOBAL_HISTORY_RECENT_URLS];
|
|
static int global_history_recent_count = 0;
|
|
|
|
static bool global_history_init;
|
|
|
|
|
|
static void ro_gui_global_history_initialise_nodes(void);
|
|
static void ro_gui_global_history_initialise_node(const char *title,
|
|
time_t base, int days_back);
|
|
static void ro_gui_global_history_add(char *title, char *url, int visit_date,
|
|
int filetype);
|
|
static void ro_gui_global_history_save_node(struct node *node, FILE *fp);
|
|
|
|
/* A basic window for the history
|
|
*/
|
|
static wimp_window history_window_definition = {
|
|
{0, 0, 600, 800},
|
|
0,
|
|
0,
|
|
wimp_TOP,
|
|
wimp_WINDOW_NEW_FORMAT | wimp_WINDOW_MOVEABLE | wimp_WINDOW_BACK_ICON |
|
|
wimp_WINDOW_CLOSE_ICON | wimp_WINDOW_TITLE_ICON |
|
|
wimp_WINDOW_TOGGLE_ICON | wimp_WINDOW_SIZE_ICON |
|
|
wimp_WINDOW_VSCROLL | wimp_WINDOW_IGNORE_XEXTENT |
|
|
wimp_WINDOW_IGNORE_YEXTENT,
|
|
wimp_COLOUR_BLACK,
|
|
wimp_COLOUR_LIGHT_GREY,
|
|
wimp_COLOUR_LIGHT_GREY,
|
|
wimp_COLOUR_WHITE,
|
|
wimp_COLOUR_DARK_GREY,
|
|
wimp_COLOUR_MID_LIGHT_GREY,
|
|
wimp_COLOUR_CREAM,
|
|
0,
|
|
{0, -16384, 16384, 0},
|
|
wimp_ICON_TEXT | wimp_ICON_INDIRECTED | wimp_ICON_HCENTRED |
|
|
wimp_ICON_VCENTRED,
|
|
wimp_BUTTON_DOUBLE_CLICK_DRAG << wimp_ICON_BUTTON_TYPE_SHIFT,
|
|
wimpspriteop_AREA,
|
|
1,
|
|
1,
|
|
{""},
|
|
0,
|
|
{}
|
|
};
|
|
|
|
|
|
/* The history window, toolbar and plot origins
|
|
*/
|
|
static wimp_w global_history_window;
|
|
struct tree *global_history_tree;
|
|
|
|
void ro_gui_global_history_initialise(void) {
|
|
char s[MAXIMUM_URL_LENGTH];
|
|
FILE *fp;
|
|
const char *title;
|
|
os_error *error;
|
|
char *node_title;
|
|
char *node_url;
|
|
int node_filetype;
|
|
int node_visited;
|
|
|
|
/* Create our window
|
|
*/
|
|
title = messages_get("GlobalHistory");
|
|
history_window_definition.title_data.indirected_text.text =
|
|
strdup(title);
|
|
history_window_definition.title_data.indirected_text.validation =
|
|
(char *) -1;
|
|
history_window_definition.title_data.indirected_text.size =
|
|
strlen(title);
|
|
error = xwimp_create_window(&history_window_definition,
|
|
&global_history_window);
|
|
if (error) {
|
|
LOG(("xwimp_create_window: 0x%x: %s",
|
|
error->errnum, error->errmess));
|
|
die(error->errmess);
|
|
}
|
|
|
|
/* Create an empty tree
|
|
*/
|
|
global_history_tree = calloc(sizeof(struct tree), 1);
|
|
if (!global_history_tree) {
|
|
warn_user("NoMemory", 0);
|
|
return;
|
|
}
|
|
global_history_tree->root = tree_create_folder_node(NULL, "Root");
|
|
if (!global_history_tree->root) {
|
|
warn_user("NoMemory", 0);
|
|
free(global_history_tree);
|
|
global_history_tree = NULL;
|
|
}
|
|
global_history_tree->root->expanded = true;
|
|
ro_gui_global_history_initialise_nodes();
|
|
tree_initialise(global_history_tree);
|
|
global_history_tree->handle = (int)global_history_window;
|
|
global_history_tree->movable = false;
|
|
|
|
/* Create our toolbar
|
|
*/
|
|
global_history_tree->toolbar = ro_gui_theme_create_toolbar(NULL,
|
|
THEME_HISTORY_TOOLBAR);
|
|
if (global_history_tree->toolbar)
|
|
ro_gui_theme_attach_toolbar(global_history_tree->toolbar,
|
|
global_history_window);
|
|
|
|
/* load recent URLs */
|
|
fp = fopen(GLOBAL_HISTORY_RECENT_READ, "r");
|
|
if (!fp)
|
|
LOG(("Failed to open file '%s' for reading",
|
|
GLOBAL_HISTORY_RECENT_READ));
|
|
else {
|
|
while (fgets(s, MAXIMUM_URL_LENGTH, fp)) {
|
|
if (s[strlen(s) - 1] == '\n')
|
|
s[strlen(s) - 1] = '\0';
|
|
global_history_add_recent(s);
|
|
}
|
|
fclose(fp);
|
|
}
|
|
|
|
/* load global history */
|
|
fp = fopen(GLOBAL_HISTORY_READ, "r");
|
|
if (!fp)
|
|
LOG(("Failed to open file '%s' for reading",
|
|
GLOBAL_HISTORY_READ));
|
|
else {
|
|
global_history_init = true;
|
|
while (fgets(s, MAXIMUM_URL_LENGTH, fp)) {
|
|
if (s[strlen(s) - 1] == '\n')
|
|
s[strlen(s) - 1] = '\0';
|
|
node_title = strdup(s);
|
|
if (!fgets(s, MAXIMUM_URL_LENGTH, fp)) {
|
|
LOG(("Error reading global history"));
|
|
warn_user("HistoryCorrupt", 0);
|
|
fclose(fp);
|
|
global_history_init = false;
|
|
return;
|
|
}
|
|
if (s[strlen(s) - 1] == '\n')
|
|
s[strlen(s) - 1] = '\0';
|
|
node_url = strdup(s);
|
|
if ((!node_title) || (!node_url)) {
|
|
LOG(("No memory to read global history node"));
|
|
warn_user("NoMemory", 0);
|
|
fclose(fp);
|
|
global_history_init = false;
|
|
return;
|
|
}
|
|
if (!fgets(s, MAXIMUM_URL_LENGTH, fp)) {
|
|
LOG(("Error reading global history"));
|
|
warn_user("HistoryCorrupt", 0);
|
|
fclose(fp);
|
|
global_history_init = false;
|
|
return;
|
|
}
|
|
node_filetype = atoi(s);
|
|
if (!fgets(s, MAXIMUM_URL_LENGTH, fp)) {
|
|
LOG(("Error reading global history"));
|
|
warn_user("HistoryCorrupt", 0);
|
|
fclose(fp);
|
|
global_history_init = false;
|
|
return;
|
|
}
|
|
node_visited = atoi(s);
|
|
ro_gui_global_history_add(node_title, node_url,
|
|
node_visited, node_filetype);
|
|
free(node_title);
|
|
node_title = NULL;
|
|
free(node_url);
|
|
node_url = NULL;
|
|
}
|
|
global_history_init = false;
|
|
fclose(fp);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Initialises the base nodes
|
|
*/
|
|
static void ro_gui_global_history_initialise_nodes(void) {
|
|
struct tm *full_time;
|
|
time_t t;
|
|
int weekday;
|
|
int i;
|
|
|
|
/* get the current time */
|
|
t = time(NULL);
|
|
if (t == -1)
|
|
return;
|
|
|
|
/* get the time at the start of today */
|
|
full_time = localtime(&t);
|
|
weekday = full_time->tm_wday;
|
|
full_time->tm_sec = 0;
|
|
full_time->tm_min = 0;
|
|
full_time->tm_hour = 0;
|
|
t = mktime(full_time);
|
|
if (t == -1)
|
|
return;
|
|
|
|
ro_gui_global_history_initialise_node(messages_get("DateToday"), t, 0);
|
|
if (weekday > 0)
|
|
ro_gui_global_history_initialise_node(
|
|
messages_get("DateYesterday"), t, -1);
|
|
for (i = 2; i <= weekday; i++)
|
|
ro_gui_global_history_initialise_node(NULL, t, -i);
|
|
ro_gui_global_history_initialise_node(messages_get("Date1Week"),
|
|
t, -weekday - 7);
|
|
ro_gui_global_history_initialise_node(messages_get("Date2Week"),
|
|
t, -weekday - 14);
|
|
ro_gui_global_history_initialise_node(messages_get("Date3Week"),
|
|
t, -weekday - 21);
|
|
}
|
|
|
|
static void ro_gui_global_history_initialise_node(const char *title,
|
|
time_t base, int days_back) {
|
|
struct tm *full_time;
|
|
char buffer[64];
|
|
struct node *node;
|
|
|
|
base += days_back * 60 * 60 * 24;
|
|
if (!title) {
|
|
full_time = localtime(&base);
|
|
strftime((char *)&buffer, (size_t)64, "%A", full_time);
|
|
node = tree_create_folder_node(NULL, buffer);
|
|
} else
|
|
node = tree_create_folder_node(NULL, title);
|
|
|
|
if (!node)
|
|
return;
|
|
|
|
node->retain_in_memory = true;
|
|
node->deleted = true;
|
|
node->editable = false;
|
|
global_history_base_node[global_history_base_node_count] = node;
|
|
global_history_base_node_time[global_history_base_node_count] = base;
|
|
global_history_base_node_count++;
|
|
}
|
|
|
|
|
|
/**
|
|
* Saves the global history and recent URL data.
|
|
*/
|
|
void ro_gui_global_history_save(void) {
|
|
FILE *fp;
|
|
int i;
|
|
|
|
/* save recent URLs */
|
|
fp = fopen(GLOBAL_HISTORY_RECENT_WRITE, "w");
|
|
if (!fp)
|
|
LOG(("Failed to open file '%s' for writing",
|
|
GLOBAL_HISTORY_RECENT_WRITE));
|
|
else {
|
|
for (i = global_history_recent_count - 1; i >= 0; i--)
|
|
if (strlen(global_history_recent_url[i]) <
|
|
MAXIMUM_URL_LENGTH)
|
|
fprintf(fp, "%s\n",
|
|
global_history_recent_url[i]);
|
|
fclose(fp);
|
|
}
|
|
|
|
/* save global history tree */
|
|
fp = fopen(GLOBAL_HISTORY_WRITE, "w");
|
|
if (!fp)
|
|
LOG(("Failed to open file '%s' for writing",
|
|
GLOBAL_HISTORY_WRITE));
|
|
else {
|
|
ro_gui_global_history_save_node(global_history_tree->root, fp);
|
|
fclose(fp);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Saves the global history for a tree node.
|
|
*/
|
|
void ro_gui_global_history_save_node(struct node *node, FILE *fp) {
|
|
struct node_element *element;
|
|
|
|
for (; node; node = node->next) {
|
|
if (node->folder) {
|
|
ro_gui_global_history_save_node(node->child, fp);
|
|
} else {
|
|
element = tree_find_element(node, TREE_ELEMENT_URL);
|
|
if (element) {
|
|
fprintf(fp, "%s\n%s\n%i\n", node->data.text,
|
|
element->text,
|
|
element->user_data);
|
|
|
|
element = tree_find_element(node,
|
|
TREE_ELEMENT_VISITED);
|
|
if (element)
|
|
fprintf(fp, "%i\n", element->user_data);
|
|
else
|
|
fprintf(fp, "-1\n");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Respond to a mouse click
|
|
*
|
|
* \param pointer the pointer state
|
|
*/
|
|
void ro_gui_global_history_click(wimp_pointer *pointer) {
|
|
ro_gui_tree_click(pointer, global_history_tree);
|
|
if (pointer->buttons == wimp_CLICK_MENU)
|
|
ro_gui_menu_create(global_history_menu, pointer->pos.x,
|
|
pointer->pos.y, pointer->w);
|
|
else
|
|
ro_gui_menu_prepare_action(pointer->w, TREE_SELECTION, false);
|
|
}
|
|
|
|
|
|
/**
|
|
* Attempts to process an interactive help message request
|
|
*
|
|
* \param x the x co-ordinate to give help for
|
|
* \param y the x co-ordinate to give help for
|
|
* \return the message code index
|
|
*/
|
|
int ro_gui_global_history_help(int x, int y) {
|
|
return -1;
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds to the global history
|
|
*
|
|
* \param g the gui_window to add to the global history
|
|
*/
|
|
void global_history_add(struct gui_window *g) {
|
|
url_func_result res;
|
|
char *norm_url = NULL;
|
|
|
|
assert(g);
|
|
|
|
if ((!g->bw->current_content) || (!global_history_tree))
|
|
return;
|
|
|
|
res = url_normalize(g->bw->current_content->url, &norm_url);
|
|
if (res != URL_FUNC_OK) {
|
|
warn_user("NoMemory", 0);
|
|
return;
|
|
}
|
|
|
|
ro_gui_global_history_add(g->bw->current_content->title,
|
|
norm_url, time(NULL),
|
|
ro_content_filetype(g->bw->current_content));
|
|
free(norm_url);
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds to the global history
|
|
*
|
|
* \param title the page title
|
|
* \param url the page URL
|
|
* \param visit_date the visit date
|
|
*/
|
|
void ro_gui_global_history_add(char *title, char *url, int visit_date,
|
|
int filetype) {
|
|
int i, j;
|
|
struct node *parent = NULL;
|
|
struct node *link;
|
|
struct node *node;
|
|
struct node_element *element;
|
|
bool before = false;
|
|
|
|
/* Find/create the node to link into
|
|
*/
|
|
for (i = 0; i < global_history_base_node_count; i++) {
|
|
if (global_history_base_node_time[i] <= visit_date) {
|
|
parent = global_history_base_node[i];
|
|
if (!parent->deleted)
|
|
break;
|
|
link = global_history_tree->root;
|
|
for (j = 0; j < i; j++) {
|
|
if (!global_history_base_node[j]->deleted) {
|
|
link = global_history_base_node[j];
|
|
before = true;
|
|
break;
|
|
}
|
|
}
|
|
tree_link_node(link, parent, before);
|
|
tree_recalculate_node_positions(
|
|
global_history_tree->root);
|
|
if (!global_history_init)
|
|
tree_redraw_area(global_history_tree,
|
|
0, 0, 16384, 16384);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (parent) {
|
|
/* Add the node at the bottom
|
|
*/
|
|
node = tree_create_URL_node_brief(parent, title, url, filetype,
|
|
visit_date);
|
|
if (node && !global_history_init)
|
|
tree_redraw_area(global_history_tree,
|
|
node->box.x - NODE_INSTEP,
|
|
0, NODE_INSTEP, 16384);
|
|
if (!global_history_init)
|
|
tree_handle_node_changed(global_history_tree, node,
|
|
true, false);
|
|
|
|
/* Remove any duplicate URL from within the parent node.
|
|
* This must be done after the duplicate has been created as
|
|
* deleting all children of a child automatically collapses
|
|
* the display.
|
|
*/
|
|
for (node = parent->child; node && (node->next);
|
|
node = node->next) {
|
|
element = tree_find_element(node, TREE_ELEMENT_URL);
|
|
if ((element) && (!strcmp(url, element->text))) {
|
|
tree_delete_node(global_history_tree, node,
|
|
false);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a URL to the recently used list
|
|
*
|
|
* \param url the URL to add
|
|
*/
|
|
void global_history_add_recent(const char *url) {
|
|
struct url_content *data;
|
|
int i;
|
|
int j = -1;
|
|
char *current;
|
|
|
|
/* by using the url_store, we get a central char* of the string that
|
|
* isn't going anywhere unless we tell it to */
|
|
data = url_store_find(url);
|
|
if (!data)
|
|
return;
|
|
|
|
/* try to find a string already there */
|
|
for (i = 0; i < global_history_recent_count; i++)
|
|
if (global_history_recent_url[i] == data->url)
|
|
j = i;
|
|
|
|
/* already at head of list */
|
|
if (j == 0)
|
|
return;
|
|
|
|
/* add to head of list */
|
|
if (j < 0) {
|
|
memmove(&global_history_recent_url[1],
|
|
&global_history_recent_url[0],
|
|
(GLOBAL_HISTORY_RECENT_URLS - 1) *
|
|
sizeof(char *));
|
|
global_history_recent_url[0] = data->url;
|
|
global_history_recent_count++;
|
|
if (global_history_recent_count > GLOBAL_HISTORY_RECENT_URLS)
|
|
global_history_recent_count =
|
|
GLOBAL_HISTORY_RECENT_URLS;
|
|
if (global_history_recent_count == 1)
|
|
ro_gui_window_prepare_navigate_all();
|
|
} else {
|
|
/* move to head of list */
|
|
current = global_history_recent_url[j];
|
|
for (i = j; i > 0; i--)
|
|
global_history_recent_url[i] =
|
|
global_history_recent_url[i - 1];
|
|
global_history_recent_url[0] = current;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Gets details of the currently used URL list.
|
|
*
|
|
* \param count set to the current number of entries in the URL array on exit
|
|
* \return the current URL array
|
|
*/
|
|
char **global_history_get_recent(int *count) {
|
|
*count = global_history_recent_count;
|
|
return global_history_recent_url;
|
|
}
|