1 /***************************************************************************
3 * Project ___| | | | _ \| |
5 * | (__| |_| | _ <| |___
6 * \___|\___/|_| \_\_____|
8 * Copyright (C) 1998 - 2011, Daniel Stenberg, <daniel@haxx.se>, et al.
10 * This software is licensed as described in the file COPYING, which
11 * you should have received as part of this distribution. The terms
12 * are also available at http://curl.haxx.se/docs/copyright.html.
14 * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15 * copies of the Software, and permit persons to whom the Software is
16 * furnished to do so, under the terms of the COPYING file.
18 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19 * KIND, either express or implied.
21 ***************************************************************************/
22 /* This is a multi threaded application that uses a progress bar to show
23 * status. It uses Gtk+ to make a smooth pulse.
25 * Written by Jud Bishop after studying the other examples provided with
28 * To compile (on a single line):
29 * gcc -ggdb `pkg-config --cflags --libs gtk+-2.0` -lcurl -lssl -lcrypto
30 * -lgthread-2.0 -dl smooth-gtk-thread.c -o smooth-gtk-thread
39 #include <curl/curl.h>
43 pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
45 gint num_urls = 9; /* Just make sure this is less than urls[]*/
46 const char * const urls[]= {
58 size_t write_file(void *ptr, size_t size, size_t nmemb, FILE *stream)
60 /* printf("write_file\n"); */
61 return fwrite(ptr, size, nmemb, stream);
64 /* http://xoap.weather.com/weather/local/46214?cc=*&dayf=5&unit=i */
65 void *pull_one_url(void *NaN)
72 /* Stop threads from entering unless j is incremented */
73 pthread_mutex_lock(&lock);
74 while ( j < num_urls )
76 printf("j = %d\n", j);
79 g_strdup_printf("xoap.weather.com/weather/local/%s?cc=*&dayf=5&unit=i\n",
82 printf( "http %s", http );
84 curl = curl_easy_init();
88 outfile = fopen(urls[j], "w");
89 /* printf("fopen\n"); */
91 /* Set the URL and transfer type */
92 curl_easy_setopt(curl, CURLOPT_URL, http);
94 /* Write to the file */
95 curl_easy_setopt(curl, CURLOPT_WRITEDATA, outfile);
96 curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_file);
98 j++; /* critical line */
99 pthread_mutex_unlock(&lock);
101 res = curl_easy_perform(curl);
106 curl_easy_cleanup(curl);
110 /* Adds more latency, testing the mutex.*/
118 gboolean pulse_bar(gpointer data)
121 gtk_progress_bar_pulse (GTK_PROGRESS_BAR (data));
124 /* Return true so the function will be called again;
125 * returning false removes this timeout function.
130 void *create_thread(void *progress_bar)
136 /* Make sure I don't create more threads than urls. */
137 for(i=0; i < NUMT && i < num_urls ; i++) {
138 error = pthread_create(&tid[i],
139 NULL, /* default attributes please */
143 fprintf(stderr, "Couldn't run thread number %d, errno %d\n", i, error);
145 fprintf(stderr, "Thread %d, gets %s\n", i, urls[i]);
148 /* Wait for all threads to terminate. */
149 for(i=0; i < NUMT && i < num_urls; i++) {
150 error = pthread_join(tid[i], NULL);
151 fprintf(stderr, "Thread %d terminated\n", i);
154 /* This stops the pulsing if you have it turned on in the progress bar
156 g_source_remove(GPOINTER_TO_INT(g_object_get_data(G_OBJECT(progress_bar),
159 /* This destroys the progress bar */
160 gtk_widget_destroy(progress_bar);
162 /* [Un]Comment this out to kill the program rather than pushing close. */
163 /* gtk_main_quit(); */
170 static gboolean cb_delete(GtkWidget *window, gpointer data)
176 int main(int argc, char **argv)
178 GtkWidget *top_window, *outside_frame, *inside_frame, *progress_bar;
180 /* Must initialize libcurl before any threads are started */
181 curl_global_init(CURL_GLOBAL_ALL);
186 gdk_threads_enter ();
188 gtk_init(&argc, &argv);
191 top_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
194 outside_frame = gtk_frame_new(NULL);
195 gtk_frame_set_shadow_type(GTK_FRAME(outside_frame), GTK_SHADOW_OUT);
196 gtk_container_add(GTK_CONTAINER(top_window), outside_frame);
199 inside_frame = gtk_frame_new(NULL);
200 gtk_frame_set_shadow_type(GTK_FRAME(inside_frame), GTK_SHADOW_IN);
201 gtk_container_set_border_width(GTK_CONTAINER(inside_frame), 5);
202 gtk_container_add(GTK_CONTAINER(outside_frame), inside_frame);
205 progress_bar = gtk_progress_bar_new();
206 gtk_progress_bar_pulse (GTK_PROGRESS_BAR (progress_bar));
207 /* Make uniform pulsing */
208 gint pulse_ref = g_timeout_add (300, pulse_bar, progress_bar);
209 g_object_set_data(G_OBJECT(progress_bar), "pulse_id",
210 GINT_TO_POINTER(pulse_ref));
211 gtk_container_add(GTK_CONTAINER(inside_frame), progress_bar);
213 gtk_widget_show_all(top_window);
214 printf("gtk_widget_show_all\n");
216 g_signal_connect(G_OBJECT (top_window), "delete-event",
217 G_CALLBACK(cb_delete), NULL);
219 if (!g_thread_create(&create_thread, progress_bar, FALSE, NULL) != 0)
220 g_warning("can't create the thread");
224 printf("gdk_threads_leave\n");