/* * Copyright (C) 2009-2010 Andy Spencer * * This program 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, either version 3 of the License, or * (at your option) any later version. * * This program 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 . */ /** * SECTION:env * @short_description: Environment plugin * * #GritsPluginEnv provides environmental information such as sky images. It can * also paint a blank overlay on the surface so that other plugins can draw * transparent overlays nicely. */ #include #include #include #include #include "env.h" /*********** * Helpers * ***********/ static void expose(GritsCallback *callback, GritsOpenGL *opengl, gpointer _env) { GritsPluginEnv *env = GRITS_PLUGIN_ENV(_env); g_debug("GritsPluginEnv: expose"); gdouble lat, lon, elev; grits_viewer_get_location(env->viewer, &lat, &lon, &elev); /* Misc */ gdouble rg = MAX(0, 1-(elev/40000)); gdouble blue = MAX(0, 1-(elev/100000)); glClearColor(MIN(0.4,rg), MIN(0.4,rg), MIN(1,blue), 1.0f); glClear(GL_COLOR_BUFFER_BIT); /* Attempt to render an atmosphere */ glEnable(GL_COLOR_MATERIAL); glDisable(GL_CULL_FACE); glDisable(GL_LIGHTING); glMatrixMode(GL_MODELVIEW); glBlendFunc(GL_SRC_ALPHA, GL_DST_ALPHA); grits_viewer_center_position(env->viewer, lat, lon, -EARTH_R); gdouble ds = EARTH_R+elev; // distance to self gdouble da = EARTH_R+300000; // distance to top of atmosphere gdouble dg = EARTH_R-100000; // distance to top of atmosphere gdouble ang = acos(EARTH_R/ds); // angle to horizon ang = MAX(ang,0.1); gdouble ar = sin(ang)*da; // top of quad fan "atomosphere"j gdouble az = cos(ang)*da; // gdouble gr = sin(ang)*dg; // bottom of quad fan "ground" gdouble gz = cos(ang)*dg; // glBegin(GL_QUAD_STRIP); for (gdouble i = 0; i <= 2*G_PI; i += G_PI/30) { glColor4f(0.3, 0.3, 1.0, 1.0); glVertex3f(gr*sin(i), gr*cos(i), gz); glColor4f(0.3, 0.3, 1.0, 0.0); glVertex3f(ar*sin(i), ar*cos(i), az); } glEnd(); } /*********** * Methods * ***********/ /** * grits_plugin_env_new: * @viewer: the #GritsViewer to use for drawing * @prefs: the #GritsPrefs for storing configurations * * Create a new instance of the environment plugin. * * Returns: the new #GritsPluginEnv */ GritsPluginEnv *grits_plugin_env_new(GritsViewer *viewer, GritsPrefs *prefs) { g_debug("GritsPluginEnv: new"); GritsPluginEnv *env = g_object_new(GRITS_TYPE_PLUGIN_ENV, NULL); env->viewer = g_object_ref(viewer); /* Create objects */ GritsCallback *callback = grits_callback_new(expose, env); GritsTile *background = grits_tile_new(NULL, NORTH, SOUTH, EAST, WEST); glGenTextures(1, &env->tex); background->data = &env->tex; /* Add renderers */ gpointer ref1, ref2; ref1 = grits_viewer_add(viewer, GRITS_OBJECT(callback), GRITS_LEVEL_BACKGROUND, FALSE); ref2 = grits_viewer_add(viewer, GRITS_OBJECT(background), GRITS_LEVEL_BACKGROUND, FALSE); env->refs = g_list_prepend(env->refs, ref1); env->refs = g_list_prepend(env->refs, ref2); return env; } /**************** * GObject code * ****************/ /* Plugin init */ static void grits_plugin_env_plugin_init(GritsPluginInterface *iface); G_DEFINE_TYPE_WITH_CODE(GritsPluginEnv, grits_plugin_env, G_TYPE_OBJECT, G_IMPLEMENT_INTERFACE(GRITS_TYPE_PLUGIN, grits_plugin_env_plugin_init)); static void grits_plugin_env_plugin_init(GritsPluginInterface *iface) { g_debug("GritsPluginEnv: plugin_init"); /* Add methods to the interface */ } /* Class/Object init */ static void grits_plugin_env_init(GritsPluginEnv *env) { g_debug("GritsPluginEnv: init"); /* Set defaults */ } static void grits_plugin_env_dispose(GObject *gobject) { g_debug("GritsPluginEnv: dispose"); GritsPluginEnv *env = GRITS_PLUGIN_ENV(gobject); /* Drop references */ if (env->viewer) { for (GList *cur = env->refs; cur; cur = cur->next) grits_viewer_remove(env->viewer, cur->data); g_list_free(env->refs); g_object_unref(env->viewer); glDeleteTextures(1, &env->tex); env->viewer = NULL; } G_OBJECT_CLASS(grits_plugin_env_parent_class)->dispose(gobject); } static void grits_plugin_env_class_init(GritsPluginEnvClass *klass) { g_debug("GritsPluginEnv: class_init"); GObjectClass *gobject_class = (GObjectClass*)klass; gobject_class->dispose = grits_plugin_env_dispose; }