2 * Copyright (C) 2009-2011 Andy Spencer <andy753421@gmail.com>
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <http://www.gnu.org/licenses/>.
20 * @short_description: Environment plugin
22 * #GritsPluginEnv provides environmental information such as sky images. It can
23 * also paint a blank overlay on the surface so that other plugins can draw
24 * transparent overlays nicely.
42 static void sky_expose(GritsCallback *sky, GritsOpenGL *opengl, gpointer _env)
44 GritsPluginEnv *env = GRITS_PLUGIN_ENV(_env);
45 g_debug("GritsPluginEnv: expose_sky");
47 gdouble lat, lon, elev;
48 grits_viewer_get_location(env->viewer, &lat, &lon, &elev);
51 gdouble rg = MAX(0, 1-(elev/40000));
52 gdouble blue = MAX(0, 1-(elev/100000));
53 glClearColor(MIN(0.4,rg), MIN(0.4,rg), MIN(1,blue), 1.0f);
54 glClear(GL_COLOR_BUFFER_BIT);
56 /* Attempt to render an atmosphere */
57 glEnable(GL_COLOR_MATERIAL);
58 glDisable(GL_CULL_FACE);
59 glDisable(GL_LIGHTING);
60 glMatrixMode(GL_MODELVIEW);
61 glBlendFunc(GL_SRC_ALPHA, GL_DST_ALPHA);
62 grits_viewer_center_position(env->viewer, lat, lon, -EARTH_R);
64 gdouble ds = EARTH_R+elev; // distance to self
65 gdouble da = EARTH_R+300000; // distance to top of atmosphere
66 gdouble dg = EARTH_R-100000; // distance to top of atmosphere
67 gdouble ang = acos(EARTH_R/ds); // angle to horizon
70 gdouble ar = sin(ang)*da; // top of quad fan "atomosphere"j
71 gdouble az = cos(ang)*da; //
73 gdouble gr = sin(ang)*dg; // bottom of quad fan "ground"
74 gdouble gz = cos(ang)*dg; //
76 glBegin(GL_QUAD_STRIP);
77 for (gdouble i = 0; i <= 2*G_PI; i += G_PI/30) {
78 glColor4f(0.3, 0.3, 1.0, 1.0); glVertex3f(gr*sin(i), gr*cos(i), gz);
79 glColor4f(0.3, 0.3, 1.0, 0.0); glVertex3f(ar*sin(i), ar*cos(i), az);
85 static void compass_draw_compass(gdouble scale)
87 gfloat thick = scale * 0.20;
90 float light_ambient[] = {0.4f, 0.4f, 0.4f, 0.4f};
91 float light_diffuse[] = {0.9f, 0.9f, 0.9f, 1.0f};
92 float light_position[] = {-scale*2, -scale*4, -scale*0.5, 1.0f};
93 glLightfv(GL_LIGHT0, GL_AMBIENT, light_ambient);
94 glLightfv(GL_LIGHT0, GL_DIFFUSE, light_diffuse);
95 glLightfv(GL_LIGHT0, GL_POSITION, light_position);
98 gdouble colors[][3] = {
104 gdouble points[][3] = {
123 glEnable(GL_POLYGON_OFFSET_FILL);
124 glPolygonOffset(1, 1);
125 for (int i = 0; i < G_N_ELEMENTS(colors); i++) {
126 gdouble *color = colors[i];
128 for (int j = 0; j < G_N_ELEMENTS(faces); j++) {
130 gdouble *v0 = points[faces[j][0]];
131 gdouble *v1 = points[faces[j][1]];
132 gdouble *v2 = points[faces[j][2]];
133 crossd3(v2, v1, v0, norm);
135 glBegin(GL_TRIANGLES);
141 glRotatef(90, 0, 0, 1);
145 glEnable(GL_POLYGON_OFFSET_LINE);
146 glDisable(GL_LIGHTING);
147 glPolygonOffset(0, 0);
148 glColor4f(0, 0, 0, 0.25);
150 for (int i = 0; i < G_N_ELEMENTS(colors); i++) {
151 glBegin(GL_LINE_STRIP);
152 for (int j = 0; j < G_N_ELEMENTS(outline); j++)
153 glVertex3dv(points[outline[j]]);
155 glRotatef(90, 0, 0, 1);
159 static gboolean compass_draw_teapot(gdouble scale, GritsPluginEnv *env)
161 static int teatime = 0;
163 static int init, argc; char *argv[] = {"", NULL};
165 teatime = grits_prefs_get_boolean(env->prefs, "grits/teatime", NULL);
166 glutInit(&argc, argv);
168 g_message("teatime=%d", teatime);
173 float light_ambient[] = {0.1f, 0.1f, 0.0f, 1.0f};
174 float light_diffuse[] = {0.9f, 0.9f, 0.9f, 1.0f};
175 float light_position[] = {-50.0f, -40.0f, -80.0f, 1.0f};
176 glLightfv(GL_LIGHT0, GL_AMBIENT, light_ambient);
177 glLightfv(GL_LIGHT0, GL_DIFFUSE, light_diffuse);
178 glLightfv(GL_LIGHT0, GL_POSITION, light_position);
181 glRotatef(-90, 0, 0, 1);
182 glRotatef(-90, 1, 0, 0);
183 glColor4f(0.9, 0.9, 0.7, 1.0);
184 glutSolidTeapot(scale * 0.60);
190 static void compass_expose(GritsCallback *compass, GritsOpenGL *opengl, gpointer _env)
192 GritsPluginEnv *env = GRITS_PLUGIN_ENV(_env);
193 g_debug("GritsPluginEnv: compass_expose");
195 grits_viewer_get_rotation(env->viewer, &x, &y, &z);
197 /* Setup projection */
199 gtk_widget_get_allocation(GTK_WIDGET(opengl), &alloc);
200 float scale = CLAMP(MIN(alloc.width,alloc.height)/2.0 * 0.1, 40, 100);
201 float offset = scale + 20;
202 glTranslatef(alloc.width - offset, offset, 0);
205 glClear(GL_DEPTH_BUFFER_BIT);
206 glEnable(GL_COLOR_MATERIAL);
207 glEnable(GL_DEPTH_TEST);
208 glDisable(GL_CULL_FACE);
209 glEnable(GL_POLYGON_SMOOTH);
210 glEnable(GL_LINE_SMOOTH);
213 x = CLAMP(x, -66, 66);;
214 glRotatef(x, 1, 0, 0);
215 glRotatef(-z, 0, 0, 1);
216 if (!compass_draw_teapot(scale, env))
217 compass_draw_compass(scale);
220 static gboolean compass_click(GritsCallback *compass, GdkEvent *evnet, GritsViewer *viewer)
222 grits_viewer_set_rotation(viewer, 0, 0, 0);
227 static void info_expose(GritsCallback *compass, GritsOpenGL *opengl, gpointer _env)
230 gtk_widget_get_allocation(GTK_WIDGET(opengl), &alloc);
232 /* Create cairo surface */
234 const gchar *label0 = "Location: %7.3lf°, %8.3lf°, %4.0fm";
235 const gchar *label1 = "Cursor: %7.3lf°, %8.3lf°, %4.0fm";
237 gdouble height = 200;
238 cairo_surface_t *surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, width, height);
239 cairo_t *cairo = cairo_create(surface);
242 gdouble lat, lon, elev;
243 grits_viewer_get_location(GRITS_VIEWER(opengl), &lat, &lon, &elev);
244 gchar *text0 = g_strdup_printf(label0, lat, lon, elev);
245 gchar *text1 = g_strdup_printf(label1, lat, lon, elev);
248 cairo_set_line_width(cairo, 3);
249 cairo_set_source_rgba(cairo, 0, 0, 0, 0.75);
250 cairo_move_to(cairo, 2, 20); cairo_text_path(cairo, text0);
251 cairo_move_to(cairo, 2, 40); cairo_text_path(cairo, text1);
255 cairo_set_source_rgba(cairo, 1, 1, 1, 1);
256 cairo_move_to(cairo, 2, 20); cairo_show_text(cairo, text0);
257 cairo_move_to(cairo, 2, 40); cairo_show_text(cairo, text1);
260 PangoLayout *layout = pango_cairo_create_layout(cairo);
261 PangoFontDescription *font = pango_font_description_from_string("Mono 9");
262 pango_layout_set_font_description(layout, font);
263 pango_font_description_free(font);
264 pango_layout_set_text(layout, text0, -1);
265 pango_cairo_update_layout(cairo, layout);
266 cairo_set_line_join(cairo, CAIRO_LINE_JOIN_ROUND);
267 cairo_move_to(cairo, 2, 40);
268 pango_cairo_layout_path(cairo, layout);
269 for (float w = 0.2; w <= 0.8; w+=0.2) {
270 cairo_set_line_width(cairo, (1-w)*8);
271 cairo_set_source_rgba(cairo, 0, 0, 0, w);
272 cairo_stroke_preserve(cairo);
274 cairo_set_source_rgba(cairo, 1, 1, 1, 1);
275 pango_cairo_show_layout(cairo, layout);
277 /* Load GL texture */
278 glEnable(GL_TEXTURE_2D);
279 glGenTextures(1, &tex);
280 glBindTexture(GL_TEXTURE_2D, tex);
281 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
282 glPixelStorei(GL_PACK_ALIGNMENT, 1);
283 glTexImage2D(GL_TEXTURE_2D, 0, 4, width, height,
284 0, GL_BGRA, GL_UNSIGNED_BYTE, cairo_image_surface_get_data(surface));
285 glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
286 glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
289 glDisable(GL_LIGHTING);
290 glDisable(GL_COLOR_MATERIAL);
291 glDisable(GL_DEPTH_TEST);
292 glEnable(GL_TEXTURE_2D);
293 glBindTexture(GL_TEXTURE_2D, tex);
294 glDisable(GL_CULL_FACE);
295 glTranslatef(alloc.width - width, alloc.height - height, 0);
297 glTexCoord2f(1, 0); glVertex3f(width, 0 , 0); // 0 - 3 0
298 glTexCoord2f(1, 1); glVertex3f(width, height, 0); // 1 - | |
299 glTexCoord2f(0, 1); glVertex3f(0 , height, 0); // 2 - | |
300 glTexCoord2f(0, 0); glVertex3f(0 , 0 , 0); // 3 - 2----1
308 * grits_plugin_env_new:
309 * @viewer: the #GritsViewer to use for drawing
310 * @prefs: the #GritsPrefs for storing configurations
312 * Create a new instance of the environment plugin.
314 * Returns: the new #GritsPluginEnv
316 GritsPluginEnv *grits_plugin_env_new(GritsViewer *viewer, GritsPrefs *prefs)
318 g_debug("GritsPluginEnv: new");
319 GritsPluginEnv *env = g_object_new(GRITS_TYPE_PLUGIN_ENV, NULL);
320 env->viewer = g_object_ref(viewer);
321 env->prefs = g_object_ref(prefs);
324 GritsCallback *sky = grits_callback_new(sky_expose, env);
325 grits_viewer_add(viewer, GRITS_OBJECT(sky), GRITS_LEVEL_BACKGROUND, FALSE);
326 env->refs = g_list_prepend(env->refs, sky);
329 GritsCallback *compass = grits_callback_new(compass_expose, env);
330 grits_viewer_add(viewer, GRITS_OBJECT(compass), GRITS_LEVEL_HUD, TRUE);
331 g_signal_connect(compass, "clicked", G_CALLBACK(compass_click), viewer);
332 grits_object_set_cursor(GRITS_OBJECT(compass), GDK_CROSS);
333 env->refs = g_list_prepend(env->refs, compass);
336 //GritsCallback *info = grits_callback_new(info_expose, env);
337 //grits_viewer_add(viewer, GRITS_OBJECT(info), GRITS_LEVEL_HUD, FALSE);
338 //env->refs = g_list_prepend(env->refs, info);
342 //GritsTile *background = grits_tile_new(NULL, NORTH, SOUTH, EAST, WEST);
343 //glGenTextures(1, &env->tex);
344 //background->data = &env->tex;
345 //grits_viewer_add(viewer, GRITS_OBJECT(background), GRITS_LEVEL_BACKGROUND, FALSE);
346 //env->refs = g_list_prepend(env->refs, background);
356 static void grits_plugin_env_plugin_init(GritsPluginInterface *iface);
357 G_DEFINE_TYPE_WITH_CODE(GritsPluginEnv, grits_plugin_env, G_TYPE_OBJECT,
358 G_IMPLEMENT_INTERFACE(GRITS_TYPE_PLUGIN,
359 grits_plugin_env_plugin_init));
360 static void grits_plugin_env_plugin_init(GritsPluginInterface *iface)
362 g_debug("GritsPluginEnv: plugin_init");
363 /* Add methods to the interface */
365 /* Class/Object init */
366 static void grits_plugin_env_init(GritsPluginEnv *env)
368 g_debug("GritsPluginEnv: init");
371 static void grits_plugin_env_dispose(GObject *gobject)
373 g_debug("GritsPluginEnv: dispose");
374 GritsPluginEnv *env = GRITS_PLUGIN_ENV(gobject);
375 /* Drop references */
377 for (GList *cur = env->refs; cur; cur = cur->next)
378 grits_viewer_remove(env->viewer, cur->data);
379 g_list_free_full(env->refs, g_object_unref);
380 g_object_unref(env->viewer);
381 g_object_unref(env->prefs);
382 glDeleteTextures(1, &env->tex);
385 G_OBJECT_CLASS(grits_plugin_env_parent_class)->dispose(gobject);
387 static void grits_plugin_env_class_init(GritsPluginEnvClass *klass)
389 g_debug("GritsPluginEnv: class_init");
390 GObjectClass *gobject_class = (GObjectClass*)klass;
391 gobject_class->dispose = grits_plugin_env_dispose;