mirror of
https://github.com/netsurf-browser/netsurf
synced 2025-01-26 04:12:03 +03:00
9aecf47408
Previously the expected behaviour for front ends using the correct API for hotlist, global history, cookie manager, and ssl cert viewer was that the front end would initialise the treeview module on startup and finalise it on application exit. However, this meant that the front ends had to include the core treeview header, which they didn't otherwise need. Since the tree module provided access to the new treeview utilities through the old tree API, and was used by front ends with no changes for the new treeview API, the tree layer refcounted initialisations of treeview-based widgets, and only called the underlying treeview init/fini functions when needed. This change moves that refcounting into the treeview module. Now the hotlist, global history, cookie manager, and ssl cert viewer widgets call call treeview init/fini as part of their own initialisation and finalisation. This means that front ends using the correct APIs for treeview-based widgets don't need to know anything about the underlying treeview, and the tree module compatibility layer has had its treeview refcounting removed. Finally, the treeview_init function took a font size parameter. Now it does not and lit gets font size from config. We probably want to add a new `treeview_font_size` option to nsoptions, and have differnent defaults on different platforms. 12pt on RISC OS, and 11pt elsewhere, most likely.
1640 lines
37 KiB
C
1640 lines
37 KiB
C
/*
|
|
* Copyright 2012 John-Mark Bell <jmb@netsurf-browser.org>
|
|
* Copyright 2013 Michael Drake <tlsa@netsurf-browser.org>
|
|
*
|
|
* 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 <errno.h>
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <dom/dom.h>
|
|
#include <dom/bindings/hubbub/parser.h>
|
|
|
|
#include "utils/corestrings.h"
|
|
#include "utils/messages.h"
|
|
#include "utils/utils.h"
|
|
#include "utils/utf8.h"
|
|
#include "utils/libdom.h"
|
|
#include "utils/log.h"
|
|
#include "utils/nsurl.h"
|
|
#include "content/urldb.h"
|
|
|
|
#include "netsurf/misc.h"
|
|
#include "desktop/gui_internal.h"
|
|
#include "desktop/hotlist.h"
|
|
#include "desktop/treeview.h"
|
|
#include "netsurf/browser_window.h"
|
|
|
|
#define N_DAYS 28
|
|
#define N_SEC_PER_DAY (60 * 60 * 24)
|
|
|
|
enum hotlist_fields {
|
|
HL_TITLE,
|
|
HL_URL,
|
|
HL_LAST_VISIT,
|
|
HL_VISITS,
|
|
HL_FOLDER,
|
|
HL_N_FIELDS
|
|
};
|
|
|
|
struct hotlist_folder {
|
|
treeview_node *folder;
|
|
struct treeview_field_data data;
|
|
};
|
|
|
|
struct hotlist_ctx {
|
|
treeview *tree;
|
|
struct treeview_field_desc fields[HL_N_FIELDS];
|
|
bool built;
|
|
struct hotlist_folder *default_folder;
|
|
};
|
|
struct hotlist_ctx hl_ctx;
|
|
|
|
struct hotlist_entry {
|
|
nsurl *url;
|
|
treeview_node *entry;
|
|
|
|
struct treeview_field_data data[HL_N_FIELDS - 1];
|
|
};
|
|
|
|
|
|
/**
|
|
* Set a hotlist entry's data from the url_data.
|
|
*
|
|
* \param e hotlist entry to set up.
|
|
* \param data Data associated with entry's URL.
|
|
* \return NSERROR_OK on success, appropriate error otherwise.
|
|
*/
|
|
static nserror hotlist_create_treeview_field_visits_data(
|
|
struct hotlist_entry *e, const struct url_data *data)
|
|
{
|
|
char buffer[16];
|
|
const char *last_visited;
|
|
char *last_visited2;
|
|
int len;
|
|
|
|
/* Last visited */
|
|
if (data->visits != 0) {
|
|
last_visited = ctime(&data->last_visit);
|
|
last_visited2 = strdup(last_visited);
|
|
len = 24;
|
|
} else {
|
|
last_visited2 = strdup("-");
|
|
len = 1;
|
|
}
|
|
if (last_visited2 == NULL) {
|
|
return NSERROR_NOMEM;
|
|
|
|
} else if (len == 24) {
|
|
assert(last_visited2[24] == '\n');
|
|
last_visited2[24] = '\0';
|
|
}
|
|
|
|
e->data[HL_LAST_VISIT].field = hl_ctx.fields[HL_LAST_VISIT].field;
|
|
e->data[HL_LAST_VISIT].value = last_visited2;
|
|
e->data[HL_LAST_VISIT].value_len = len;
|
|
|
|
/* Visits */
|
|
len = snprintf(buffer, 16, "%u", data->visits);
|
|
if (len == 16) {
|
|
len--;
|
|
buffer[len] = '\0';
|
|
}
|
|
|
|
e->data[HL_VISITS].field = hl_ctx.fields[HL_VISITS].field;
|
|
e->data[HL_VISITS].value = strdup(buffer);
|
|
if (e->data[HL_VISITS].value == NULL) {
|
|
free((void*)e->data[HL_LAST_VISIT].value);
|
|
return NSERROR_NOMEM;
|
|
}
|
|
e->data[HL_VISITS].value_len = len;
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/**
|
|
* Set a hotlist entry's data from the url_data.
|
|
*
|
|
* \param e hotlist entry to set up
|
|
* \param title Title for entry, or NULL if using title from data
|
|
* \param data Data associated with entry's URL
|
|
* \return NSERROR_OK on success, appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_create_treeview_field_data(
|
|
struct hotlist_entry *e, const char *title,
|
|
const struct url_data *data)
|
|
{
|
|
nserror err;
|
|
|
|
/* "URL" field */
|
|
e->data[HL_URL].field = hl_ctx.fields[HL_URL].field;
|
|
e->data[HL_URL].value = nsurl_access(e->url);
|
|
e->data[HL_URL].value_len = nsurl_length(e->url);
|
|
|
|
/* "Title" field */
|
|
if (title == NULL) {
|
|
/* Title not provided; use one from URL data */
|
|
title = (data->title != NULL) ?
|
|
data->title : nsurl_access(e->url);
|
|
}
|
|
|
|
e->data[HL_TITLE].field = hl_ctx.fields[HL_TITLE].field;
|
|
e->data[HL_TITLE].value = strdup(title);
|
|
if (e->data[HL_TITLE].value == NULL) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
e->data[HL_TITLE].value_len = (e->data[HL_TITLE].value != NULL) ?
|
|
strlen(title) : 0;
|
|
|
|
/* "Last visited" and "Visits" fields */
|
|
err = hotlist_create_treeview_field_visits_data(e, data);
|
|
if (err != NSERROR_OK) {
|
|
free((void*)e->data[HL_TITLE].value);
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/**
|
|
* Add a hotlist entry to the treeview
|
|
*
|
|
* \param e Entry to add to treeview
|
|
* \param relation Existing node to insert as relation of, or NULL
|
|
* \param rel Folder's relationship to relation
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*
|
|
* It is assumed that the entry is unique (for its URL) in the global
|
|
* hotlist table
|
|
*/
|
|
static nserror hotlist_entry_insert(struct hotlist_entry *e,
|
|
treeview_node *relation, enum treeview_relationship rel)
|
|
{
|
|
nserror err;
|
|
|
|
err = treeview_create_node_entry(hl_ctx.tree, &(e->entry),
|
|
relation, rel, e->data, e, hl_ctx.built ?
|
|
TREE_OPTION_NONE : TREE_OPTION_SUPPRESS_RESIZE |
|
|
TREE_OPTION_SUPPRESS_REDRAW);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/**
|
|
* Delete a hotlist entry
|
|
*
|
|
* This does not delete the treeview node, rather it should only be called from
|
|
* the treeview node delete event message.
|
|
*
|
|
* \param e Entry to delete
|
|
*/
|
|
static void hotlist_delete_entry_internal(struct hotlist_entry *e)
|
|
{
|
|
assert(e != NULL);
|
|
assert(e->entry == NULL);
|
|
|
|
/* Destroy fields */
|
|
free((void *)e->data[HL_TITLE].value); /* Eww */
|
|
free((void *)e->data[HL_LAST_VISIT].value); /* Eww */
|
|
free((void *)e->data[HL_VISITS].value); /* Eww */
|
|
nsurl_unref(e->url);
|
|
|
|
/* Destroy entry */
|
|
free(e);
|
|
}
|
|
|
|
|
|
/**
|
|
* Create hotlist entry data for URL.
|
|
*
|
|
* \param url URL for entry to add to hotlist.
|
|
* \param title Title for entry, or NULL if using title from data
|
|
* \param data URL data for the entry, or NULL
|
|
* \param entry Updated to new hotlist entry data
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_create_entry(nsurl *url, const char *title,
|
|
const struct url_data *data, struct hotlist_entry **entry)
|
|
{
|
|
nserror err;
|
|
struct hotlist_entry *e;
|
|
|
|
assert(url != NULL);
|
|
|
|
*entry = NULL;
|
|
|
|
if (data == NULL) {
|
|
/* Get the URL data */
|
|
data = urldb_get_url_data(url);
|
|
if (data == NULL) {
|
|
/* No entry in database, so add one */
|
|
urldb_add_url(url);
|
|
/* now attempt to get url data */
|
|
data = urldb_get_url_data(url);
|
|
}
|
|
if (data == NULL) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
}
|
|
|
|
/* Create new local hotlist entry */
|
|
e = malloc(sizeof(struct hotlist_entry));
|
|
if (e == NULL) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
|
|
e->url = nsurl_ref(url);
|
|
e->entry = NULL;
|
|
|
|
err = hotlist_create_treeview_field_data(e, title, data);
|
|
if (err != NSERROR_OK) {
|
|
nsurl_unref(e->url);
|
|
free(e);
|
|
return err;
|
|
}
|
|
|
|
*entry = e;
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/**
|
|
* Add an entry to the hotlist (creates the entry).
|
|
*
|
|
* \param url URL for entry to add to hotlist.
|
|
* \param title Title for entry, or NULL if using title from data
|
|
* \param data URL data for the entry, or NULL
|
|
* \param relation Existing node to insert as relation of, or NULL
|
|
* \param rel Entry's relationship to relation
|
|
* \param entry Updated to new treeview entry node
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_add_entry_internal(nsurl *url, const char *title,
|
|
const struct url_data *data, treeview_node *relation,
|
|
enum treeview_relationship rel, treeview_node **entry)
|
|
{
|
|
nserror err;
|
|
struct hotlist_entry *e;
|
|
|
|
err = hotlist_create_entry(url, title, data, &e);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
err = hotlist_entry_insert(e, relation, rel);
|
|
if (err != NSERROR_OK) {
|
|
hotlist_delete_entry_internal(e);
|
|
return err;
|
|
}
|
|
|
|
/* Make this URL persistent */
|
|
urldb_set_url_persistence(url, true);
|
|
|
|
*entry = e->entry;
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/**
|
|
* Add folder to the hotlist (creates the folder).
|
|
*
|
|
* \param title Title for folder, or NULL if using title from data
|
|
* \param relation Existing node to insert as relation of, or NULL
|
|
* \param rel Folder's relationship to relation
|
|
* \param folder Updated to new hotlist folder data
|
|
* \param default_folder Add to the default folder.
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_add_folder_internal(
|
|
const char *title, treeview_node *relation,
|
|
enum treeview_relationship rel, struct hotlist_folder **folder,
|
|
bool default_folder)
|
|
{
|
|
struct hotlist_folder *f;
|
|
treeview_node_options_flags flags = TREE_OPTION_NONE;
|
|
treeview_node *n;
|
|
nserror err;
|
|
|
|
if (title == NULL) {
|
|
title = messages_get("NewFolderName");
|
|
}
|
|
|
|
/* Create the title field */
|
|
f = malloc(sizeof(struct hotlist_folder));
|
|
if (f == NULL) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
f->data.field = hl_ctx.fields[HL_FOLDER].field;
|
|
f->data.value = strdup(title);
|
|
if (f->data.value == NULL) {
|
|
free(f);
|
|
return NSERROR_NOMEM;
|
|
}
|
|
f->data.value_len = strlen(title);
|
|
|
|
if (!hl_ctx.built)
|
|
flags |= TREE_OPTION_SUPPRESS_RESIZE |
|
|
TREE_OPTION_SUPPRESS_REDRAW;
|
|
if (default_folder)
|
|
flags |= TREE_OPTION_SPECIAL_DIR;
|
|
|
|
err = treeview_create_node_folder(hl_ctx.tree,
|
|
&n, relation, rel, &f->data, f, flags);
|
|
if (err != NSERROR_OK) {
|
|
free((void*)f->data.value); /* Eww */
|
|
free(f);
|
|
return err;
|
|
}
|
|
|
|
f->folder = n;
|
|
*folder = f;
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
static nserror hotlist_tree_node_folder_cb(
|
|
struct treeview_node_msg msg, void *data)
|
|
{
|
|
struct hotlist_folder *f = data;
|
|
const char *old_text;
|
|
bool match;
|
|
|
|
switch (msg.msg) {
|
|
case TREE_MSG_NODE_DELETE:
|
|
if (f == hl_ctx.default_folder)
|
|
hl_ctx.default_folder = NULL;
|
|
free((void*)f->data.value); /* Eww */
|
|
free(f);
|
|
break;
|
|
|
|
case TREE_MSG_NODE_EDIT:
|
|
if (lwc_string_isequal(hl_ctx.fields[HL_FOLDER].field,
|
|
msg.data.node_edit.field, &match) ==
|
|
lwc_error_ok && match == true &&
|
|
msg.data.node_edit.text != NULL &&
|
|
msg.data.node_edit.text[0] != '\0') {
|
|
/* Requst to change the folder title text */
|
|
old_text = f->data.value;
|
|
f->data.value = strdup(msg.data.node_edit.text);
|
|
|
|
if (f->data.value == NULL) {
|
|
f->data.value = old_text;
|
|
} else {
|
|
f->data.value_len = strlen(f->data.value);
|
|
treeview_update_node_folder(hl_ctx.tree,
|
|
f->folder, &f->data, f);
|
|
free((void *)old_text);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TREE_MSG_NODE_LAUNCH:
|
|
break;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
/**
|
|
* callback for hotlist treeview entry manipulation.
|
|
*/
|
|
static nserror
|
|
hotlist_tree_node_entry_cb(struct treeview_node_msg msg, void *data)
|
|
{
|
|
struct hotlist_entry *e = data;
|
|
const char *old_text;
|
|
nsurl *old_url;
|
|
nsurl *url;
|
|
nserror err = NSERROR_OK;
|
|
bool match;
|
|
|
|
switch (msg.msg) {
|
|
case TREE_MSG_NODE_DELETE:
|
|
e->entry = NULL;
|
|
hotlist_delete_entry_internal(e);
|
|
break;
|
|
|
|
case TREE_MSG_NODE_EDIT:
|
|
if (lwc_string_isequal(hl_ctx.fields[HL_TITLE].field,
|
|
msg.data.node_edit.field, &match) ==
|
|
lwc_error_ok && match == true &&
|
|
msg.data.node_edit.text != NULL &&
|
|
msg.data.node_edit.text[0] != '\0') {
|
|
/* Requst to change the entry title text */
|
|
old_text = e->data[HL_TITLE].value;
|
|
e->data[HL_TITLE].value =
|
|
strdup(msg.data.node_edit.text);
|
|
|
|
if (e->data[HL_TITLE].value == NULL) {
|
|
e->data[HL_TITLE].value = old_text;
|
|
} else {
|
|
e->data[HL_TITLE].value_len =
|
|
strlen(e->data[HL_TITLE].value);
|
|
treeview_update_node_entry(hl_ctx.tree,
|
|
e->entry, e->data, e);
|
|
free((void *)old_text);
|
|
}
|
|
|
|
} else if (lwc_string_isequal(hl_ctx.fields[HL_URL].field,
|
|
msg.data.node_edit.field, &match) ==
|
|
lwc_error_ok && match == true &&
|
|
msg.data.node_edit.text != NULL &&
|
|
msg.data.node_edit.text[0] != '\0') {
|
|
/* Requst to change the entry URL text */
|
|
err = nsurl_create(msg.data.node_edit.text, &url);
|
|
if (err == NSERROR_OK) {
|
|
old_url = e->url;
|
|
|
|
e->url = url;
|
|
e->data[HL_URL].value = nsurl_access(url);
|
|
e->data[HL_URL].value_len = nsurl_length(e->url);
|
|
|
|
treeview_update_node_entry(hl_ctx.tree,
|
|
e->entry, e->data, e);
|
|
nsurl_unref(old_url);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TREE_MSG_NODE_LAUNCH:
|
|
{
|
|
struct browser_window *existing = NULL;
|
|
enum browser_window_create_flags flags = BW_CREATE_HISTORY;
|
|
|
|
/* TODO: Set existing to window that new tab appears in */
|
|
|
|
if (msg.data.node_launch.mouse &
|
|
(BROWSER_MOUSE_MOD_1 | BROWSER_MOUSE_MOD_2) ||
|
|
existing == NULL) {
|
|
/* Shift or Ctrl launch, open in new window rather
|
|
* than tab. */
|
|
/* TODO: flags ^= BW_CREATE_TAB; */
|
|
}
|
|
|
|
err = browser_window_create(flags, e->url, NULL,
|
|
existing, NULL);
|
|
}
|
|
break;
|
|
}
|
|
return err;
|
|
}
|
|
|
|
struct treeview_callback_table hl_tree_cb_t = {
|
|
.folder = hotlist_tree_node_folder_cb,
|
|
.entry = hotlist_tree_node_entry_cb
|
|
};
|
|
|
|
|
|
|
|
typedef struct {
|
|
treeview *tree;
|
|
treeview_node *rel;
|
|
enum treeview_relationship relshp;
|
|
bool last_was_h4;
|
|
dom_string *title;
|
|
} hotlist_load_ctx;
|
|
|
|
|
|
/**
|
|
* Parse an entry represented as a li.
|
|
*
|
|
* \param li DOM node for parsed li
|
|
* \param ctx Our hotlist loading context.
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_load_entry(dom_node *li, hotlist_load_ctx *ctx)
|
|
{
|
|
dom_node *a;
|
|
dom_string *title1, *url1;
|
|
const char *title;
|
|
nsurl *url;
|
|
dom_exception derror;
|
|
nserror err;
|
|
|
|
/* The li must contain an "a" element */
|
|
a = libdom_find_first_element(li, corestring_lwc_a);
|
|
if (a == NULL) {
|
|
LOG("Missing <a> in <li>");
|
|
return NSERROR_INVALID;
|
|
}
|
|
|
|
derror = dom_node_get_text_content(a, &title1);
|
|
if (derror != DOM_NO_ERR) {
|
|
LOG("No title");
|
|
dom_node_unref(a);
|
|
return NSERROR_INVALID;
|
|
}
|
|
|
|
derror = dom_element_get_attribute(a, corestring_dom_href, &url1);
|
|
if (derror != DOM_NO_ERR || url1 == NULL) {
|
|
LOG("No URL");
|
|
dom_string_unref(title1);
|
|
dom_node_unref(a);
|
|
return NSERROR_INVALID;
|
|
}
|
|
dom_node_unref(a);
|
|
|
|
if (title1 != NULL) {
|
|
title = dom_string_data(title1);
|
|
} else {
|
|
title = messages_get("NoTitle");
|
|
}
|
|
|
|
/* Need to get URL as a nsurl object */
|
|
err = nsurl_create(dom_string_data(url1), &url);
|
|
dom_string_unref(url1);
|
|
|
|
if (err != NSERROR_OK) {
|
|
LOG("Failed normalising '%s'", dom_string_data(url1));
|
|
|
|
if (title1 != NULL) {
|
|
dom_string_unref(title1);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
/* Add the entry */
|
|
err = hotlist_add_entry_internal(url, title, NULL, ctx->rel,
|
|
ctx->relshp, &ctx->rel);
|
|
nsurl_unref(url);
|
|
if (title1 != NULL) {
|
|
dom_string_unref(title1);
|
|
}
|
|
ctx->relshp = TREE_REL_NEXT_SIBLING;
|
|
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/*
|
|
* Callback for libdom_iterate_child_elements, which dispite the namespace is
|
|
* a NetSurf function.
|
|
*
|
|
* \param node Node that is a child of the directory UL node
|
|
* \param ctx Our hotlist loading context.
|
|
*/
|
|
static nserror hotlist_load_directory_cb(dom_node *node, void *ctx);
|
|
|
|
/**
|
|
* Parse a directory represented as a ul.
|
|
*
|
|
* \param ul DOM node for parsed ul.
|
|
* \param ctx The hotlist context.
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_load_directory(dom_node *ul, hotlist_load_ctx *ctx)
|
|
{
|
|
assert(ul != NULL);
|
|
assert(ctx != NULL);
|
|
|
|
return libdom_iterate_child_elements(ul,
|
|
hotlist_load_directory_cb, ctx);
|
|
}
|
|
|
|
|
|
/* Documented above, in forward declaration */
|
|
nserror hotlist_load_directory_cb(dom_node *node, void *ctx)
|
|
{
|
|
/* TODO: return appropriate errors */
|
|
hotlist_load_ctx *current_ctx = ctx;
|
|
dom_string *name;
|
|
dom_exception error;
|
|
nserror err;
|
|
|
|
/* The ul may contain entries as a li, or directories as
|
|
* an h4 followed by a ul. Non-element nodes may be present
|
|
* (eg. text, comments), and are ignored. */
|
|
|
|
error = dom_node_get_node_name(node, &name);
|
|
if (error != DOM_NO_ERR || name == NULL)
|
|
return NSERROR_DOM;
|
|
|
|
if (dom_string_caseless_lwc_isequal(name, corestring_lwc_li)) {
|
|
/* Entry handling */
|
|
hotlist_load_entry(node, current_ctx);
|
|
current_ctx->last_was_h4 = false;
|
|
|
|
} else if (dom_string_caseless_lwc_isequal(name, corestring_lwc_h4)) {
|
|
/* Directory handling, part 1: Get title from H4 */
|
|
dom_string *title;
|
|
|
|
error = dom_node_get_text_content(node, &title);
|
|
if (error != DOM_NO_ERR || title == NULL) {
|
|
LOG("Empty <h4> or memory exhausted.");
|
|
dom_string_unref(name);
|
|
return NSERROR_DOM;
|
|
}
|
|
|
|
if (current_ctx->title != NULL)
|
|
dom_string_unref(current_ctx->title);
|
|
current_ctx->title = title;
|
|
current_ctx->last_was_h4 = true;
|
|
|
|
} else if (current_ctx->last_was_h4 &&
|
|
dom_string_caseless_lwc_isequal(name,
|
|
corestring_lwc_ul)) {
|
|
/* Directory handling, part 2: Make node, and handle children */
|
|
const char *title;
|
|
dom_string *id;
|
|
struct hotlist_folder *f;
|
|
hotlist_load_ctx new_ctx;
|
|
treeview_node *rel;
|
|
bool default_folder = false;
|
|
|
|
/* Check if folder should be default folder */
|
|
error = dom_element_get_attribute(node, corestring_dom_id, &id);
|
|
if (error != DOM_NO_ERR) {
|
|
dom_string_unref(name);
|
|
return NSERROR_NOMEM;
|
|
}
|
|
if (id != NULL) {
|
|
if (dom_string_lwc_isequal(id, corestring_lwc_default))
|
|
default_folder = true;
|
|
|
|
dom_string_unref(id);
|
|
}
|
|
|
|
title = dom_string_data(current_ctx->title);
|
|
|
|
/* Add folder node */
|
|
err = hotlist_add_folder_internal(title, current_ctx->rel,
|
|
current_ctx->relshp, &f, default_folder);
|
|
if (err != NSERROR_OK) {
|
|
dom_string_unref(name);
|
|
return NSERROR_NOMEM;
|
|
}
|
|
|
|
if (default_folder)
|
|
hl_ctx.default_folder = f;
|
|
|
|
rel = f->folder;
|
|
current_ctx->rel = rel;
|
|
current_ctx->relshp = TREE_REL_NEXT_SIBLING;
|
|
|
|
new_ctx.tree = current_ctx->tree;
|
|
new_ctx.rel = rel;
|
|
new_ctx.relshp = TREE_REL_FIRST_CHILD;
|
|
new_ctx.last_was_h4 = false;
|
|
new_ctx.title = NULL;
|
|
|
|
/* And load its contents */
|
|
err = hotlist_load_directory(node, &new_ctx);
|
|
if (err != NSERROR_OK) {
|
|
dom_string_unref(name);
|
|
return NSERROR_NOMEM;
|
|
}
|
|
|
|
if (new_ctx.title != NULL) {
|
|
dom_string_unref(new_ctx.title);
|
|
new_ctx.title = NULL;
|
|
}
|
|
current_ctx->last_was_h4 = false;
|
|
} else {
|
|
current_ctx->last_was_h4 = false;
|
|
}
|
|
|
|
dom_string_unref(name);
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/*
|
|
* Get path for writing hotlist to
|
|
*
|
|
* \param path The final path of the hotlist
|
|
* \param loaded Updated to the path to write the holist to
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_get_temp_path(const char *path, char **temp_path)
|
|
{
|
|
const char *extension = "-bk";
|
|
char *joined;
|
|
int len;
|
|
|
|
len = strlen(path) + strlen(extension);
|
|
|
|
joined = malloc(len + 1);
|
|
if (joined == NULL) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
|
|
if (snprintf(joined, len + 1, "%s%s", path, extension) != len) {
|
|
free(joined);
|
|
return NSERROR_UNKNOWN;
|
|
}
|
|
|
|
*temp_path = joined;
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/*
|
|
* Load the hotlist data from file
|
|
*
|
|
* \param path The path to load the hotlist file from, or NULL
|
|
* \param loaded Updated to true iff hotlist file loaded, else set false
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_load(const char *path, bool *loaded)
|
|
{
|
|
dom_document *document;
|
|
dom_node *html, *body, *ul;
|
|
hotlist_load_ctx ctx;
|
|
char *temp_path;
|
|
nserror err;
|
|
|
|
*loaded = false;
|
|
|
|
/* Handle no path */
|
|
if (path == NULL) {
|
|
LOG("No hotlist file path provided.");
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
/* Get temp hotlist write path */
|
|
err = hotlist_get_temp_path(path, &temp_path);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
/* Load hotlist file */
|
|
err = libdom_parse_file(path, "iso-8859-1", &document);
|
|
if (err != NSERROR_OK) {
|
|
err = libdom_parse_file(temp_path, "iso-8859-1", &document);
|
|
if (err != NSERROR_OK) {
|
|
free(temp_path);
|
|
return err;
|
|
}
|
|
}
|
|
free(temp_path);
|
|
|
|
/* Find HTML element */
|
|
html = libdom_find_first_element((dom_node *) document,
|
|
corestring_lwc_html);
|
|
if (html == NULL) {
|
|
dom_node_unref(document);
|
|
guit->misc->warning("TreeLoadError", "(<html> not found)");
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
/* Find BODY element */
|
|
body = libdom_find_first_element(html, corestring_lwc_body);
|
|
if (body == NULL) {
|
|
dom_node_unref(html);
|
|
dom_node_unref(document);
|
|
guit->misc->warning("TreeLoadError", "(<html>...<body> not found)");
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
/* Find UL element */
|
|
ul = libdom_find_first_element(body, corestring_lwc_ul);
|
|
if (ul == NULL) {
|
|
dom_node_unref(body);
|
|
dom_node_unref(html);
|
|
dom_node_unref(document);
|
|
guit->misc->warning("TreeLoadError",
|
|
"(<html>...<body>...<ul> not found.)");
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
/* Set up the hotlist loading context */
|
|
ctx.tree = hl_ctx.tree;
|
|
ctx.rel = NULL;
|
|
ctx.relshp = TREE_REL_FIRST_CHILD;
|
|
ctx.last_was_h4 = false;
|
|
ctx.title = NULL;
|
|
|
|
err = hotlist_load_directory(ul, &ctx);
|
|
|
|
if (ctx.title != NULL) {
|
|
dom_string_unref(ctx.title);
|
|
ctx.title = NULL;
|
|
}
|
|
|
|
dom_node_unref(ul);
|
|
dom_node_unref(body);
|
|
dom_node_unref(html);
|
|
dom_node_unref(document);
|
|
|
|
if (err != NSERROR_OK) {
|
|
guit->misc->warning("TreeLoadError", "(Failed building tree.)");
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
*loaded = true;
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/*
|
|
* Generate default hotlist
|
|
*
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_generate(void)
|
|
{
|
|
int i;
|
|
struct hotlist_folder *f;
|
|
treeview_node *e;
|
|
const char *title;
|
|
nserror err;
|
|
nsurl *url;
|
|
static const struct {
|
|
const char *url;
|
|
const char *msg_key;
|
|
} default_entries[] = {
|
|
{ "http://www.netsurf-browser.org/",
|
|
"HotlistHomepage" },
|
|
{ "http://www.netsurf-browser.org/downloads/",
|
|
"HotlistDownloads" },
|
|
{ "http://www.netsurf-browser.org/documentation",
|
|
"HotlistDocumentation" },
|
|
{ "http://www.netsurf-browser.org/contact",
|
|
"HotlistContact" }
|
|
};
|
|
const int n_entries = sizeof(default_entries) /
|
|
sizeof(default_entries[0]);
|
|
|
|
/* First make "NetSurf" folder for defualt entries */
|
|
title = "NetSurf";
|
|
err = hotlist_add_folder_internal(title, NULL,
|
|
TREE_REL_FIRST_CHILD, &f, false);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
/* And add entries as children of folder node */
|
|
for (i = n_entries - 1; i >= 0; i--) {
|
|
/* Get URL as nsurl object */
|
|
err = nsurl_create(default_entries[i].url, &url);
|
|
if (err != NSERROR_OK) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
|
|
title = messages_get(default_entries[i].msg_key);
|
|
|
|
/* Build the node */
|
|
err = hotlist_add_entry_internal(url, title,
|
|
NULL, f->folder, TREE_REL_FIRST_CHILD, &e);
|
|
nsurl_unref(url);
|
|
|
|
if (err != NSERROR_OK) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/* Save the hotlist to to a file at the given path
|
|
*
|
|
* \param path Path to save hostlist file to.
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_save(const char *path)
|
|
{
|
|
nserror res = NSERROR_OK;
|
|
char *temp_path;
|
|
|
|
/* Get path to export to */
|
|
res = hotlist_get_temp_path(path, &temp_path);
|
|
if (res != NSERROR_OK) {
|
|
return res;
|
|
}
|
|
|
|
/* Export to temp path */
|
|
res = hotlist_export(temp_path, NULL);
|
|
if (res != NSERROR_OK) {
|
|
goto cleanup;
|
|
}
|
|
|
|
/* Remove old hotlist to handle non-POSIX rename() implementations. */
|
|
(void)remove(path);
|
|
|
|
/* Replace any old hotlist file with the one we just saved */
|
|
if (rename(temp_path, path) != 0) {
|
|
res = NSERROR_SAVE_FAILED;
|
|
LOG("Error renaming hotlist: %s.", strerror(errno));
|
|
goto cleanup;
|
|
}
|
|
|
|
cleanup:
|
|
free(temp_path);
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
struct treeview_export_walk_ctx {
|
|
FILE *fp;
|
|
};
|
|
|
|
/** Callback for treeview_walk node entering */
|
|
static nserror hotlist_export_enter_cb(void *ctx, void *node_data,
|
|
enum treeview_node_type type, bool *abort)
|
|
{
|
|
struct treeview_export_walk_ctx *tw = ctx;
|
|
nserror ret;
|
|
|
|
if (type == TREE_NODE_ENTRY) {
|
|
struct hotlist_entry *e = node_data;
|
|
char *t_text;
|
|
char *u_text;
|
|
|
|
ret = utf8_to_html(e->data[HL_TITLE].value, "iso-8859-1",
|
|
e->data[HL_TITLE].value_len, &t_text);
|
|
if (ret != NSERROR_OK)
|
|
return NSERROR_SAVE_FAILED;
|
|
|
|
ret = utf8_to_html(e->data[HL_URL].value, "iso-8859-1",
|
|
e->data[HL_URL].value_len, &u_text);
|
|
if (ret != NSERROR_OK) {
|
|
free(t_text);
|
|
return NSERROR_SAVE_FAILED;
|
|
}
|
|
|
|
fprintf(tw->fp, "<li><a href=\"%s\">%s</a></li>\n",
|
|
u_text, t_text);
|
|
|
|
free(t_text);
|
|
free(u_text);
|
|
|
|
} else if (type == TREE_NODE_FOLDER) {
|
|
struct hotlist_folder *f = node_data;
|
|
char *f_text;
|
|
|
|
ret = utf8_to_html(f->data.value, "iso-8859-1",
|
|
f->data.value_len, &f_text);
|
|
if (ret != NSERROR_OK)
|
|
return NSERROR_SAVE_FAILED;
|
|
|
|
if (f == hl_ctx.default_folder) {
|
|
fprintf(tw->fp, "<h4>%s</h4>\n<ul id=\"default\">\n",
|
|
f_text);
|
|
} else {
|
|
fprintf(tw->fp, "<h4>%s</h4>\n<ul>\n", f_text);
|
|
}
|
|
|
|
free(f_text);
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
/** Callback for treeview_walk node leaving */
|
|
static nserror hotlist_export_leave_cb(void *ctx, void *node_data,
|
|
enum treeview_node_type type, bool *abort)
|
|
{
|
|
struct treeview_export_walk_ctx *tw = ctx;
|
|
|
|
if (type == TREE_NODE_FOLDER) {
|
|
fputs("</ul>\n", tw->fp);
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_export(const char *path, const char *title)
|
|
{
|
|
struct treeview_export_walk_ctx tw;
|
|
nserror err;
|
|
FILE *fp;
|
|
|
|
fp = fopen(path, "w");
|
|
if (fp == NULL)
|
|
return NSERROR_SAVE_FAILED;
|
|
|
|
if (title == NULL)
|
|
title = "NetSurf hotlist";
|
|
|
|
/* The Acorn Browse Hotlist format, which we mimic[*], is invalid HTML
|
|
* claming to be valid.
|
|
* [*] Why? */
|
|
fputs("<!DOCTYPE html "
|
|
"PUBLIC \"//W3C/DTD HTML 4.01//EN\" "
|
|
"\"http://www.w3.org/TR/html4/strict.dtd\">\n", fp);
|
|
fputs("<html>\n<head>\n", fp);
|
|
fputs("<meta http-equiv=\"Content-Type\" "
|
|
"content=\"text/html; charset=iso-8859-1\">\n", fp);
|
|
fprintf(fp, "<title>%s</title>\n", title);
|
|
fputs("</head>\n<body>\n<ul>\n", fp);
|
|
|
|
tw.fp = fp;
|
|
err = treeview_walk(hl_ctx.tree, NULL,
|
|
hotlist_export_enter_cb,
|
|
hotlist_export_leave_cb,
|
|
&tw, TREE_NODE_ENTRY | TREE_NODE_FOLDER);
|
|
if (err != NSERROR_OK)
|
|
return err;
|
|
|
|
fputs("</ul>\n</body>\n</html>\n", fp);
|
|
|
|
fclose(fp);
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
struct hotlist_iterate_ctx {
|
|
hotlist_folder_enter_cb enter_cb;
|
|
hotlist_address_cb address_cb;
|
|
hotlist_folder_leave_cb leave_cb;
|
|
void *ctx;
|
|
};
|
|
/** Callback for hotlist_iterate node entering */
|
|
static nserror hotlist_iterate_enter_cb(void *ctx, void *node_data,
|
|
enum treeview_node_type type, bool *abort)
|
|
{
|
|
struct hotlist_iterate_ctx *data = ctx;
|
|
|
|
if (type == TREE_NODE_ENTRY && data->address_cb != NULL) {
|
|
struct hotlist_entry *e = node_data;
|
|
data->address_cb(data->ctx, e->url,
|
|
e->data[HL_TITLE].value);
|
|
|
|
} else if (type == TREE_NODE_FOLDER && data->enter_cb != NULL) {
|
|
struct hotlist_folder *f = node_data;
|
|
data->enter_cb(data->ctx, f->data.value);
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
/** Callback for hotlist_iterate node leaving */
|
|
static nserror hotlist_iterate_leave_cb(void *ctx, void *node_data,
|
|
enum treeview_node_type type, bool *abort)
|
|
{
|
|
struct hotlist_iterate_ctx *data = ctx;
|
|
|
|
if (type == TREE_NODE_FOLDER && data->leave_cb != NULL) {
|
|
data->leave_cb(data->ctx);
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_iterate(void *ctx,
|
|
hotlist_folder_enter_cb enter_cb,
|
|
hotlist_address_cb address_cb,
|
|
hotlist_folder_leave_cb leave_cb)
|
|
{
|
|
struct hotlist_iterate_ctx data;
|
|
nserror err;
|
|
|
|
data.enter_cb = enter_cb;
|
|
data.address_cb = address_cb;
|
|
data.leave_cb = leave_cb;
|
|
data.ctx = ctx;
|
|
|
|
err = treeview_walk(hl_ctx.tree, NULL,
|
|
hotlist_iterate_enter_cb,
|
|
hotlist_iterate_leave_cb,
|
|
&data, TREE_NODE_ENTRY | TREE_NODE_FOLDER);
|
|
if (err != NSERROR_OK)
|
|
return err;
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/**
|
|
* Initialise the treeview entry feilds
|
|
*
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_initialise_entry_fields(void)
|
|
{
|
|
int i;
|
|
const char *label;
|
|
|
|
for (i = 0; i < HL_N_FIELDS; i++)
|
|
hl_ctx.fields[i].field = NULL;
|
|
|
|
hl_ctx.fields[HL_TITLE].flags = TREE_FLAG_DEFAULT |
|
|
TREE_FLAG_ALLOW_EDIT;
|
|
label = "TreeviewLabelTitle";
|
|
label = messages_get(label);
|
|
if (lwc_intern_string(label, strlen(label),
|
|
&hl_ctx.fields[HL_TITLE].field) !=
|
|
lwc_error_ok) {
|
|
goto error;
|
|
}
|
|
|
|
hl_ctx.fields[HL_URL].flags = TREE_FLAG_ALLOW_EDIT |
|
|
TREE_FLAG_COPY_TEXT;
|
|
label = "TreeviewLabelURL";
|
|
label = messages_get(label);
|
|
if (lwc_intern_string(label, strlen(label),
|
|
&hl_ctx.fields[HL_URL].field) !=
|
|
lwc_error_ok) {
|
|
goto error;
|
|
}
|
|
|
|
hl_ctx.fields[HL_LAST_VISIT].flags = TREE_FLAG_SHOW_NAME;
|
|
label = "TreeviewLabelLastVisit";
|
|
label = messages_get(label);
|
|
if (lwc_intern_string(label, strlen(label),
|
|
&hl_ctx.fields[HL_LAST_VISIT].field) !=
|
|
lwc_error_ok) {
|
|
goto error;
|
|
}
|
|
|
|
hl_ctx.fields[HL_VISITS].flags = TREE_FLAG_SHOW_NAME;
|
|
label = "TreeviewLabelVisits";
|
|
label = messages_get(label);
|
|
if (lwc_intern_string(label, strlen(label),
|
|
&hl_ctx.fields[HL_VISITS].field) !=
|
|
lwc_error_ok) {
|
|
goto error;
|
|
}
|
|
|
|
hl_ctx.fields[HL_FOLDER].flags = TREE_FLAG_DEFAULT |
|
|
TREE_FLAG_ALLOW_EDIT;
|
|
label = "TreeviewLabelFolder";
|
|
label = messages_get(label);
|
|
if (lwc_intern_string(label, strlen(label),
|
|
&hl_ctx.fields[HL_FOLDER].field) !=
|
|
lwc_error_ok) {
|
|
return false;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
|
|
error:
|
|
for (i = 0; i < HL_N_FIELDS; i++)
|
|
if (hl_ctx.fields[i].field != NULL)
|
|
lwc_string_unref(hl_ctx.fields[i].field);
|
|
|
|
return NSERROR_UNKNOWN;
|
|
}
|
|
|
|
|
|
/*
|
|
* Populate the hotlist from file, or generate default hotlist if no file
|
|
*
|
|
* \param path The path to load the hotlist file from, or NULL
|
|
* \return NSERROR_OK on success, or appropriate error otherwise
|
|
*/
|
|
static nserror hotlist_populate(const char *path)
|
|
{
|
|
nserror err;
|
|
bool loaded;
|
|
|
|
/* Load hotlist file */
|
|
err = hotlist_load(path, &loaded);
|
|
|
|
if (loaded && err == NSERROR_OK)
|
|
return err;
|
|
|
|
/* Couldn't load hotlist, generate a default one */
|
|
err = hotlist_generate();
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_init(struct core_window_callback_table *cw_t,
|
|
void *core_window_handle, const char *path)
|
|
{
|
|
nserror err;
|
|
|
|
err = treeview_init();
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
LOG("Loading hotlist");
|
|
|
|
hl_ctx.tree = NULL;
|
|
hl_ctx.built = false;
|
|
hl_ctx.default_folder = NULL;
|
|
|
|
/* Init. hotlist treeview entry fields */
|
|
err = hotlist_initialise_entry_fields();
|
|
if (err != NSERROR_OK) {
|
|
hl_ctx.tree = NULL;
|
|
return err;
|
|
}
|
|
|
|
/* Create the hotlist treeview */
|
|
err = treeview_create(&hl_ctx.tree, &hl_tree_cb_t,
|
|
HL_N_FIELDS, hl_ctx.fields,
|
|
cw_t, core_window_handle,
|
|
TREEVIEW_NO_FLAGS);
|
|
if (err != NSERROR_OK) {
|
|
hl_ctx.tree = NULL;
|
|
return err;
|
|
}
|
|
|
|
/* Populate the hotlist */
|
|
err = hotlist_populate(path);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
/* Hotlist tree is built
|
|
* We suppress the treeview height callback on entry insertion before
|
|
* the treeview is built. */
|
|
hl_ctx.built = true;
|
|
|
|
/* Inform client of window height */
|
|
treeview_get_height(hl_ctx.tree);
|
|
|
|
LOG("Loaded hotlist");
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_fini(const char *path)
|
|
{
|
|
int i;
|
|
nserror err;
|
|
|
|
LOG("Finalising hotlist");
|
|
|
|
/* Save the hotlist */
|
|
err = hotlist_save(path);
|
|
if (err != NSERROR_OK) {
|
|
LOG("Problem saving the hotlist.");
|
|
}
|
|
|
|
/* Destroy the hotlist treeview */
|
|
err = treeview_destroy(hl_ctx.tree);
|
|
hl_ctx.built = false;
|
|
|
|
/* Free hotlist treeview entry fields */
|
|
for (i = 0; i < HL_N_FIELDS; i++)
|
|
if (hl_ctx.fields[i].field != NULL)
|
|
lwc_string_unref(hl_ctx.fields[i].field);
|
|
|
|
err = treeview_fini();
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
LOG("Finalised hotlist");
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_add_url(nsurl *url)
|
|
{
|
|
treeview_node *entry;
|
|
nserror err;
|
|
|
|
/* If we don't have a hotlist at the moment, just return OK */
|
|
if (hl_ctx.tree == NULL)
|
|
return NSERROR_OK;
|
|
|
|
/* Make the default folder, if we don't have one */
|
|
if (hl_ctx.default_folder == NULL) {
|
|
const char *temp = messages_get("HotlistDefaultFolderName");
|
|
struct hotlist_folder *f;
|
|
err = hotlist_add_folder_internal(temp, NULL,
|
|
TREE_REL_FIRST_CHILD, &f, true);
|
|
if (err != NSERROR_OK)
|
|
return err;
|
|
|
|
if (f == NULL)
|
|
return NSERROR_NOMEM;
|
|
|
|
hl_ctx.default_folder = f;
|
|
}
|
|
|
|
/* Add new entry to default folder */
|
|
err = hotlist_add_entry_internal(url, NULL, NULL,
|
|
hl_ctx.default_folder->folder,
|
|
TREE_REL_FIRST_CHILD, &entry);
|
|
if (err != NSERROR_OK)
|
|
return err;
|
|
|
|
/* Ensure default folder is expanded */
|
|
err = treeview_node_expand(hl_ctx.tree, hl_ctx.default_folder->folder);
|
|
if (err != NSERROR_OK)
|
|
return err;
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
struct treeview_has_url_walk_ctx {
|
|
nsurl *url;
|
|
bool found;
|
|
};
|
|
/** Callback for treeview_walk */
|
|
static nserror hotlist_has_url_walk_cb(void *ctx, void *node_data,
|
|
enum treeview_node_type type, bool *abort)
|
|
{
|
|
struct treeview_has_url_walk_ctx *tw = ctx;
|
|
|
|
if (type == TREE_NODE_ENTRY) {
|
|
struct hotlist_entry *e = node_data;
|
|
|
|
if (nsurl_compare(e->url, tw->url, NSURL_COMPLETE) == true) {
|
|
/* Found what we're looking for */
|
|
tw->found = true;
|
|
*abort = true;
|
|
}
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
/* Exported interface, documented in hotlist.h */
|
|
bool hotlist_has_url(nsurl *url)
|
|
{
|
|
nserror err;
|
|
struct treeview_has_url_walk_ctx tw = {
|
|
.url = url,
|
|
.found = false
|
|
};
|
|
|
|
if (hl_ctx.built == false)
|
|
return false;
|
|
|
|
err = treeview_walk(hl_ctx.tree, NULL, hotlist_has_url_walk_cb, NULL,
|
|
&tw, TREE_NODE_ENTRY);
|
|
if (err != NSERROR_OK)
|
|
return false;
|
|
|
|
return tw.found;
|
|
}
|
|
|
|
|
|
struct treeview_remove_url_walk_ctx {
|
|
nsurl *url;
|
|
};
|
|
/** Callback for treeview_walk */
|
|
static nserror hotlist_remove_url_walk_cb(void *ctx, void *node_data,
|
|
enum treeview_node_type type, bool *abort)
|
|
{
|
|
struct treeview_remove_url_walk_ctx *tw = ctx;
|
|
|
|
if (type == TREE_NODE_ENTRY) {
|
|
struct hotlist_entry *e = node_data;
|
|
|
|
if (nsurl_compare(e->url, tw->url, NSURL_COMPLETE) == true) {
|
|
/* Found what we're looking for: delete it */
|
|
treeview_delete_node(hl_ctx.tree, e->entry,
|
|
TREE_OPTION_NONE);
|
|
}
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
/* Exported interface, documented in hotlist.h */
|
|
void hotlist_remove_url(nsurl *url)
|
|
{
|
|
nserror err;
|
|
struct treeview_remove_url_walk_ctx tw = {
|
|
.url = url
|
|
};
|
|
|
|
if (hl_ctx.built == false)
|
|
return;
|
|
|
|
err = treeview_walk(hl_ctx.tree, NULL, NULL, hotlist_remove_url_walk_cb,
|
|
&tw, TREE_NODE_ENTRY);
|
|
if (err != NSERROR_OK)
|
|
return;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
struct treeview_update_url_walk_ctx {
|
|
nsurl *url;
|
|
const struct url_data *data;
|
|
};
|
|
/** Callback for treeview_walk */
|
|
static nserror hotlist_update_url_walk_cb(void *ctx, void *node_data,
|
|
enum treeview_node_type type, bool *abort)
|
|
{
|
|
struct treeview_update_url_walk_ctx *tw = ctx;
|
|
struct hotlist_entry *e = node_data;
|
|
nserror err;
|
|
|
|
if (type != TREE_NODE_ENTRY) {
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
if (nsurl_compare(e->url, tw->url, NSURL_COMPLETE) == true) {
|
|
/* Found match: Update the entry data */
|
|
free((void *)e->data[HL_LAST_VISIT].value); /* Eww */
|
|
free((void *)e->data[HL_VISITS].value); /* Eww */
|
|
|
|
if (tw->data == NULL) {
|
|
/* Get the URL data */
|
|
tw->data = urldb_get_url_data(tw->url);
|
|
if (tw->data == NULL) {
|
|
/* No entry in database, so add one */
|
|
urldb_add_url(tw->url);
|
|
/* now attempt to get url data */
|
|
tw->data = urldb_get_url_data(tw->url);
|
|
}
|
|
if (tw->data == NULL) {
|
|
return NSERROR_NOMEM;
|
|
}
|
|
}
|
|
|
|
err = hotlist_create_treeview_field_visits_data(e, tw->data);
|
|
if (err != NSERROR_OK)
|
|
return err;
|
|
|
|
err = treeview_update_node_entry(hl_ctx.tree,
|
|
e->entry, e->data, e);
|
|
if (err != NSERROR_OK)
|
|
return err;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
/* Exported interface, documented in hotlist.h */
|
|
void hotlist_update_url(nsurl *url)
|
|
{
|
|
nserror err;
|
|
struct treeview_update_url_walk_ctx tw = {
|
|
.url = url,
|
|
.data = NULL
|
|
};
|
|
|
|
if (hl_ctx.built == false)
|
|
return;
|
|
|
|
err = treeview_walk(hl_ctx.tree, NULL, hotlist_update_url_walk_cb, NULL,
|
|
&tw, TREE_NODE_ENTRY);
|
|
if (err != NSERROR_OK)
|
|
return;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_add_entry(nsurl *url, const char *title, bool at_y, int y)
|
|
{
|
|
nserror err;
|
|
treeview_node *entry;
|
|
treeview_node *relation;
|
|
enum treeview_relationship rel;
|
|
|
|
if (url == NULL) {
|
|
err = nsurl_create("http://netsurf-browser.org/", &url);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
assert(url != NULL);
|
|
} else {
|
|
nsurl_ref(url);
|
|
}
|
|
|
|
err = treeview_get_relation(hl_ctx.tree, &relation, &rel, at_y, y);
|
|
if (err != NSERROR_OK) {
|
|
nsurl_unref(url);
|
|
return err;
|
|
}
|
|
|
|
err = hotlist_add_entry_internal(url, title, NULL,
|
|
relation, rel, &entry);
|
|
if (err != NSERROR_OK) {
|
|
nsurl_unref(url);
|
|
return err;
|
|
}
|
|
|
|
nsurl_unref(url);
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_add_folder(const char *title, bool at_y, int y)
|
|
{
|
|
nserror err;
|
|
struct hotlist_folder *f;
|
|
treeview_node *relation;
|
|
enum treeview_relationship rel;
|
|
|
|
err = treeview_get_relation(hl_ctx.tree, &relation, &rel, at_y, y);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
err = hotlist_add_folder_internal(title, relation, rel, &f, false);
|
|
if (err != NSERROR_OK) {
|
|
return err;
|
|
}
|
|
|
|
return NSERROR_OK;
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
void hotlist_redraw(int x, int y, struct rect *clip,
|
|
const struct redraw_context *ctx)
|
|
{
|
|
treeview_redraw(hl_ctx.tree, x, y, clip, ctx);
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
void hotlist_mouse_action(browser_mouse_state mouse, int x, int y)
|
|
{
|
|
treeview_mouse_action(hl_ctx.tree, mouse, x, y);
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
bool hotlist_keypress(uint32_t key)
|
|
{
|
|
return treeview_keypress(hl_ctx.tree, key);
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
bool hotlist_has_selection(void)
|
|
{
|
|
return treeview_has_selection(hl_ctx.tree);
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
bool hotlist_get_selection(nsurl **url, const char **title)
|
|
{
|
|
struct hotlist_entry *e;
|
|
enum treeview_node_type type;
|
|
void *v;
|
|
|
|
type = treeview_get_selection(hl_ctx.tree, &v);
|
|
if (type != TREE_NODE_ENTRY || v == NULL) {
|
|
*url = NULL;
|
|
*title = NULL;
|
|
return false;
|
|
}
|
|
|
|
e = (struct hotlist_entry *)v;
|
|
|
|
*url = e->url;
|
|
*title = e->data[HL_TITLE].value;
|
|
return true;
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
void hotlist_edit_selection(void)
|
|
{
|
|
treeview_edit_selection(hl_ctx.tree);
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_expand(bool only_folders)
|
|
{
|
|
return treeview_expand(hl_ctx.tree, only_folders);
|
|
}
|
|
|
|
|
|
/* Exported interface, documented in hotlist.h */
|
|
nserror hotlist_contract(bool all)
|
|
{
|
|
return treeview_contract(hl_ctx.tree, all);
|
|
}
|
|
|