OE代码样例
/* -*-c++-*- */ /* osgEarth - Dynamic map generation toolkit for OpenSceneGraph * Copyright 2016 Pelican Mapping * http://osgearth.org * * osgEarth is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS * IN THE SOFTWARE. * * You should have received a copy of the GNU Lesser General Public License * along with this program. If not, see <http://www.gnu.org/licenses/> */ #include <osgEarth/MapNode> #include <osgEarthUtil/EarthManipulator> #include <osgEarthUtil/ExampleResources> #include <osgEarthAnnotation/ImageOverlay> #include <osgEarthAnnotation/CircleNode> #include <osgEarthAnnotation/RectangleNode> #include <osgEarthAnnotation/EllipseNode> #include <osgEarthAnnotation/PlaceNode> #include <osgEarthAnnotation/LabelNode> #include <osgEarthAnnotation/LocalGeometryNode> #include <osgEarthAnnotation/FeatureNode> #include <osgEarthAnnotation/ModelNode> #include <osgEarthAnnotation/AnnotationEditing> #include <osgEarthAnnotation/ImageOverlayEditor> #include <osgEarthSymbology/GeometryFactory> #include <osgViewer/Viewer> using namespace osgEarth; using namespace osgEarth::Annotation; using namespace osgEarth::Features; using namespace osgEarth::Util; //------------------------------------------------------------------ int usage(char** argv) { OE_WARN << "Usage: " << argv[0] << " <earthfile>" << std::endl; return -1; } //------------------------------------------------------------------ int main(int argc, char** argv) { osg::Group* root = new osg::Group(); // try to load an earth file. osg::ArgumentParser arguments(&argc, argv); osgViewer::Viewer viewer(arguments); viewer.setCameraManipulator(new EarthManipulator()); // load an earth file and parse demo arguments osg::Node* node = MapNodeHelper().load(arguments, &viewer); if (!node) return usage(argv); root->addChild(node); // find the map node that we loaded. MapNode* mapNode = MapNode::findMapNode(node); if (!mapNode) return usage(argv); // Group to hold all our annotation elements. osg::Group* annoGroup = new osg::Group(); root->addChild(annoGroup); // Make a group for labels osg::Group* labelGroup = new osg::Group(); annoGroup->addChild(labelGroup); osg::Group* editGroup = new osg::Group(); root->addChild(editGroup); // Style our labels: Style labelStyle; labelStyle.getOrCreate<TextSymbol>()->alignment() = TextSymbol::ALIGN_CENTER_CENTER; labelStyle.getOrCreate<TextSymbol>()->fill()->color() = Color::Yellow; // A lat/long SRS for specifying points. const SpatialReference* geoSRS = mapNode->getMapSRS()->getGeographicSRS(); //-------------------------------------------------------------------- // A series of place nodes (an icon with a text label) { Style pm; pm.getOrCreate<IconSymbol>()->url()->setLiteral("E:/temp/OE/placemark32.png"); pm.getOrCreate<IconSymbol>()->declutter() = true; pm.getOrCreate<TextSymbol>()->halo() = Color("#5f5f5f"); // bunch of pins: labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -74.00, 40.71), "New York", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -77.04, 38.85), "Washington, DC", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -118.40, 33.93), "Los Angeles", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -71.03, 42.37), "Boston", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -157.93, 21.35), "Honolulu", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, 139.75, 35.68), "Tokyo", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -90.25, 29.98), "New Orleans", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -80.28, 25.82), "Miami", pm)); labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -117.17, 32.72), "San Diego", pm)); // test with an LOD: osg::LOD* lod = new osg::LOD(); lod->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, 14.68, 50.0), "Prague", pm), 0.0, 2e6); labelGroup->addChild(lod); // absolute altitude: labelGroup->addChild(new PlaceNode(mapNode, GeoPoint(geoSRS, -87.65, 41.90, 1000, ALTMODE_ABSOLUTE), "Chicago", pm)); } //-------------------------------------------------------------------- // a box that follows lines of latitude (rhumb line interpolation, the default) { Geometry* geom = new Polygon(); geom->push_back(osg::Vec3d(0, 40, 0)); geom->push_back(osg::Vec3d(-60, 40, 0)); geom->push_back(osg::Vec3d(-60, 60, 0)); geom->push_back(osg::Vec3d(0, 60, 0)); Feature* feature = new Feature(geom, geoSRS); feature->geoInterp() = GEOINTERP_RHUMB_LINE; Style geomStyle; geomStyle.getOrCreate<LineSymbol>()->stroke()->color() = Color::Cyan; geomStyle.getOrCreate<LineSymbol>()->stroke()->width() = 5.0f; geomStyle.getOrCreate<LineSymbol>()->tessellationSize() = 75000; geomStyle.getOrCreate<AltitudeSymbol>()->clamping() = AltitudeSymbol::CLAMP_TO_TERRAIN; geomStyle.getOrCreate<AltitudeSymbol>()->technique() = AltitudeSymbol::TECHNIQUE_GPU; FeatureNode* fnode = new FeatureNode(mapNode, feature, geomStyle); annoGroup->addChild(fnode); labelGroup->addChild(new LabelNode(mapNode, GeoPoint(geoSRS, -30, 50), "Rhumb line polygon", labelStyle)); } //-------------------------------------------------------------------- // another rhumb box that crosses the antimeridian { Geometry* geom = new Polygon(); geom->push_back(-160., -30.); geom->push_back(150., -20.); geom->push_back(160., -45.); geom->push_back(-150., -40.); Style geomStyle; Feature* feature = new Feature(geom, geoSRS); feature->geoInterp() = GEOINTERP_RHUMB_LINE; geomStyle.getOrCreate<LineSymbol>()->stroke()->color() = Color::Lime; geomStyle.getOrCreate<LineSymbol>()->stroke()->width() = 3.0f; geomStyle.getOrCreate<LineSymbol>()->tessellationSize() = 75000; geomStyle.getOrCreate<AltitudeSymbol>()->clamping() = AltitudeSymbol::CLAMP_TO_TERRAIN; geomStyle.getOrCreate<AltitudeSymbol>()->technique() = AltitudeSymbol::TECHNIQUE_GPU; FeatureNode* gnode = new FeatureNode(mapNode, feature, geomStyle); annoGroup->addChild(gnode); labelGroup->addChild(new LabelNode(mapNode, GeoPoint(geoSRS, -175, -35), "Antimeridian polygon", labelStyle)); } //-------------------------------------------------------------------- // A path using great-circle interpolation. // Keep a pointer to it so we can modify it later on. FeatureNode* pathNode = 0; { Geometry* path = new LineString(); path->push_back(osg::Vec3d(-74, 40.714, 0)); // New York path->push_back(osg::Vec3d(139.75, 35.68, 0)); // Tokyo Feature* pathFeature = new Feature(path, geoSRS); pathFeature->geoInterp() = GEOINTERP_GREAT_CIRCLE; Style pathStyle; pathStyle.getOrCreate<LineSymbol>()->stroke()->color() = Color::White; pathStyle.getOrCreate<LineSymbol>()->stroke()->width() = 1.0f; pathStyle.getOrCreate<LineSymbol>()->tessellationSize() = 75000; pathStyle.getOrCreate<PointSymbol>()->size() = 5; pathStyle.getOrCreate<PointSymbol>()->fill()->color() = Color::Red; pathStyle.getOrCreate<AltitudeSymbol>()->clamping() = AltitudeSymbol::CLAMP_TO_TERRAIN; pathStyle.getOrCreate<AltitudeSymbol>()->technique() = AltitudeSymbol::TECHNIQUE_GPU; //OE_INFO << "Path extent = " << pathFeature->getExtent().toString() << std::endl; pathNode = new FeatureNode(mapNode, pathFeature, pathStyle); annoGroup->addChild(pathNode); labelGroup->addChild(new LabelNode(mapNode, GeoPoint(geoSRS, -170, 61.2), "Great circle path", labelStyle)); } //-------------------------------------------------------------------- // Two circle segments around New Orleans. { Style circleStyle; circleStyle.getOrCreate<PolygonSymbol>()->fill()->color() = Color(Color::Cyan, 0.5); circleStyle.getOrCreate<AltitudeSymbol>()->clamping() = AltitudeSymbol::CLAMP_TO_TERRAIN; circleStyle.getOrCreate<AltitudeSymbol>()->technique() = AltitudeSymbol::TECHNIQUE_DRAPE; CircleNode* circle = new CircleNode( mapNode, GeoPoint(geoSRS, -90.25, 29.98, 1000., ALTMODE_RELATIVE), Distance(300, Units::KILOMETERS), circleStyle, Angle(-45.0, Units::DEGREES), Angle(45.0, Units::DEGREES), true); annoGroup->addChild(circle); editGroup->addChild(new CircleNodeEditor(circle)); } { Style circleStyle; circleStyle.getOrCreate<PolygonSymbol>()->fill()->color() = Color(Color::Red, 0.5); circleStyle.getOrCreate<AltitudeSymbol>()->clamping() = AltitudeSymbol::CLAMP_TO_TERRAIN; circleStyle.getOrCreate<AltitudeSymbol>()->technique() = AltitudeSymbol::TECHNIQUE_DRAPE; CircleNode* circle = new CircleNode( mapNode, GeoPoint(geoSRS, -90.25, 29.98, 1000., ALTMODE_RELATIVE), Distance(300, Units::KILOMETERS), circleStyle, Angle(45.0, Units::DEGREES), Angle(360.0 - 45.0, Units::DEGREES), true); annoGroup->addChild(circle); editGroup->addChild(new CircleNodeEditor(circle)); } //-------------------------------------------------------------------- // An extruded ellipse around Miami. { Style ellipseStyle; ellipseStyle.getOrCreate<PolygonSymbol>()->fill()->color() = Color(Color::Orange, 0.75); ellipseStyle.getOrCreate<ExtrusionSymbol>()->height() = 250000.0; // meters MSL EllipseNode* ellipse = new EllipseNode( mapNode, GeoPoint(geoSRS, -80.28, 25.82, 0.0, ALTMODE_RELATIVE), Distance(250, Units::MILES), Distance(100, Units::MILES), Angle(0, Units::DEGREES), ellipseStyle, Angle(45.0, Units::DEGREES), Angle(360.0 - 45.0, Units::DEGREES), true); annoGroup->addChild(ellipse); editGroup->addChild(new EllipseNodeEditor(ellipse)); } { Style ellipseStyle; ellipseStyle.getOrCreate<PolygonSymbol>()->fill()->color() = Color(Color::Blue, 0.75); ellipseStyle.getOrCreate<ExtrusionSymbol>()->height() = 250000.0; // meters MSL EllipseNode* ellipse = new EllipseNode( mapNode, GeoPoint(geoSRS, -80.28, 25.82, 0.0, ALTMODE_RELATIVE), Distance(250, Units::MILES), Distance(100, Units::MILES), Angle(0, Units::DEGREES), ellipseStyle, Angle(-40.0, Units::DEGREES), Angle(40.0, Units::DEGREES), true); annoGroup->addChild(ellipse); editGroup->addChild(new EllipseNodeEditor(ellipse)); } //-------------------------------------------------------------------- { // A rectangle around San Diego Style rectStyle; rectStyle.getOrCreate<PolygonSymbol>()->fill()->color() = Color(Color::Green, 0.5); rectStyle.getOrCreate<AltitudeSymbol>()->clamping() = AltitudeSymbol::CLAMP_TO_TERRAIN; rectStyle.getOrCreate<AltitudeSymbol>()->technique() = AltitudeSymbol::TECHNIQUE_DRAPE; RectangleNode* rect = new RectangleNode( mapNode, GeoPoint(geoSRS, -117.172, 32.721), Distance(300, Units::KILOMETERS), Distance(600, Units::KILOMETERS), rectStyle); annoGroup->addChild(rect); editGroup->addChild(new RectangleNodeEditor(rect)); } //-------------------------------------------------------------------- // An extruded polygon roughly the shape of Utah. Here we demonstrate the // FeatureNode, where you create a geographic geometry and use it as an // annotation. { Geometry* utah = new Polygon(); utah->push_back(-114.052, 37.0); utah->push_back(-109.054, 37.0); utah->push_back(-109.054, 41.0); utah->push_back(-111.040, 41.0); utah->push_back(-111.080, 42.059); utah->push_back(-114.080, 42.024); Style utahStyle; utahStyle.getOrCreate<ExtrusionSymbol>()->height() = 250000.0; // meters MSL utahStyle.getOrCreate<PolygonSymbol>()->fill()->color() = Color(Color::White, 0.8); Feature* utahFeature = new Feature(utah, geoSRS); FeatureNode* featureNode = new FeatureNode(mapNode, utahFeature, utahStyle); annoGroup->addChild(featureNode); } //-------------------------------------------------------------------- // an image overlay. { ImageOverlay* imageOverlay = 0L; osg::Image* image = osgDB::readImageFile("E:/temp/OE/placemark32.png/USFLAG.TGA"); //osg::Image* image = osgDB::readImageFile("../data/USFLAG.TGA"); if (image) { imageOverlay = new ImageOverlay(mapNode, image); imageOverlay->setBounds(Bounds(-100.0, 35.0, -90.0, 40.0)); annoGroup->addChild(imageOverlay); editGroup->addChild(new ImageOverlayEditor(imageOverlay)); } } //-------------------------------------------------------------------- // a model node with auto scaling. { Style style; style.getOrCreate<ModelSymbol>()->autoScale() = true; style.getOrCreate<ModelSymbol>()->url()->setLiteral("E:/temp/OE/red_flag.osg.50.scale"); ModelNode* modelNode = new ModelNode(mapNode, style); modelNode->setPosition(GeoPoint(geoSRS, -100, 52)); annoGroup->addChild(modelNode); } //-------------------------------------------------------------------- // initialize the viewer: viewer.setSceneData(root); viewer.getCamera()->setSmallFeatureCullingPixelSize(-1.0f); osg::ref_ptr<osg::GraphicsContext::Traits> traits = new osg::GraphicsContext::Traits; traits->x = 200; traits->y = 200; traits->width = 800; traits->height = 680; traits->windowDecoration = true; traits->doubleBuffer = true; traits->sharedContext = 0; osg::ref_ptr<osg::GraphicsContext> gc = osg::GraphicsContext::createGraphicsContext(traits.get()); osg::ref_ptr<osg::Camera> camera = new osg::Camera; camera->setGraphicsContext(gc.get()); camera->setViewport(new osg::Viewport(0, 0, traits->width, traits->height)); GLenum buffer = traits->doubleBuffer ? GL_BACK : GL_FRONT; camera->setDrawBuffer(buffer); camera->setReadBuffer(buffer); viewer.addSlave(camera); return viewer.run(); }
earth文件样例为 : annotation.earth
<!-- osgEarth Sample - Annotations --> <map name="readymap.org" type="geocentric" version="2"> <image name="ReadyMap.org - Imagery" driver="tms"> <url>http://readymap.org/readymap/tiles/1.0.0/7/</url> </image> <elevation name="ReadyMap.org - Elevation" driver="tms"> <url>http://readymap.org/readymap/tiles/1.0.0/116/</url> </elevation> <viewpoints> <viewpoint name="Annotation Samples" lat="33" long="-118" range="500000" heading="35.27" pitch="-35" /> </viewpoints> <!--annotations name="Annotations Group A"> <label text="Label"> <position lat="34" long="-120" /> <style type="text/css"> text-align: center_center; text-size: 20; text-declutter: true; text-halo: #777; text-bbox-fill: #00FF0033; text-bbox-margin: 3; text-bbox-border: #FFFFFFFF; text-bbox-border-width: 1; </style> </label> <label text="Label with offsets"> <position lat="16" long="-120" /> <style type="text/css"> text-align: center_center; text-size: 20; text-declutter: true; text-halo: #777; text-offset-x: 28; text-offset-y: -100; text-bbox-fill: #00FF0033; text-bbox-margin: 3; text-bbox-border: #FFFFFFFF; text-bbox-border-width: 1; </style> </label> <place text="Place"> <position lat="35" long="-110"/> <icon>../data/placemark32.png</icon> <style type="text/css"> text-declutter: true; text-halo: #777; </style> </place> <place text="Extruded Place"> <position lat="35" long="-105" alt="5000"/> <icon>../data/placemark32.png</icon> <style type="text/css"> text-declutter: true; text-halo: #777; </style> </place> <circle name="draped circle"> <position lat="34.051" long="-117.974"/> <radius>50km</radius> <style type="text/css"> fill: #ffff0080; stroke: #ffffff; stroke-width: 2px; altitude-clamping: terrain-drape; </style> </circle> <label text="Draped circle" lat="34.051" long="-117.974"/> <circle name="scene-clamped circle"> <position lat="22.074" long="-159.606"/> <radius>1.2km</radius> <style type="text/css"> stroke: #ffffff; stroke-width: 2px; altitude-clamping: terrain-scene; altitude-binding: vertex; render-depth-offset-auto: true; </style> </circle> <label text="scene-clamped circle" lat="22.074" long="-159.606"/> <ellipse name="ellipse relative"> <position lat="40" long="-100" hat="5000"/> <radius_major>50km</radius_major> <radius_minor>40km</radius_minor> <style type="text/css"> fill: #ff00ff7f; stroke: #ffffff; </style> </ellipse> <label text="HAT Ellipse" lat="40" long="-100.0"/> </annotations> <annotations name="Annotations Group B"> <ellipse name="ellipse extruded"> <position lat="32" long="-100.0"/> <radius_major>50km</radius_major> <radius_minor>20km</radius_minor> <style type="text/css"> fill: #ff7f007f; stroke: #ff0000ff; extrusion-height: 5000; </style> </ellipse> <label text="Extruded Ellipse" lat="32" long="-100.0"/> <feature name="Extruded Line"> <srs>wgs84</srs> <geometry> LINESTRING(-80.37 34.039, -80.09 33.96, -79.75 34, -79.43 33.37, -79.48 32.88) </geometry> <style type="text/css"> fill: #ff00ff7f; stroke: #ffff00; stroke-width: 3; stroke-crease-angle: 45.0; extrusion-height: 30000; render-lighting: true; </style> </feature> <label text="Extruded Line" lat="32" long="-80"/> <feature name="GPU-Clamped Line"> <srs>wgs84</srs> <geometry> LINESTRING(-110 47, -110 43, -120 43, -120 42) </geometry> <style type="text/css"> stroke: #ff3000; stroke-width: 3; stroke-tessellation-size: 1km; altitude-clamping: terrain-gpu; </style> </feature> <label text="GPU-Clamped Line" lat="44" long="-115"> <style type="text/css"> text-align: center_center; text-geographic-course: 45.0; text-bbox-fill: #FFFF0033; text-bbox-margin: 8; text-bbox-border: #FFFF00; text-bbox-border-width: 0.5; </style> </label> <feature name="Draped Polygon"> <srs>wgs84</srs> <geometry> POLYGON((-100 47, -100 49, -95 48, -96 45, -98 42)) </geometry> <style type="text/css"> fill: #ffff007f; stroke: #ffffff; stroke-width: 2px; altitude-clamping: terrain-drape; </style> </feature> <label text="Draped Polygon" lat="45" long="-98"/> <model name="Auto-Scaled Model"> <position lat="43" long="-100"/> <style> model: "../data/red_flag.osg.45.scale"; model-scale: auto; </style> </model> <label text="Auto-Scaled Model" lat="42.5" long="-100"/> <imageoverlay> <url>../data/fractal.png</url> <alpha>1.0</alpha> <geometry>POLYGON((-81 26, -80.5 26, -80.5 26.5, -81 26.5))</geometry> </imageoverlay> <label text="ImageOverlay" lat="26" long="-81"/> <local_geometry name="3D geometry"> <geometry> POLYGON((0 0 0, -25000 0 45000, 0 0 75000, 25000 0 45000)) </geometry> <position lat="33.4" long="-116.6"/> <style type="text/css"> fill: #00ff00; stroke: #ffff00; stroke-width: 2px; render-lighting: false; </style> <horizon_culling>true</horizon_culling> </local_geometry> <label text="3D Geometry" lat="33.4" long="-116.6"/> <feature name="Long Line"> <srs>wgs84</srs> <geometry> LINESTRING(10 0, 10 65) </geometry> <style type="text/css"> stroke: #ffff00; stroke-width: 3; stroke-tessellation-size: 1km; altitude-clamping: terrain; altitude-technique: gpu; render-lighting: false; </style> </feature> <label text="Tessellated line" lat="35" long="10"> <style type="text/css"> text-align: center_bottom; text-geographic-course: 0; </style> </label> </annotations--> </map>