2004-02-13 19:09:12 +03:00
|
|
|
/*
|
|
|
|
* Copyright 2003 Phil Mellor <monkeyson@users.sourceforge.net>
|
|
|
|
* Copyright 2003 John M Bell <jmb202@ecs.soton.ac.uk>
|
|
|
|
* Copyright 2004 James Bursa <bursa@users.sourceforge.net>
|
2005-12-01 05:59:55 +03:00
|
|
|
* Copyright 2005 Richard Wilson <info@tinct.net>
|
2007-08-08 20:16:03 +04:00
|
|
|
*
|
|
|
|
* 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/>.
|
2004-02-13 19:09:12 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
/** \file
|
|
|
|
* Option reading and saving (implementation).
|
|
|
|
*
|
|
|
|
* Options are stored in the format key:value, one per line. For bool options,
|
|
|
|
* value is "0" or "1".
|
|
|
|
*/
|
|
|
|
|
2004-12-09 13:30:44 +03:00
|
|
|
#include <assert.h>
|
2004-02-13 19:09:12 +03:00
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2008-08-11 20:44:12 +04:00
|
|
|
#include <strings.h>
|
2007-05-31 02:39:54 +04:00
|
|
|
#include <libxml/HTMLparser.h>
|
|
|
|
#include <libxml/HTMLtree.h>
|
|
|
|
#include "content/urldb.h"
|
|
|
|
#include "css/css.h"
|
|
|
|
#include "desktop/options.h"
|
|
|
|
#include "desktop/tree.h"
|
|
|
|
#include "utils/log.h"
|
|
|
|
#include "utils/messages.h"
|
2009-02-06 05:04:25 +03:00
|
|
|
#include "utils/url.h"
|
2007-05-31 02:39:54 +04:00
|
|
|
#include "utils/utils.h"
|
2004-02-13 19:09:12 +03:00
|
|
|
|
2006-07-24 02:32:33 +04:00
|
|
|
#if defined(riscos)
|
2007-05-31 02:39:54 +04:00
|
|
|
#include "riscos/options.h"
|
2006-07-24 02:32:33 +04:00
|
|
|
#elif defined(nsgtk)
|
2007-05-31 02:39:54 +04:00
|
|
|
#include "gtk/options.h"
|
2008-07-29 13:28:45 +04:00
|
|
|
#elif defined(nsbeos)
|
|
|
|
#include "beos/options.h"
|
2008-08-25 17:27:44 +04:00
|
|
|
#elif defined(nsamiga)
|
|
|
|
#include "amiga/options.h"
|
2009-06-28 22:32:47 +04:00
|
|
|
#elif defined(nsframebuffer)
|
|
|
|
#include "framebuffer/options.h"
|
2004-02-13 19:09:12 +03:00
|
|
|
#else
|
|
|
|
#define EXTRA_OPTION_DEFINE
|
|
|
|
#define EXTRA_OPTION_TABLE
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/** An HTTP proxy should be used. */
|
|
|
|
bool option_http_proxy = false;
|
|
|
|
/** Hostname of proxy. */
|
|
|
|
char *option_http_proxy_host = 0;
|
|
|
|
/** Proxy port. */
|
|
|
|
int option_http_proxy_port = 8080;
|
2004-07-27 19:49:28 +04:00
|
|
|
/** Proxy authentication method. */
|
|
|
|
int option_http_proxy_auth = OPTION_HTTP_PROXY_AUTH_NONE;
|
|
|
|
/** Proxy authentication user name */
|
|
|
|
char *option_http_proxy_auth_user = 0;
|
|
|
|
/** Proxy authentication password */
|
|
|
|
char *option_http_proxy_auth_pass = 0;
|
2004-02-14 02:07:42 +03:00
|
|
|
/** Default font size / 0.1pt. */
|
2007-12-09 14:13:25 +03:00
|
|
|
int option_font_size = 128;
|
2004-02-14 02:07:42 +03:00
|
|
|
/** Minimum font size. */
|
2007-12-09 14:13:25 +03:00
|
|
|
int option_font_min_size = 85;
|
2006-03-31 14:41:21 +04:00
|
|
|
/** Default sans serif font */
|
|
|
|
char *option_font_sans;
|
|
|
|
/** Default serif font */
|
|
|
|
char *option_font_serif;
|
|
|
|
/** Default monospace font */
|
|
|
|
char *option_font_mono;
|
|
|
|
/** Default cursive font */
|
|
|
|
char *option_font_cursive;
|
|
|
|
/** Default fantasy font */
|
|
|
|
char *option_font_fantasy;
|
2004-03-08 03:03:58 +03:00
|
|
|
/** Accept-Language header. */
|
|
|
|
char *option_accept_language = 0;
|
2006-12-28 03:42:14 +03:00
|
|
|
/** Accept-Charset header. */
|
|
|
|
char *option_accept_charset = 0;
|
2004-06-21 19:09:59 +04:00
|
|
|
/** Preferred maximum size of memory cache / bytes. */
|
|
|
|
int option_memory_cache_size = 2 * 1024 * 1024;
|
2006-01-07 03:39:14 +03:00
|
|
|
/** Preferred expiry age of disc cache / days. */
|
|
|
|
int option_disc_cache_age = 28;
|
2004-07-31 03:40:01 +04:00
|
|
|
/** Whether to block advertisements */
|
|
|
|
bool option_block_ads = false;
|
2004-10-05 03:54:42 +04:00
|
|
|
/** Minimum GIF animation delay */
|
|
|
|
int option_minimum_gif_delay = 10;
|
2004-10-02 01:31:55 +04:00
|
|
|
/** Whether to send the referer HTTP header */
|
|
|
|
bool option_send_referer = true;
|
2004-12-09 13:30:44 +03:00
|
|
|
/** Whether to animate images */
|
|
|
|
bool option_animate_images = true;
|
2005-12-30 02:18:17 +03:00
|
|
|
/** How many days to retain URL data for */
|
|
|
|
int option_expire_url = 28;
|
2006-01-03 02:31:29 +03:00
|
|
|
/** Default font family */
|
|
|
|
int option_font_default = CSS_FONT_FAMILY_SANS_SERIF;
|
2006-01-08 04:51:33 +03:00
|
|
|
/** ca-bundle location */
|
|
|
|
char *option_ca_bundle = 0;
|
2007-06-27 19:39:56 +04:00
|
|
|
/** ca-path location */
|
|
|
|
char *option_ca_path = 0;
|
2006-01-08 04:51:33 +03:00
|
|
|
/** Cookie file location */
|
|
|
|
char *option_cookie_file = 0;
|
|
|
|
/** Cookie jar loaction */
|
|
|
|
char *option_cookie_jar = 0;
|
2006-03-29 19:26:54 +04:00
|
|
|
/** Home page location */
|
|
|
|
char *option_homepage_url = 0;
|
2006-07-28 19:52:37 +04:00
|
|
|
/** URL completion in url bar */
|
|
|
|
bool option_url_suggestion = true;
|
2006-08-29 04:17:35 +04:00
|
|
|
/** default x position of new windows */
|
|
|
|
int option_window_x = 0;
|
|
|
|
/** default y position of new windows */
|
|
|
|
int option_window_y = 0;
|
|
|
|
/** default width of new windows */
|
|
|
|
int option_window_width = 0;
|
|
|
|
/** default height of new windows */
|
|
|
|
int option_window_height = 0;
|
|
|
|
/** width of screen when above options were saved */
|
|
|
|
int option_window_screen_width = 0;
|
|
|
|
/** height of screen when above options were saved */
|
|
|
|
int option_window_screen_height = 0;
|
|
|
|
/** default size of status bar vs. h scroll bar */
|
|
|
|
#ifdef nsgtk
|
2006-08-29 04:42:30 +04:00
|
|
|
int option_toolbar_status_width = 400;
|
2006-08-29 04:17:35 +04:00
|
|
|
#else
|
|
|
|
int option_toolbar_status_width = 6667;
|
|
|
|
#endif
|
2007-08-07 07:55:18 +04:00
|
|
|
/** default window scale */
|
|
|
|
int option_scale = 100;
|
2008-03-01 02:03:26 +03:00
|
|
|
/* Whether to reflow web pages while objects are fetching */
|
|
|
|
bool option_incremental_reflow = true;
|
|
|
|
/* Minimum time between HTML reflows while objects are fetching */
|
|
|
|
#ifdef riscos
|
2008-03-11 03:14:39 +03:00
|
|
|
unsigned int option_min_reflow_period = 100; /* time in cs */
|
2008-03-01 02:03:26 +03:00
|
|
|
#else
|
2008-03-11 03:14:39 +03:00
|
|
|
unsigned int option_min_reflow_period = 25; /* time in cs */
|
2008-03-01 02:03:26 +03:00
|
|
|
#endif
|
Second merge of Adam Blokus' GSoC work from his branch 'branches/adamblokus/netsurf'.
Merged revisions 4195-4211,4216,4219-4220,4222-4234,4236-4250,4252-4262,4264-4266,4268-4326,4329-4335,4338-4342,4344-4411,4413-4420,4422-4436,4438-4491,4494-4506,4508-4514,4516,4518-4552,4554,4556-4564,4567-4568,4570-4574,4576-4686,4689-4692,4694,4698-4709,4715-4723,4725-4755,4757-4769,4771-4919,4921-4996,4998-5110,5112-5117 via svnmerge from
svn://svn.netsurf-browser.org/branches/adamblokus/netsurf
........
r4736 | adamblokus | 2008-07-26 13:46:54 +0200 (Sat, 26 Jul 2008) | 2 lines
Sorting out some problems with svn.
........
r4737 | adamblokus | 2008-07-26 13:54:36 +0200 (Sat, 26 Jul 2008) | 4 lines
Added export tab to the options dialog.
Added the possibility of changing some print options.
........
r4897 | adamblokus | 2008-08-04 17:59:05 +0200 (Mon, 04 Aug 2008) | 5 lines
Added checking of horizontal clipping.
Added better table loosening.
Changed some minor bugs.
Applied changes in the Export options tab according to the review from tlsa.
........
r4905 | adamblokus | 2008-08-05 01:53:34 +0200 (Tue, 05 Aug 2008) | 2 lines
Fixed bug which made it impossible to export pdf's.
........
r4919 | adamblokus | 2008-08-05 16:39:33 +0200 (Tue, 05 Aug 2008) | 2 lines
Fixed some memory leaks which caused Netsurf to break.
........
r4927 | adamblokus | 2008-08-06 02:26:30 +0200 (Wed, 06 Aug 2008) | 4 lines
Fixed bug with filenames which crashed Netsurf.
Turned anti aliasing off for printing.
Fixed some scaling issues.
........
r4928 | adamblokus | 2008-08-06 17:52:44 +0200 (Wed, 06 Aug 2008) | 5 lines
Added new export/print options:
- suppressing images
- turning off backgrounds
- toggled loosening
........
r4950 | adamblokus | 2008-08-07 21:15:21 +0200 (Thu, 07 Aug 2008) | 5 lines
Added new options to PDF export:
- document compression
- document encryption
Added PDF password dialog
........
r4954 | adamblokus | 2008-08-07 22:11:31 +0200 (Thu, 07 Aug 2008) | 2 lines
Added saving print settings.
........
r4956 | adamblokus | 2008-08-07 22:44:48 +0200 (Thu, 07 Aug 2008) | 2 lines
Fixes to PDF encryption
........
r4970 | adamblokus | 2008-08-09 15:26:24 +0200 (Sat, 09 Aug 2008) | 3 lines
Fixed bug in plotting tiled bitmaps.
Fixed bug with too long text decorations.
........
r4977 | adamblokus | 2008-08-09 19:18:56 +0200 (Sat, 09 Aug 2008) | 2 lines
Fixed JPG embedding bug.
........
r4988 | adamblokus | 2008-08-10 16:59:51 +0200 (Sun, 10 Aug 2008) | 3 lines
Added clip checking to pdf plotters. No more "blank" clips.
Made PDF compression a default setting.
........
r4995 | adamblokus | 2008-08-10 20:03:00 +0200 (Sun, 10 Aug 2008) | 2 lines
Fixed Haru crash on font-size==0.
........
r4996 | adamblokus | 2008-08-10 21:04:43 +0200 (Sun, 10 Aug 2008) | 2 lines
Added changing text mode only if necessary.
........
r5045 | adamblokus | 2008-08-11 21:26:26 +0200 (Mon, 11 Aug 2008) | 3 lines
Removing gtk stuff from core code.
Little fix in options.
........
r5048 | adamblokus | 2008-08-11 21:57:45 +0200 (Mon, 11 Aug 2008) | 2 lines
Better font size checking in PDF export.
........
r5050 | adamblokus | 2008-08-11 22:19:56 +0200 (Mon, 11 Aug 2008) | 2 lines
Fixed riscos text scale bug.
........
r5073 | adamblokus | 2008-08-12 17:40:57 +0200 (Tue, 12 Aug 2008) | 2 lines
Added missing tooltips
........
r5092 | adamblokus | 2008-08-13 17:09:25 +0200 (Wed, 13 Aug 2008) | 2 lines
Moved /pdf folder to desktop/save_pdf
........
r5110 | adamblokus | 2008-08-13 22:44:50 +0200 (Wed, 13 Aug 2008) | 2 lines
Added comments.
........
r5113 | adamblokus | 2008-08-13 23:07:35 +0200 (Wed, 13 Aug 2008) | 2 lines
Cosmetic changes
........
r5116 | adamblokus | 2008-08-14 16:10:18 +0200 (Thu, 14 Aug 2008) | 2 lines
Fixed bug with BOX_INLINE_END in tree duplication.
........
r5117 | joty | 2008-08-14 21:47:46 +0200 (Thu, 14 Aug 2008) | 1 line
Improvement for r5116: use local vars when possible; rename global last to box_duplicate_last; check on box_duplicate_main_tree failure.
........
svn path=/trunk/netsurf/; revision=5118
2008-08-15 00:32:10 +04:00
|
|
|
/** top margin of exported page*/
|
|
|
|
int option_margin_top = DEFAULT_MARGIN_TOP_MM;
|
|
|
|
/** bottom margin of exported page*/
|
|
|
|
int option_margin_bottom = DEFAULT_MARGIN_BOTTOM_MM;
|
|
|
|
/** left margin of exported page*/
|
|
|
|
int option_margin_left = DEFAULT_MARGIN_LEFT_MM;
|
|
|
|
/** right margin of exported page*/
|
|
|
|
int option_margin_right = DEFAULT_MARGIN_RIGHT_MM;
|
|
|
|
/** scale of exported content*/
|
|
|
|
int option_export_scale = DEFAULT_EXPORT_SCALE * 100;
|
|
|
|
/**suppressing images in printed content*/
|
|
|
|
bool option_suppress_images = false;
|
|
|
|
/**turning off all backgrounds for printed content*/
|
|
|
|
bool option_remove_backgrounds = false;
|
|
|
|
/**turning on content loosening for printed content*/
|
|
|
|
bool option_enable_loosening = true;
|
|
|
|
/**compression of PDF documents*/
|
|
|
|
bool option_enable_PDF_compression = true;
|
|
|
|
/**setting a password and encoding PDF documents*/
|
|
|
|
bool option_enable_PDF_password = false;
|
2006-08-29 04:17:35 +04:00
|
|
|
|
2006-03-14 17:21:01 +03:00
|
|
|
/* Fetcher configuration */
|
|
|
|
/** Maximum simultaneous active fetchers */
|
|
|
|
int option_max_fetchers = 24;
|
|
|
|
/** Maximum simultaneous active fetchers per host.
|
2008-07-18 13:52:18 +04:00
|
|
|
* (<=option_max_fetchers else it makes no sense)
|
|
|
|
* Note that rfc2616 section 8.1.4 says that there should be no more than
|
|
|
|
* two keepalive connections per host. None of the main browsers follow this
|
|
|
|
* as it slows page fetches down considerably.
|
|
|
|
* See https://bugzilla.mozilla.org/show_bug.cgi?id=423377#c4
|
2006-03-14 17:21:01 +03:00
|
|
|
*/
|
2008-07-18 13:52:18 +04:00
|
|
|
int option_max_fetchers_per_host = 5;
|
2006-03-14 17:21:01 +03:00
|
|
|
/** Maximum number of inactive fetchers cached.
|
|
|
|
* The total number of handles netsurf will therefore have open
|
|
|
|
* is this plus option_max_fetchers.
|
|
|
|
*/
|
|
|
|
int option_max_cached_fetch_handles = 6;
|
2007-06-27 20:27:52 +04:00
|
|
|
/** Suppress debug output from cURL. */
|
|
|
|
bool option_suppress_curl_debug = true;
|
|
|
|
|
2006-09-02 19:52:41 +04:00
|
|
|
/** Whether to allow target="_blank" */
|
|
|
|
bool option_target_blank = true;
|
2006-03-14 17:21:01 +03:00
|
|
|
|
2008-11-29 03:24:09 +03:00
|
|
|
/** Whether second mouse button opens in new tab */
|
|
|
|
bool option_button_2_tab = true;
|
|
|
|
|
2004-02-13 19:09:12 +03:00
|
|
|
EXTRA_OPTION_DEFINE
|
|
|
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
const char *key;
|
|
|
|
enum { OPTION_BOOL, OPTION_INTEGER, OPTION_STRING } type;
|
|
|
|
void *p;
|
|
|
|
} option_table[] = {
|
2008-11-29 03:24:09 +03:00
|
|
|
{ "http_proxy", OPTION_BOOL, &option_http_proxy },
|
|
|
|
{ "http_proxy_host", OPTION_STRING, &option_http_proxy_host },
|
|
|
|
{ "http_proxy_port", OPTION_INTEGER, &option_http_proxy_port },
|
|
|
|
{ "http_proxy_auth", OPTION_INTEGER, &option_http_proxy_auth },
|
|
|
|
{ "http_proxy_auth_user",
|
|
|
|
OPTION_STRING, &option_http_proxy_auth_user },
|
|
|
|
{ "http_proxy_auth_pass",
|
|
|
|
OPTION_STRING, &option_http_proxy_auth_pass },
|
|
|
|
{ "font_size", OPTION_INTEGER, &option_font_size },
|
|
|
|
{ "font_min_size", OPTION_INTEGER, &option_font_min_size },
|
|
|
|
{ "font_sans", OPTION_STRING, &option_font_sans },
|
|
|
|
{ "font_serif", OPTION_STRING, &option_font_serif },
|
|
|
|
{ "font_mono", OPTION_STRING, &option_font_mono },
|
|
|
|
{ "font_cursive", OPTION_STRING, &option_font_cursive },
|
|
|
|
{ "font_fantasy", OPTION_STRING, &option_font_fantasy },
|
|
|
|
{ "accept_language", OPTION_STRING, &option_accept_language },
|
|
|
|
{ "accept_charset", OPTION_STRING, &option_accept_charset },
|
|
|
|
{ "memory_cache_size", OPTION_INTEGER, &option_memory_cache_size },
|
|
|
|
{ "disc_cache_age", OPTION_INTEGER, &option_disc_cache_age },
|
|
|
|
{ "block_advertisements",
|
|
|
|
OPTION_BOOL, &option_block_ads },
|
|
|
|
{ "minimum_gif_delay", OPTION_INTEGER, &option_minimum_gif_delay },
|
|
|
|
{ "send_referer", OPTION_BOOL, &option_send_referer },
|
|
|
|
{ "animate_images", OPTION_BOOL, &option_animate_images },
|
|
|
|
{ "expire_url", OPTION_INTEGER, &option_expire_url },
|
|
|
|
{ "font_default", OPTION_INTEGER, &option_font_default },
|
|
|
|
{ "ca_bundle", OPTION_STRING, &option_ca_bundle },
|
|
|
|
{ "ca_path", OPTION_STRING, &option_ca_path },
|
|
|
|
{ "cookie_file", OPTION_STRING, &option_cookie_file },
|
|
|
|
{ "cookie_jar", OPTION_STRING, &option_cookie_jar },
|
|
|
|
{ "homepage_url", OPTION_STRING, &option_homepage_url },
|
|
|
|
{ "url_suggestion", OPTION_BOOL, &option_url_suggestion },
|
|
|
|
{ "window_x", OPTION_INTEGER, &option_window_x },
|
|
|
|
{ "window_y", OPTION_INTEGER, &option_window_y },
|
|
|
|
{ "window_width", OPTION_INTEGER, &option_window_width },
|
|
|
|
{ "window_height", OPTION_INTEGER, &option_window_height },
|
|
|
|
{ "window_screen_width",
|
|
|
|
OPTION_INTEGER, &option_window_screen_width },
|
|
|
|
{ "window_screen_height",
|
|
|
|
OPTION_INTEGER, &option_window_screen_height },
|
|
|
|
{ "toolbar_status_size",
|
|
|
|
OPTION_INTEGER, &option_toolbar_status_width },
|
|
|
|
{ "scale", OPTION_INTEGER, &option_scale },
|
|
|
|
{ "incremental_reflow", OPTION_BOOL, &option_incremental_reflow },
|
|
|
|
{ "min_reflow_period", OPTION_INTEGER, &option_min_reflow_period },
|
2006-03-14 17:21:01 +03:00
|
|
|
/* Fetcher options */
|
2008-11-29 03:24:09 +03:00
|
|
|
{ "max_fetchers", OPTION_INTEGER, &option_max_fetchers },
|
2006-03-14 17:21:01 +03:00
|
|
|
{ "max_fetchers_per_host",
|
2008-11-29 03:24:09 +03:00
|
|
|
OPTION_INTEGER, &option_max_fetchers_per_host },
|
2006-03-14 17:21:01 +03:00
|
|
|
{ "max_cached_fetch_handles",
|
2008-11-29 03:24:09 +03:00
|
|
|
OPTION_INTEGER, &option_max_cached_fetch_handles },
|
|
|
|
{ "suppress_curl_debug",OPTION_BOOL, &option_suppress_curl_debug },
|
|
|
|
{ "target_blank", OPTION_BOOL, &option_target_blank },
|
|
|
|
{ "button_2_tab", OPTION_BOOL, &option_button_2_tab },
|
|
|
|
/* PDF / Print options*/
|
|
|
|
{ "margin_top", OPTION_INTEGER, &option_margin_top},
|
|
|
|
{ "margin_bottom", OPTION_INTEGER, &option_margin_bottom},
|
|
|
|
{ "margin_left", OPTION_INTEGER, &option_margin_left},
|
|
|
|
{ "margin_right", OPTION_INTEGER, &option_margin_right},
|
|
|
|
{ "export_scale", OPTION_INTEGER, &option_export_scale},
|
|
|
|
{ "suppress_images", OPTION_BOOL, &option_suppress_images},
|
|
|
|
{ "remove_backgrounds", OPTION_BOOL, &option_remove_backgrounds},
|
|
|
|
{ "enable_loosening", OPTION_BOOL, &option_enable_loosening},
|
|
|
|
{ "enable_PDF_compression",
|
|
|
|
OPTION_BOOL, &option_enable_PDF_compression},
|
|
|
|
{ "enable_PDF_password",
|
|
|
|
OPTION_BOOL, &option_enable_PDF_password},
|
2004-02-13 19:09:12 +03:00
|
|
|
EXTRA_OPTION_TABLE
|
|
|
|
};
|
|
|
|
|
|
|
|
#define option_table_entries (sizeof option_table / sizeof option_table[0])
|
|
|
|
|
|
|
|
|
2005-02-07 17:28:43 +03:00
|
|
|
static void options_load_tree_directory(xmlNode *ul, struct node *directory);
|
|
|
|
static void options_load_tree_entry(xmlNode *li, struct node *directory);
|
|
|
|
xmlNode *options_find_tree_element(xmlNode *node, const char *name);
|
|
|
|
bool options_save_tree_directory(struct node *directory, xmlNode *node);
|
|
|
|
bool options_save_tree_entry(struct node *entry, xmlNode *node);
|
2004-12-09 13:30:44 +03:00
|
|
|
|
|
|
|
|
2004-02-13 19:09:12 +03:00
|
|
|
/**
|
|
|
|
* Read options from a file.
|
|
|
|
*
|
|
|
|
* \param path name of file to read options from
|
|
|
|
*
|
|
|
|
* Option variables corresponding to lines in the file are updated. Missing
|
|
|
|
* options are unchanged. If the file fails to open, options are unchanged.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void options_read(const char *path)
|
|
|
|
{
|
|
|
|
char s[100];
|
|
|
|
FILE *fp;
|
|
|
|
|
|
|
|
fp = fopen(path, "r");
|
|
|
|
if (!fp) {
|
|
|
|
LOG(("failed to open file '%s'", path));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (fgets(s, 100, fp)) {
|
|
|
|
char *colon, *value;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
if (s[0] == 0 || s[0] == '#')
|
|
|
|
continue;
|
|
|
|
colon = strchr(s, ':');
|
|
|
|
if (colon == 0)
|
|
|
|
continue;
|
|
|
|
s[strlen(s) - 1] = 0; /* remove \n at end */
|
|
|
|
*colon = 0; /* terminate key */
|
|
|
|
value = colon + 1;
|
|
|
|
|
|
|
|
for (i = 0; i != option_table_entries; i++) {
|
|
|
|
if (strcasecmp(s, option_table[i].key) != 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
switch (option_table[i].type) {
|
|
|
|
case OPTION_BOOL:
|
|
|
|
*((bool *) option_table[i].p) =
|
|
|
|
value[0] == '1';
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_INTEGER:
|
|
|
|
*((int *) option_table[i].p) =
|
|
|
|
atoi(value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_STRING:
|
|
|
|
free(*((char **) option_table[i].p));
|
|
|
|
*((char **) option_table[i].p) =
|
|
|
|
strdup(value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(fp);
|
2004-02-14 02:07:42 +03:00
|
|
|
|
|
|
|
if (option_font_size < 50)
|
|
|
|
option_font_size = 50;
|
|
|
|
if (1000 < option_font_size)
|
|
|
|
option_font_size = 1000;
|
|
|
|
if (option_font_min_size < 10)
|
|
|
|
option_font_min_size = 10;
|
|
|
|
if (500 < option_font_min_size)
|
|
|
|
option_font_min_size = 500;
|
2004-06-21 19:09:59 +04:00
|
|
|
|
|
|
|
if (option_memory_cache_size < 0)
|
|
|
|
option_memory_cache_size = 0;
|
2004-02-13 19:09:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save options to a file.
|
|
|
|
*
|
|
|
|
* \param path name of file to write options to
|
|
|
|
*
|
|
|
|
* Errors are ignored.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void options_write(const char *path)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
FILE *fp;
|
|
|
|
|
|
|
|
fp = fopen(path, "w");
|
|
|
|
if (!fp) {
|
|
|
|
LOG(("failed to open file '%s' for writing", path));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i != option_table_entries; i++) {
|
|
|
|
fprintf(fp, "%s:", option_table[i].key);
|
|
|
|
switch (option_table[i].type) {
|
|
|
|
case OPTION_BOOL:
|
|
|
|
fprintf(fp, "%c", *((bool *) option_table[i].p) ?
|
|
|
|
'1' : '0');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_INTEGER:
|
|
|
|
fprintf(fp, "%i", *((int *) option_table[i].p));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_STRING:
|
|
|
|
if (*((char **) option_table[i].p))
|
|
|
|
fprintf(fp, "%s", *((char **) option_table[i].p));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
fprintf(fp, "\n");
|
2009-04-21 00:31:13 +04:00
|
|
|
}
|
2004-02-13 19:09:12 +03:00
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
}
|
2004-12-09 13:30:44 +03:00
|
|
|
|
2006-01-01 21:52:30 +03:00
|
|
|
/**
|
|
|
|
* Dump user options to stderr
|
|
|
|
*/
|
|
|
|
void options_dump(void)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i != option_table_entries; i++) {
|
|
|
|
fprintf(stderr, "%s:", option_table[i].key);
|
|
|
|
switch (option_table[i].type) {
|
|
|
|
case OPTION_BOOL:
|
|
|
|
fprintf(stderr, "%c",
|
|
|
|
*((bool *) option_table[i].p) ?
|
|
|
|
'1' : '0');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_INTEGER:
|
|
|
|
fprintf(stderr, "%i",
|
|
|
|
*((int *) option_table[i].p));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_STRING:
|
|
|
|
if (*((char **) option_table[i].p))
|
|
|
|
fprintf(stderr, "%s",
|
|
|
|
*((char **) option_table[i].p));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
}
|
2004-12-09 13:30:44 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads a hotlist as a tree from a specified file.
|
|
|
|
*
|
|
|
|
* \param filename name of file to read
|
|
|
|
* \return the hotlist file represented as a tree, or NULL on failure
|
|
|
|
*/
|
2005-02-07 17:28:43 +03:00
|
|
|
struct tree *options_load_tree(const char *filename) {
|
2004-12-09 13:30:44 +03:00
|
|
|
xmlDoc *doc;
|
|
|
|
xmlNode *html, *body, *ul;
|
|
|
|
struct tree *tree;
|
|
|
|
|
|
|
|
doc = htmlParseFile(filename, "iso-8859-1");
|
|
|
|
if (!doc) {
|
|
|
|
warn_user("HotlistLoadError", messages_get("ParsingFail"));
|
|
|
|
return NULL;
|
|
|
|
}
|
2006-01-01 21:52:30 +03:00
|
|
|
|
2005-02-07 17:28:43 +03:00
|
|
|
html = options_find_tree_element((xmlNode *) doc, "html");
|
|
|
|
body = options_find_tree_element(html, "body");
|
|
|
|
ul = options_find_tree_element(body, "ul");
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!ul) {
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
warn_user("HotlistLoadError",
|
|
|
|
"(<html>...<body>...<ul> not found.)");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
tree = calloc(sizeof(struct tree), 1);
|
|
|
|
if (!tree) {
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
tree->root = tree_create_folder_node(NULL, "Root");
|
2009-02-06 05:04:25 +03:00
|
|
|
if (!tree->root) {
|
|
|
|
free(tree);
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
2004-12-09 13:30:44 +03:00
|
|
|
|
2005-02-07 17:28:43 +03:00
|
|
|
options_load_tree_directory(ul, tree->root);
|
2004-12-09 13:30:44 +03:00
|
|
|
tree->root->expanded = true;
|
|
|
|
tree_initialise(tree);
|
|
|
|
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return tree;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse a directory represented as a ul.
|
|
|
|
*
|
|
|
|
* \param ul xmlNode for parsed ul
|
|
|
|
* \param directory directory to add this directory to
|
|
|
|
*/
|
2005-02-07 17:28:43 +03:00
|
|
|
void options_load_tree_directory(xmlNode *ul, struct node *directory) {
|
2004-12-09 13:30:44 +03:00
|
|
|
char *title;
|
|
|
|
struct node *dir;
|
|
|
|
xmlNode *n;
|
2006-01-01 21:52:30 +03:00
|
|
|
|
2004-12-09 13:30:44 +03:00
|
|
|
assert(ul);
|
|
|
|
assert(directory);
|
|
|
|
|
|
|
|
for (n = ul->children; n; n = n->next) {
|
|
|
|
/* 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. */
|
|
|
|
|
|
|
|
if (n->type != XML_ELEMENT_NODE)
|
|
|
|
continue;
|
|
|
|
|
2008-02-25 20:58:00 +03:00
|
|
|
if (strcmp((const char *) n->name, "li") == 0) {
|
2004-12-09 13:30:44 +03:00
|
|
|
/* entry */
|
2005-02-07 17:28:43 +03:00
|
|
|
options_load_tree_entry(n, directory);
|
2004-12-09 13:30:44 +03:00
|
|
|
|
2008-02-25 20:58:00 +03:00
|
|
|
} else if (strcmp((const char *) n->name, "h4") == 0) {
|
2004-12-09 13:30:44 +03:00
|
|
|
/* directory */
|
|
|
|
title = (char *) xmlNodeGetContent(n);
|
|
|
|
if (!title) {
|
|
|
|
warn_user("HotlistLoadError", "(Empty <h4> "
|
|
|
|
"or memory exhausted.)");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = n->next;
|
|
|
|
n && n->type != XML_ELEMENT_NODE;
|
|
|
|
n = n->next)
|
|
|
|
;
|
2008-02-25 20:58:00 +03:00
|
|
|
if (!n || strcmp((const char *) n->name, "ul") != 0) {
|
2004-12-09 13:30:44 +03:00
|
|
|
/* next element isn't expected ul */
|
|
|
|
free(title);
|
|
|
|
warn_user("HotlistLoadError", "(Expected "
|
|
|
|
"<ul> not present.)");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dir = tree_create_folder_node(directory, title);
|
2009-02-06 05:04:25 +03:00
|
|
|
if (!dir) {
|
|
|
|
free(title);
|
|
|
|
|
2004-12-09 13:30:44 +03:00
|
|
|
return;
|
2009-02-06 05:04:25 +03:00
|
|
|
}
|
2005-02-07 17:28:43 +03:00
|
|
|
options_load_tree_directory(n, dir);
|
2004-12-09 13:30:44 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse an entry represented as a li.
|
|
|
|
*
|
|
|
|
* \param li xmlNode for parsed li
|
|
|
|
* \param directory directory to add this entry to
|
|
|
|
*/
|
2005-02-07 17:28:43 +03:00
|
|
|
void options_load_tree_entry(xmlNode *li, struct node *directory) {
|
2009-02-06 05:04:25 +03:00
|
|
|
char *url = NULL, *url1 = NULL;
|
|
|
|
char *title = NULL;
|
2004-12-09 13:30:44 +03:00
|
|
|
struct node *entry;
|
|
|
|
xmlNode *n;
|
2006-04-10 03:21:13 +04:00
|
|
|
const struct url_data *data;
|
2009-02-06 05:04:25 +03:00
|
|
|
url_func_result res;
|
2004-12-09 13:30:44 +03:00
|
|
|
|
|
|
|
for (n = li->children; n; n = n->next) {
|
2005-12-31 07:31:37 +03:00
|
|
|
/* The li must contain an "a" element */
|
2004-12-09 13:30:44 +03:00
|
|
|
if (n->type == XML_ELEMENT_NODE &&
|
2008-02-25 20:58:00 +03:00
|
|
|
strcmp((const char *) n->name, "a") == 0) {
|
2009-02-06 05:04:25 +03:00
|
|
|
url1 = (char *) xmlGetProp(n, (const xmlChar *) "href");
|
2004-12-09 13:30:44 +03:00
|
|
|
title = (char *) xmlNodeGetContent(n);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-06 05:04:25 +03:00
|
|
|
if (!url1 || !title) {
|
2004-12-09 13:30:44 +03:00
|
|
|
warn_user("HotlistLoadError", "(Missing <a> in <li> or "
|
|
|
|
"memory exhausted.)");
|
|
|
|
return;
|
|
|
|
}
|
2006-01-01 21:52:30 +03:00
|
|
|
|
2009-02-06 05:04:25 +03:00
|
|
|
/* We're loading external input.
|
|
|
|
* This may be garbage, so attempt to normalise
|
|
|
|
*/
|
|
|
|
res = url_normalize(url1, &url);
|
|
|
|
if (res != URL_FUNC_OK) {
|
|
|
|
LOG(("Failed normalising '%s'", url1));
|
|
|
|
|
|
|
|
if (res == URL_FUNC_NOMEM)
|
|
|
|
warn_user("NoMemory", NULL);
|
|
|
|
|
|
|
|
xmlFree(url1);
|
|
|
|
xmlFree(title);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No longer need this */
|
|
|
|
xmlFree(url1);
|
|
|
|
|
2006-04-10 03:21:13 +04:00
|
|
|
data = urldb_get_url_data(url);
|
2006-04-15 19:23:23 +04:00
|
|
|
if (!data) {
|
2006-04-10 03:21:13 +04:00
|
|
|
/* No entry in database, so add one */
|
|
|
|
urldb_add_url(url);
|
2006-04-15 19:23:23 +04:00
|
|
|
/* now attempt to get url data */
|
|
|
|
data = urldb_get_url_data(url);
|
|
|
|
}
|
2009-02-06 05:04:25 +03:00
|
|
|
if (!data) {
|
|
|
|
xmlFree(title);
|
|
|
|
free(url);
|
|
|
|
|
2005-12-31 07:31:37 +03:00
|
|
|
return;
|
2009-02-06 05:04:25 +03:00
|
|
|
}
|
2006-04-15 19:23:23 +04:00
|
|
|
|
2006-06-15 02:46:12 +04:00
|
|
|
/* Make this URL persistent */
|
|
|
|
urldb_set_url_persistence(url, true);
|
|
|
|
|
2005-12-31 07:31:37 +03:00
|
|
|
if (!data->title)
|
2006-04-10 03:21:13 +04:00
|
|
|
urldb_set_url_title(url, title);
|
2006-04-15 19:23:23 +04:00
|
|
|
|
2006-04-10 03:21:13 +04:00
|
|
|
entry = tree_create_URL_node(directory, url, data, title);
|
2009-05-28 19:57:54 +04:00
|
|
|
if (entry == NULL) {
|
|
|
|
/** \todo why isn't this fatal? */
|
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
}
|
2009-02-06 05:04:25 +03:00
|
|
|
|
2005-12-01 05:59:55 +03:00
|
|
|
xmlFree(title);
|
2009-02-06 05:04:25 +03:00
|
|
|
free(url);
|
2004-12-09 13:30:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Search the children of an xmlNode for an element.
|
|
|
|
*
|
|
|
|
* \param node xmlNode to search children of, or 0
|
|
|
|
* \param name name of element to find
|
|
|
|
* \return first child of node which is an element and matches name, or
|
|
|
|
* 0 if not found or parameter node is 0
|
|
|
|
*/
|
2005-02-07 17:28:43 +03:00
|
|
|
xmlNode *options_find_tree_element(xmlNode *node, const char *name) {
|
2004-12-09 13:30:44 +03:00
|
|
|
xmlNode *n;
|
|
|
|
if (!node)
|
|
|
|
return 0;
|
|
|
|
for (n = node->children;
|
|
|
|
n && !(n->type == XML_ELEMENT_NODE &&
|
2008-02-25 20:58:00 +03:00
|
|
|
strcmp((const char *) n->name, name) == 0);
|
2004-12-09 13:30:44 +03:00
|
|
|
n = n->next)
|
|
|
|
;
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Perform a save to a specified file
|
|
|
|
*
|
|
|
|
* /param filename the file to save to
|
|
|
|
*/
|
2005-02-07 17:28:43 +03:00
|
|
|
bool options_save_tree(struct tree *tree, const char *filename, const char *page_title) {
|
2004-12-09 13:30:44 +03:00
|
|
|
int res;
|
|
|
|
xmlDoc *doc;
|
|
|
|
xmlNode *html, *head, *title, *body;
|
|
|
|
|
|
|
|
/* Unfortunately the Browse Hotlist format is invalid HTML,
|
|
|
|
* so this is a lie. */
|
2008-02-25 20:58:00 +03:00
|
|
|
doc = htmlNewDoc(
|
|
|
|
(const xmlChar *) "http://www.w3.org/TR/html4/strict.dtd",
|
|
|
|
(const xmlChar *) "-//W3C//DTD HTML 4.01//EN");
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!doc) {
|
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-02-25 20:58:00 +03:00
|
|
|
html = xmlNewNode(NULL, (const xmlChar *) "html");
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!html) {
|
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
xmlDocSetRootElement(doc, html);
|
|
|
|
|
2008-02-25 20:58:00 +03:00
|
|
|
head = xmlNewChild(html, NULL, (const xmlChar *) "head", NULL);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!head) {
|
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-03-01 02:03:26 +03:00
|
|
|
title = xmlNewTextChild(head, NULL, (const xmlChar *) "title",
|
2008-02-25 20:58:00 +03:00
|
|
|
(const xmlChar *) page_title);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!title) {
|
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-02-25 20:58:00 +03:00
|
|
|
body = xmlNewChild(html, NULL, (const xmlChar *) "body", NULL);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!body) {
|
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2005-02-07 17:28:43 +03:00
|
|
|
if (!options_save_tree_directory(tree->root, body)) {
|
2004-12-09 13:30:44 +03:00
|
|
|
warn_user("NoMemory", 0);
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
doc->charset = XML_CHAR_ENCODING_UTF8;
|
|
|
|
res = htmlSaveFileEnc(filename, doc, "iso-8859-1");
|
|
|
|
if (res == -1) {
|
|
|
|
warn_user("HotlistSaveError", 0);
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
xmlFreeDoc(doc);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a directory to the HTML tree for saving.
|
|
|
|
*
|
|
|
|
* \param directory hotlist directory to add
|
|
|
|
* \param node node to add ul to
|
|
|
|
* \return true on success, false on memory exhaustion
|
|
|
|
*/
|
2005-02-07 17:28:43 +03:00
|
|
|
bool options_save_tree_directory(struct node *directory, xmlNode *node) {
|
2004-12-09 13:30:44 +03:00
|
|
|
struct node *child;
|
|
|
|
xmlNode *ul, *h4;
|
|
|
|
|
2008-02-25 20:58:00 +03:00
|
|
|
ul = xmlNewChild(node, NULL, (const xmlChar *) "ul", NULL);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!ul)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (child = directory->child; child; child = child->next) {
|
|
|
|
if (!child->folder) {
|
|
|
|
/* entry */
|
2005-02-07 17:28:43 +03:00
|
|
|
if (!options_save_tree_entry(child, ul))
|
2004-12-09 13:30:44 +03:00
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
/* directory */
|
|
|
|
/* invalid HTML */
|
2008-03-01 02:03:26 +03:00
|
|
|
h4 = xmlNewTextChild(ul, NULL,
|
|
|
|
(const xmlChar *) "h4",
|
2008-02-25 20:58:00 +03:00
|
|
|
(const xmlChar *) child->data.text);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!h4)
|
|
|
|
return false;
|
|
|
|
|
2005-02-07 17:28:43 +03:00
|
|
|
if (!options_save_tree_directory(child, ul))
|
2004-12-09 13:30:44 +03:00
|
|
|
return false;
|
|
|
|
} }
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add an entry to the HTML tree for saving.
|
|
|
|
*
|
|
|
|
* The node must contain a sequence of node_elements in the following order:
|
|
|
|
*
|
|
|
|
* \param entry hotlist entry to add
|
|
|
|
* \param node node to add li to
|
|
|
|
* \return true on success, false on memory exhaustion
|
|
|
|
*/
|
2005-02-07 17:28:43 +03:00
|
|
|
bool options_save_tree_entry(struct node *entry, xmlNode *node) {
|
2004-12-09 13:30:44 +03:00
|
|
|
xmlNode *li, *a;
|
|
|
|
xmlAttr *href;
|
|
|
|
struct node_element *element;
|
|
|
|
|
2008-02-25 20:58:00 +03:00
|
|
|
li = xmlNewChild(node, NULL, (const xmlChar *) "li", NULL);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!li)
|
|
|
|
return false;
|
|
|
|
|
2008-03-01 02:03:26 +03:00
|
|
|
a = xmlNewTextChild(li, NULL, (const xmlChar *) "a",
|
2008-02-25 20:58:00 +03:00
|
|
|
(const xmlChar *) entry->data.text);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!a)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
element = tree_find_element(entry, TREE_ELEMENT_URL);
|
|
|
|
if (!element)
|
|
|
|
return false;
|
2008-03-01 02:03:26 +03:00
|
|
|
href = xmlNewProp(a, (const xmlChar *) "href",
|
2008-02-25 20:58:00 +03:00
|
|
|
(const xmlChar *) element->text);
|
2004-12-09 13:30:44 +03:00
|
|
|
if (!href)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|