Keep child objects of a HTML content in a list, instead of an array

svn path=/trunk/netsurf/; revision=11963
This commit is contained in:
John Mark Bell 2011-03-10 23:08:34 +00:00
parent 02361309ed
commit 8f3ed71cba
3 changed files with 127 additions and 118 deletions

View File

@ -190,9 +190,6 @@ bool xml_to_box(xmlNode *n, struct content *c)
root.float_children = NULL; root.float_children = NULL;
root.next_float = NULL; root.next_float = NULL;
c->data.html.object_count = 0;
c->data.html.object = 0;
/* The root box's style */ /* The root box's style */
if (!convert_xml_to_box(n, c, NULL, &root, if (!convert_xml_to_box(n, c, NULL, &root,
&inline_container, 0, 0, 0)) &inline_container, 0, 0, 0))

View File

@ -66,7 +66,7 @@ static bool html_find_stylesheets(struct content *c, xmlNode *html);
static bool html_process_style_element(struct content *c, unsigned int *index, static bool html_process_style_element(struct content *c, unsigned int *index,
xmlNode *style); xmlNode *style);
static void html_inline_style_done(struct content_css_data *css, void *pw); static void html_inline_style_done(struct content_css_data *css, void *pw);
static bool html_replace_object(struct content *c, unsigned int i, static bool html_replace_object(struct content_html_object *object,
const char *url); const char *url);
static nserror html_object_callback(hlcache_handle *object, static nserror html_object_callback(hlcache_handle *object,
const hlcache_event *event, void *pw); const hlcache_event *event, void *pw);
@ -123,15 +123,14 @@ bool html_create(struct content *c, const http_parameter *params)
html->stylesheet_count = 0; html->stylesheet_count = 0;
html->stylesheets = NULL; html->stylesheets = NULL;
html->select_ctx = NULL; html->select_ctx = NULL;
html->object_count = 0; html->num_objects = 0;
html->object = NULL; html->object_list = NULL;
html->forms = NULL; html->forms = NULL;
html->imagemaps = NULL; html->imagemaps = NULL;
html->bw = NULL; html->bw = NULL;
html->frameset = NULL; html->frameset = NULL;
html->iframe = NULL; html->iframe = NULL;
html->page = NULL; html->page = NULL;
html->index = 0;
html->box = NULL; html->box = NULL;
html->font_func = &nsfont; html->font_func = &nsfont;
@ -1250,9 +1249,7 @@ bool html_fetch_object(struct content *c, const char *url, struct box *box,
int available_width, int available_height, int available_width, int available_height,
bool background) bool background)
{ {
unsigned int i = c->data.html.object_count;
struct content_html_object *object; struct content_html_object *object;
hlcache_handle *c_fetch;
hlcache_child_context child; hlcache_child_context child;
char *url2; char *url2;
url_func_result res; url_func_result res;
@ -1268,29 +1265,37 @@ bool html_fetch_object(struct content *c, const char *url, struct box *box,
return res != URL_FUNC_NOMEM; return res != URL_FUNC_NOMEM;
} }
object = talloc(c, struct content_html_object);
if (object == NULL) {
free(url2);
return false;
}
object->parent = c;
object->next = NULL;
object->content = NULL;
object->box = box;
object->permitted_types = permitted_types;
object->background = background;
error = hlcache_handle_retrieve(url2, 0, content__get_url(c), NULL, error = hlcache_handle_retrieve(url2, 0, content__get_url(c), NULL,
html_object_callback, c, &child, permitted_types, html_object_callback, object, &child, permitted_types,
&c_fetch); &object->content);
/* No longer need normalized url */ /* No longer need normalized url */
free(url2); free(url2);
if (error == NSERROR_OK) { if (error != NSERROR_OK) {
talloc_free(object);
return error != NSERROR_NOMEM;
}
/* add to object list */ /* add to object list */
object = talloc_realloc(c, c->data.html.object, object->next = c->data.html.object_list;
struct content_html_object, i + 1); c->data.html.object_list = object;
if (object == NULL) {
hlcache_handle_release(c_fetch); c->data.html.num_objects++;
return false;
}
c->data.html.object = object;
c->data.html.object[i].box = box;
c->data.html.object[i].permitted_types = permitted_types;
c->data.html.object[i].background = background;
c->data.html.object[i].content = c_fetch;
c->data.html.object_count++;
c->active++; c->active++;
}
return error != NSERROR_NOMEM; return error != NSERROR_NOMEM;
} }
@ -1299,34 +1304,36 @@ bool html_fetch_object(struct content *c, const char *url, struct box *box,
/** /**
* Start a fetch for an object required by a page, replacing an existing object. * Start a fetch for an object required by a page, replacing an existing object.
* *
* \param c content of type CONTENT_HTML * \param object Object to replace
* \param i index of object to replace in c->data.html.object
* \param url URL of object to fetch (copied) * \param url URL of object to fetch (copied)
* \return true on success, false on memory exhaustion * \return true on success, false on memory exhaustion
*/ */
bool html_replace_object(struct content *c, unsigned int i, const char *url) bool html_replace_object(struct content_html_object *object, const char *url)
{ {
hlcache_handle *c_fetch; struct content *c;
hlcache_child_context child; hlcache_child_context child;
struct content *page; struct content *page;
char *url2; char *url2;
url_func_result res; url_func_result res;
nserror error; nserror error;
assert(c->type == CONTENT_HTML); assert(object != NULL);
c = object->parent;
child.charset = c->data.html.encoding; child.charset = c->data.html.encoding;
child.quirks = c->quirks; child.quirks = c->quirks;
if (c->data.html.object[i].content) { if (object->content != NULL) {
/* remove existing object */ /* remove existing object */
if (content_get_status(c->data.html.object[i].content) != if (content_get_status(object->content) != CONTENT_STATUS_DONE)
CONTENT_STATUS_DONE)
c->active--; c->active--;
hlcache_handle_release(c->data.html.object[i].content);
c->data.html.object[i].content = NULL; hlcache_handle_release(object->content);
c->data.html.object[i].box->object = NULL; object->content = NULL;
object->box->object = NULL;
} }
res = url_normalize(url, &url2); res = url_normalize(url, &url2);
@ -1336,16 +1343,14 @@ bool html_replace_object(struct content *c, unsigned int i, const char *url)
/* initialise fetch */ /* initialise fetch */
error = hlcache_handle_retrieve(url2, 0, content__get_url(c), NULL, error = hlcache_handle_retrieve(url2, 0, content__get_url(c), NULL,
html_object_callback, c, &child, html_object_callback, c, &child,
c->data.html.object[i].permitted_types, object->permitted_types,
&c_fetch); &object->content);
free(url2); free(url2);
if (error != NSERROR_OK) if (error != NSERROR_OK)
return false; return false;
c->data.html.object[i].content = c_fetch;
for (page = c; page; page = page->data.html.page) { for (page = c; page; page = page->data.html.page) {
assert(page->type == CONTENT_HTML); assert(page->type == CONTENT_HTML);
page->active++; page->active++;
@ -1363,21 +1368,11 @@ bool html_replace_object(struct content *c, unsigned int i, const char *url)
nserror html_object_callback(hlcache_handle *object, nserror html_object_callback(hlcache_handle *object,
const hlcache_event *event, void *pw) const hlcache_event *event, void *pw)
{ {
struct content *c = pw; struct content_html_object *o = pw;
unsigned int i; struct content *c = o->parent;
struct content_html_object *o;
int x, y; int x, y;
struct box *box; struct box *box;
/* Find object record in parent */
for (i = 0, o = c->data.html.object; i != c->data.html.object_count;
i++, o++) {
if (o->content == object)
break;
}
assert(i != c->data.html.object_count);
box = o->box; box = o->box;
switch (event->type) { switch (event->type) {
@ -1388,7 +1383,7 @@ nserror html_object_callback(hlcache_handle *object,
if (c->data.html.bw != NULL) if (c->data.html.bw != NULL)
content_open(object, content_open(object,
c->data.html.bw, c, c->data.html.bw, c,
i, box, 0, box,
box->object_params); box->object_params);
break; break;
} }
@ -1404,8 +1399,7 @@ nserror html_object_callback(hlcache_handle *object,
html_set_status(c, messages_get("BadObject")); html_set_status(c, messages_get("BadObject"));
content_broadcast(c, CONTENT_MSG_STATUS, event->data); content_broadcast(c, CONTENT_MSG_STATUS, event->data);
html_object_failed(box, c, html_object_failed(box, c, o->background);
c->data.html.object[i].background);
break; break;
case CONTENT_MSG_READY: case CONTENT_MSG_READY:
@ -1480,10 +1474,22 @@ nserror html_object_callback(hlcache_handle *object,
break; break;
case CONTENT_MSG_REFRESH: case CONTENT_MSG_REFRESH:
if (content_get_type(object) == CONTENT_HTML) if (content_get_type(object) == CONTENT_HTML) {
struct content_html_object *o;
for (o = c->data.html.object_list; o != NULL;
o = o->next) {
if (o->content == object)
break;
}
assert(o != NULL);
/* only for HTML objects */ /* only for HTML objects */
schedule(event->data.delay * 100, schedule(event->data.delay * 100,
html_object_refresh, object); html_object_refresh, o);
}
break; break;
default: default:
@ -1592,23 +1598,21 @@ bool html_object_type_permitted(const content_type type,
void html_object_refresh(void *p) void html_object_refresh(void *p)
{ {
hlcache_handle *h = (hlcache_handle *) p; struct content_html_object *object = p;
struct content *c = hlcache_handle_get_content(h);
const char *refresh_url; const char *refresh_url;
assert(content_get_type(h) == CONTENT_HTML); assert(content_get_type(object->content) == CONTENT_HTML);
refresh_url = content_get_refresh_url(h); refresh_url = content_get_refresh_url(object->content);
/* Ignore if refresh URL has gone /* Ignore if refresh URL has gone
* (may happen if fetch errored) */ * (may happen if fetch errored) */
if (refresh_url == NULL) if (refresh_url == NULL)
return; return;
content_invalidate_reuse_data(h); content_invalidate_reuse_data(object->content);
if (!html_replace_object(c->data.html.page, c->data.html.index, if (!html_replace_object(object, refresh_url)) {
refresh_url)) {
/** \todo handle memory exhaustion */ /** \todo handle memory exhaustion */
} }
} }
@ -1619,23 +1623,23 @@ void html_object_refresh(void *p)
void html_stop(struct content *c) void html_stop(struct content *c)
{ {
unsigned int i; struct content_html_object *object;
hlcache_handle *object;
assert(c->status == CONTENT_STATUS_READY); assert(c->status == CONTENT_STATUS_READY);
for (i = 0; i != c->data.html.object_count; i++) { for (object = c->data.html.object_list; object != NULL;
object = c->data.html.object[i].content; object = object->next) {
if (object == NULL) if (object->content == NULL)
continue; continue;
if (content_get_status(object) == CONTENT_STATUS_DONE) if (content_get_status(object->content) == CONTENT_STATUS_DONE)
; /* already loaded: do nothing */ ; /* already loaded: do nothing */
else if (content_get_status(object) == CONTENT_STATUS_READY) else if (content_get_status(object->content) ==
hlcache_handle_abort(object); CONTENT_STATUS_READY)
hlcache_handle_abort(object->content);
else { else {
hlcache_handle_release(object); hlcache_handle_release(object->content);
c->data.html.object[i].content = NULL; object->content = NULL;
} }
} }
c->status = CONTENT_STATUS_DONE; c->status = CONTENT_STATUS_DONE;
@ -1770,16 +1774,20 @@ void html_destroy(struct content *c)
} }
/* Free objects */ /* Free objects */
for (i = 0; i != html->object_count; i++) { while (html->object_list != NULL) {
LOG(("object %i %p", i, html->object[i].content)); struct content_html_object *victim = html->object_list;
if (html->object[i].content != NULL) {
if (content_get_type(html->object[i].content) ==
CONTENT_HTML)
schedule_remove(html_object_refresh,
html->object[i].content);
hlcache_handle_release(html->object[i].content); LOG(("object %p", victim->content));
if (victim->content != NULL) {
if (content_get_type(victim->content) == CONTENT_HTML)
schedule_remove(html_object_refresh, victim);
hlcache_handle_release(victim->content);
} }
html->object_list = victim->next;
talloc_free(victim);
} }
} }
@ -1844,18 +1852,18 @@ bool html_clone(const struct content *old, struct content *new_content)
void html_set_status(struct content *c, const char *extra) void html_set_status(struct content *c, const char *extra)
{ {
unsigned int stylesheets = 0, objects = 0; unsigned int stylesheets = 0, objects = 0;
if (c->data.html.object_count == 0) if (c->data.html.num_objects == 0)
stylesheets = c->data.html.stylesheet_count - c->active; stylesheets = c->data.html.stylesheet_count - c->active;
else { else {
stylesheets = c->data.html.stylesheet_count; stylesheets = c->data.html.stylesheet_count;
objects = c->data.html.object_count - c->active; objects = c->data.html.num_objects - c->active;
} }
content_set_status(c, "%u/%u %s %u/%u %s %s", content_set_status(c, "%u/%u %s %u/%u %s %s",
stylesheets, c->data.html.stylesheet_count, stylesheets, c->data.html.stylesheet_count,
messages_get((c->data.html.stylesheet_count == 1) ? messages_get((c->data.html.stylesheet_count == 1) ?
"styl" : "styls"), "styl" : "styls"),
objects, c->data.html.object_count, objects, c->data.html.num_objects,
messages_get((c->data.html.object_count == 1) ? messages_get((c->data.html.num_objects == 1) ?
"obj" : "objs"), "obj" : "objs"),
extra); extra);
} }
@ -1869,21 +1877,25 @@ void html_open(struct content *c, struct browser_window *bw,
struct content *page, unsigned int index, struct box *box, struct content *page, unsigned int index, struct box *box,
struct object_params *params) struct object_params *params)
{ {
unsigned int i; struct content_html_object *object, *next;
c->data.html.bw = bw; c->data.html.bw = bw;
c->data.html.page = page; c->data.html.page = page;
c->data.html.index = index;
c->data.html.box = box; c->data.html.box = box;
for (i = 0; i != c->data.html.object_count; i++) {
if (c->data.html.object[i].content == 0) for (object = c->data.html.object_list; object != NULL; object = next) {
next = object->next;
if (object->content == NULL)
continue; continue;
if (content_get_type(c->data.html.object[i].content) ==
CONTENT_UNKNOWN) if (content_get_type(object->content) == CONTENT_UNKNOWN)
continue; continue;
content_open(c->data.html.object[i].content,
bw, c, i, content_open(object->content,
c->data.html.object[i].box, bw, c, 0,
c->data.html.object[i].box->object_params); object->box,
object->box->object_params);
} }
} }
@ -1894,24 +1906,23 @@ void html_open(struct content *c, struct browser_window *bw,
void html_close(struct content *c) void html_close(struct content *c)
{ {
unsigned int i; struct content_html_object *object, *next;
c->data.html.bw = 0; c->data.html.bw = 0;
for (i = 0; i != c->data.html.object_count; i++) { for (object = c->data.html.object_list; object != NULL; object = next) {
if (c->data.html.object[i].content == NULL) next = object->next;
if (object->content == NULL)
continue; continue;
if (content_get_type(c->data.html.object[i].content) == if (content_get_type(object->content) == CONTENT_UNKNOWN)
CONTENT_UNKNOWN)
continue; continue;
if (content_get_type(c->data.html.object[i].content) == if (content_get_type(object->content) == CONTENT_HTML)
CONTENT_HTML) schedule_remove(html_object_refresh, object);
schedule_remove(html_object_refresh,
c->data.html.object[i].content);
content_close(c->data.html.object[i].content); content_close(object->content);
} }
} }
@ -2124,7 +2135,7 @@ struct html_stylesheet *html_get_stylesheets(hlcache_handle *h, unsigned int *n)
* *
* \param h Content to retrieve objects from * \param h Content to retrieve objects from
* \param n Pointer to location to receive number of objects * \param n Pointer to location to receive number of objects
* \return Pointer to array of objects * \return Pointer to list of objects
*/ */
struct content_html_object *html_get_objects(hlcache_handle *h, unsigned int *n) struct content_html_object *html_get_objects(hlcache_handle *h, unsigned int *n)
{ {
@ -2134,9 +2145,9 @@ struct content_html_object *html_get_objects(hlcache_handle *h, unsigned int *n)
assert(c->type == CONTENT_HTML); assert(c->type == CONTENT_HTML);
assert(n != NULL); assert(n != NULL);
*n = c->data.html.object_count; *n = c->data.html.num_objects;
return c->data.html.object; return c->data.html.object_list;
} }
/** /**

View File

@ -85,6 +85,9 @@ typedef enum {
/** An object (<img>, <object>, etc.) in a CONTENT_HTML document. */ /** An object (<img>, <object>, etc.) in a CONTENT_HTML document. */
struct content_html_object { struct content_html_object {
struct content *parent; /**< Parent document */
struct content_html_object *next; /**< Next in chain */
struct hlcache_handle *content; /**< Content, or 0. */ struct hlcache_handle *content; /**< Content, or 0. */
struct box *box; /**< Node in box tree containing it. */ struct box *box; /**< Node in box tree containing it. */
/** Pointer to array of permitted content_type, terminated by /** Pointer to array of permitted content_type, terminated by
@ -157,10 +160,10 @@ struct content_html_data {
/**< Style selection context */ /**< Style selection context */
css_select_ctx *select_ctx; css_select_ctx *select_ctx;
/** Number of entries in object. */ /** Number of entries in object_list. */
unsigned int object_count; unsigned int num_objects;
/** Objects. Each may be 0. */ /** List of objects. */
struct content_html_object *object; struct content_html_object *object_list;
/** Forms, in reverse order to document. */ /** Forms, in reverse order to document. */
struct form *forms; struct form *forms;
/** Hash table of imagemaps. */ /** Hash table of imagemaps. */
@ -178,8 +181,6 @@ struct content_html_data {
/** Content of type CONTENT_HTML containing this, or 0 if not an object /** Content of type CONTENT_HTML containing this, or 0 if not an object
* within a page. */ * within a page. */
struct content *page; struct content *page;
/** Index in page->data.html.object, or 0 if not an object. */
unsigned int index;
/** Box containing this, or 0 if not an object. */ /** Box containing this, or 0 if not an object. */
struct box *box; struct box *box;
}; };