/** Document (meta) refresh. * @file */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include <stdio.h> #include <stdlib.h> #include <string.h> #include "elinks.h" #include "config/options.h" #include "document/document.h" #include "document/refresh.h" #include "document/view.h" #include "main/select.h" #include "main/timer.h" #include "protocol/uri.h" #include "session/download.h" #include "session/session.h" #include "session/task.h" #include "util/error.h" #include "util/memory.h" #include "util/string.h" struct document_refresh * init_document_refresh(char *url, unsigned long seconds) { struct document_refresh *refresh; refresh = (struct document_refresh *)mem_alloc(sizeof(*refresh)); if (!refresh) return NULL; refresh->uri = get_uri(url, URI_NONE); if (!refresh->uri) { mem_free(refresh); return NULL; } refresh->seconds = seconds; refresh->timer = TIMER_ID_UNDEF; refresh->restart = 1; return refresh; } void kill_document_refresh(struct document_refresh *refresh) { kill_timer(&refresh->timer); } void done_document_refresh(struct document_refresh *refresh) { kill_document_refresh(refresh); done_uri(refresh->uri); mem_free(refresh); } /** Timer callback for document_refresh.timer. As explained in * install_timer(), this function must erase the expired timer ID from * all variables. */ static void do_document_refresh(void *data) { struct document_view *doc_view = (struct document_view *)data; struct document_refresh *refresh = doc_view->document->refresh; struct type_query *type_query; assert(refresh); refresh->timer = TIMER_ID_UNDEF; /* The expired timer ID has now been erased. */ /* When refreshing documents that will trigger a download (like * sourceforge's download pages) make sure that we do not endlessly * trigger the download (bug 289). */ foreach (type_query, doc_view->session->type_queries) if (compare_uri(refresh->uri, type_query->uri, URI_BASE)) return; if (compare_uri(refresh->uri, doc_view->document->uri, 0)) { /* If the refreshing is for the current URI, force a reload. */ reload_frame(doc_view->session, doc_view->name, CACHE_MODE_FORCE_RELOAD); } else { /* This makes sure that we send referer. */ goto_uri_frame(doc_view->session, refresh->uri, doc_view->name, CACHE_MODE_NORMAL); /* XXX: A possible very wrong work-around for refreshing used when * downloading files. */ refresh->restart = 0; } } static void start_document_refresh(struct document_refresh *refresh, struct document_view *doc_view) { milliseconds_T minimum = (milliseconds_T) get_opt_int("document.browse.minimum_refresh_time", doc_view->session); milliseconds_T refresh_delay = sec_to_ms(refresh->seconds); milliseconds_T time = ms_max(refresh_delay, minimum); struct type_query *type_query; /* FIXME: This is just a work-around for stopping more than one timer * from being started at anytime. The refresh timer should maybe belong * to the session? The multiple refresh timers is triggered by * http://ttforums.owenrudge.net/login.php when pressing 'Log in' and * waiting for it to refresh. --jonas */ if (!refresh->restart || refresh->timer != TIMER_ID_UNDEF) return; /* Like bug 289 another sourceforge download thingy this time with * number 434. It should take care when refreshing to the same URI or * what ever the cause is. */ foreach (type_query, doc_view->session->type_queries) if (compare_uri(refresh->uri, type_query->uri, URI_BASE)) return; if (time == 0) { /* Use register_bottom_half() instead of install_timer() * because the latter asserts that the delay is greater * than 0. */ register_bottom_half(do_document_refresh, doc_view); return; } install_timer(&refresh->timer, time, do_document_refresh, doc_view); } void start_document_refreshes(struct session *ses) { assert(ses); if (!ses->doc_view || !ses->doc_view->document || !get_opt_bool("document.browse.refresh", ses)) return; if (document_has_frames(ses->doc_view->document)) { struct document_view *doc_view; foreach (doc_view, ses->scrn_frames) { if (!doc_view->document || !doc_view->document->refresh) continue; start_document_refresh(doc_view->document->refresh, doc_view); } } if (!ses->doc_view->document->refresh) return; start_document_refresh(ses->doc_view->document->refresh, ses->doc_view); }