यदि पहले से ही एक फीचरलेयर मौजूद है, तो कैसे सत्यापित करें?


9

मैं वर्तमान में एक ऐसी परियोजना पर काम कर रहा हूं, जब उपयोगकर्ता हमारे सिस्टम में अपने मैप (mxd) को लोड करते हैं, हम उनके लिए कई कस्टम करतब दिखाते हैं। हालाँकि, मेरी समस्या यह है कि मुझे पता नहीं है कि अगर मैंने उन परतों को पहले से ही बनाया है, तो जांच कैसे करें (जैसे कि उपयोगकर्ता mxd को लोड करता है, परतों का निर्माण, सहेजना, mxd को फिर से लोड करना, यह सत्यापित करना चाहिए कि क्या परतें पहले से मौजूद हैं)।

क्या ArcEngine10 में एक FeatuerLayerClass के लिए एक अद्वितीय आईडी है, इसमें OIDName और FeatureLayerClass.FeatureClass में ObjectClassID हैं, लेकिन वे काम नहीं करते हैं (ObjectClassId असाइन नहीं कर सकते हैं, और OIDName के लिए UniqueId का उपयोग करना चाहते हैं)?

मैंने इस तरह से फीचरलेयरस्केल व्यावसायिक वस्तु के रूप में अपनी परत बनाई।

कोड:

    /// <summary>
    ///     Unique Route LayerId
    /// </summary>
    public static Guid RouteFeatureLayerId
    {
        get { return Guid.Parse("ba25a332-0e48-4ce5-a4c5-38dc36c0700c"); }
    }

    /// <summary>
    ///     Feature class that stores info on the routes
    /// </summary>
    public FeatureLayerClass RouteFeatureLayer
    {
        get
        {
            if (_routeFeatureClass == null)
            {
                IPropertySet property = new PropertySetClass();
                property.SetProperty("Id", RouteFeatureLayerId);

                _routeFeatureClass = new FeatureLayerClass();
                _routeFeatureClass.FeatureClass = CreateFeatureClass(Workspace, null, ShapeType.Polylines.ToString(), CreateFields(ShapeType.Polylines, FeatureLayerType.Routes), null, null, "");
                _routeFeatureClass.Name = "Routes";
                _routeFeatureClass.Visible = true;
                _routeFeatureClass.Cached = true;
                _routeFeatureClass.AddExtension(property);
                CustomLayers.Add(_routeFeatureClass); 

            }

            return _routeFeatureClass;
        }
        set
        {
            _routeFeatureClass = value;
        }
    }

कार्यक्षेत्र बनाना

    /// <summary>
    ///     Create a workspace for the shapefile or geodatabase
    /// </summary>
private IWorkspace CreateWorkspace(string workspaceType, string workspaceDirectory)
{
    Type factoryType = null;
    IWorkspaceFactory workspaceFactory = null;

    switch (workspaceType)
    {
        case "Shapefile":
            // Instantiate a Shapefile workspace factory
            factoryType = Type.GetTypeFromProgID("esriDataSourcesFile.ShapefileWorkspaceFactory");
            break;
        case "PersonalGeodatabase":
            // Instantiate an Access workspace factory
            factoryType = Type.GetTypeFromProgID("esriDataSourcesGDB.AccessWorkspaceFactory");
            break;
        case "FileGeodatabase":
            // Instantiate a file geodatabase workspace factory
            factoryType = Type.GetTypeFromProgID("esriDataSourcesGDB.FileGDBWorkspaceFactory");
            break;
    }

    workspaceFactory = (IWorkspaceFactory)Activator.CreateInstance(factoryType);

    //Create a directory hierarchy to seperate out datasets created for Points, Polylines, and Polygons
    Directory.CreateDirectory(workspaceDirectory);

    IWorkspaceName workspaceName = workspaceFactory.Create(workspaceDirectory + "\\", workspaceType, null, 0);
    IName Name = (IName)workspaceName;
    IWorkspace workspace = (IWorkspace)(Name.Open());
    return workspace;

}

FeatureClass बनाना

        /// <summary>
        ///     Helper to create a Feature Class.
        /// </summary>
        private IFeatureClass CreateFeatureClass(IWorkspace workspace, IFeatureDataset featureDataset, string featureClassName, IFields fields, ESRI.ArcGIS.esriSystem.UID CLSID, ESRI.ArcGIS.esriSystem.UID CLSEXT, string configKeyword)
        {
            IFeatureClass featureClass = null;
            IFeatureWorkspace featureWorkspace = (IFeatureWorkspace)workspace; // Explicit Cast
            string shapeFieldName = String.Empty;

            try
            {
                if (featureClassName == "")
                {
                    return null; // name was not passed in
                }
                //else if (((IWorkspace2)workspace).get_NameExists(esriDatasetType.esriDTFeatureClass, featureClassName))
                //{
                //    featureClass = featureWorkspace.OpenFeatureClass(featureClassName); // feature class with that name already exists
                //    return featureClass;
                //}

                // assign the class id value if not assigned
                if (CLSID == null)
                {
                    CLSID = new ESRI.ArcGIS.esriSystem.UIDClass();
                    CLSID.Value = "esriGeoDatabase.Feature";
                }

                // locate the shape field
                for (Int32 j = 0; j < fields.FieldCount; j++)
                {
                    if (fields.get_Field(j).Type == esriFieldType.esriFieldTypeGeometry)
                    {
                        shapeFieldName = fields.get_Field(j).Name;
                    }
                }

                // finally create and return the feature class
                if (featureDataset == null)
                {
                    // if no feature dataset passed in, create at the workspace level
                    featureClass = featureWorkspace.CreateFeatureClass(featureClassName, fields, CLSID, CLSEXT, esriFeatureType.esriFTSimple, shapeFieldName, configKeyword);
                }
                else
                {
                    featureClass = featureDataset.CreateFeatureClass(featureClassName, fields, CLSID, CLSEXT, esriFeatureType.esriFTSimple, shapeFieldName, configKeyword);
                }
            }
            catch (Exception ex)
            {
                Debug.Assert(false, ex.ToString());
                Logger.Log.Debug(ex);
            }
            return featureClass;

        }

परत पाने के लिए कोड

            /// <summary>
            ///     Finds the layer
            /// </summary>
            /// <returns>the subcatchment layer</returns>
            private IGeoFeatureLayer GetLayer(FeatureLayerClass featureLayer)
            {
                IGeoFeatureLayer layer = null;
                ILayerExtensions layerExtension;

                for (int x = 0; x < MapControl.LayerCount; x++)
                {
                    layerExtension = ((ILayerExtensions)MapControl.get_Layer(x));

                    if (featureLayer.ExtensionCount > 0 && layerExtension.ExtensionCount > 0 &&
                        layerExtension.get_Extension(0) is PropertySetClass &&
                        featureLayer.get_Extension(0) is PropertySetClass &&
                        ((PropertySetClass)layerExtension.get_Extension(0)).GetProperty("Id") == ((PropertySetClass)featureLayer.get_Extension(0)).GetProperty("Id"))
                    {
                        layer = MapControl.get_Layer(x) as IGeoFeatureLayer;
                        break;
                    }
                }

                return layer;
            }

धन्यवाद और सादर, केविन

जवाबों:


7

फ़ीचर क्लासेस और ऑब्जेक्ट क्लासेस में उनकी आईडी होती है, जो कि एक सिंगल जियोडेटाबेस के भीतर अद्वितीय होती हैं। यह अक्सर आपके समान अधिकांश परिदृश्यों को संतुष्ट करता है।

यदि आप विशिष्ट रूप से इसकी विशेषता वर्ग के आधार पर एक परत की पहचान नहीं कर सकते हैं, तो आप परत के साथ मनमाना डेटा संग्रहीत करने के लिए परत एक्सटेंशन का लाभ उठा सकते हैं।

एक परत एक्सटेंशन को ILayerExtensions इंटरफ़ेस के माध्यम से एक परत में जोड़ा जा सकता है। अब, परत एक्सटेंशन के लिए कोई सामान्य इंटरफ़ेस नहीं है, लेकिन वे आमतौर पर IPersistStream के माध्यम से कुछ दृढ़ता को लागू करते हैं । आपका लेयर एक्सटेंशन कुछ खास नहीं करेगा लेकिन कुछ डेटा को स्टोर करेगा जिसके द्वारा आप विशिष्ट रूप से अपनी अतिरिक्त लेयर की पहचान करेंगे।

तो आपका कार्य निम्नानुसार होगा:

  • एक COM क्लास बनाएं जो आपके झंडे (या किसी प्रकार की उत्पन्न आईडी) को स्टोर करे। इस वर्ग के लिए IPersistStream लागू करें। संपादित करें: आप आसानी से अपनी खुद की क्लास बनाने के बजाय, लेयर एक्सटेंशन ऑब्जेक्ट के रूप में एक प्रॉपर्टीसेट का उपयोग कर सकते हैं ।
  • जब आप एक लेयर जोड़ रहे हैं, तो मैप में सभी लेयर्स के माध्यम से लूप करें और जांच लें कि उनमें से किसी में आपका लेयर एक्सटेंशन असाइन किया गया है, जिसमें संग्रहीत डेटा की अपेक्षा है।
  • यदि ऐसा है, तो परत को न जोड़ें क्योंकि यह पहले से मौजूद है।
  • यदि नहीं, तो परत जोड़ें, और ILayerExtensions के माध्यम से अपनी परत विस्तार की एक आवृत्ति जोड़ें।

मैं एक बहुत ही इसी तरह की समस्या थी और परतों का विस्तार सबसे अच्छा फिट निकला।

EDIT: नीचे मैं हेल्पर स्टैटिक क्लास के लिए कुछ कोड पोस्ट करता हूं जो आपको लेयर एक्सटेंशन (.NET 3.5 या उच्चतर आवश्यक) में संग्रहीत प्रॉपर्टीसेट के अंदर सेट किए गए गुणों के साथ जल्दी से काम करने की अनुमति देता है। यह एक्सटेंशन ऑब्जेक्ट तक पहुंचने और इसे बनाने का ख्याल रखता है यदि पहले से ही परत को नहीं सौंपा गया है। यह इस तरह प्रयोग किया जाता है:

        // 1) is a particular property ("MY.KEY") set on a layer?
        var isPropertySet = PropertySetLayerExtensionHelper.ExtensionPropertySetContainsKey(layer, "MY.KEY");

        // 2) set a property with a value on the layer:
        PropertySetLayerExtensionHelper.ExtensionPropertySetSetValueForKey(layer, "MY.KEY", "SomeValue");

        // 3) retrieve a value for the given key stored at some point before:
        var value = PropertySetLayerExtensionHelper.ExtensionPropertySetGetValueForKey(layer, "MY.KEY");

"SomeValue" के बजाय आप संभवतः किसी प्रकार के लेयर आइडेंटिफ़ायर को वहां उत्पन्न और संग्रहीत करेंगे।

यहां PropertySetLayerExtensionHelperकक्षा के लिए पूर्ण स्रोत कोड दिया गया है :

public static class PropertySetLayerExtensionHelper
{
    /// <summary>
    /// Returns whether the property set stored in the layer extensions contains a value for the given key.
    /// </summary>
    /// <param name="layer">The layer.</param>
    /// <param name="key">The key.</param>
    /// <returns>Whether the property set stored in the layer extensions contains a value for the given key.</returns>
    public static bool ExtensionPropertySetContainsKey(ILayer layer, string key)
    {
        if (layer == null) throw new ArgumentNullException("layer");
        if (key == null) throw new ArgumentNullException("key");

        var propertySet = GetPropertySetInLayerExtension(layer);
        return propertySet != null
            && propertySet.AsEnumerable().Any(pair => pair.Key.Equals(key, StringComparison.OrdinalIgnoreCase));
    }

    /// <summary>
    /// Returns the value for the given key from the property set stored in the layer extension or <b>null</b>
    /// if no such key is present.
    /// </summary>
    /// <param name="layer">The layer.</param>
    /// <param name="key">The key.</param>
    /// <returns>The value for the given key from the property set stored in the layer extension or <b>null</b>
    /// if no such key is present.</returns>
    public static object ExtensionPropertySetGetValueForKey(ILayer layer, string key)
    {
        if (layer == null) throw new ArgumentNullException("layer");
        if (key == null) throw new ArgumentNullException("key");

        var propertySet = GetPropertySetInLayerExtension(layer);
        if (propertySet == null) return null;

        return propertySet.AsEnumerable()
            .Where(p => p.Key.Equals(key, StringComparison.OrdinalIgnoreCase))
            .Select(p => p.Value)
            .FirstOrDefault();
    }

    /// <summary>
    /// Sets the value for the given key in the property set stored in a layer extension. If there is
    /// no property set among the layer's extensions, it is created and assigned to the layer.
    /// </summary>
    /// <param name="layer">The layer.</param>
    /// <param name="key">The key.</param>
    /// <param name="value">The value for the given key.</param>
    public static void ExtensionPropertySetSetValueForKey(ILayer layer, string key, object value)
    {
        if (layer == null) throw new ArgumentNullException("layer");
        if (key == null) throw new ArgumentNullException("key");

        var propertySet = GetOrCreatePropertySetInLayerExtension(layer);
        if (propertySet == null)
        {
            throw new InvalidOperationException("The given layer does not support layer extensions.");
        }

        propertySet.SetProperty(key, value);
    }

    /// <summary>
    /// Returns a property set from a layer extension.
    /// </summary>
    /// <param name="layer">The layer.</param>
    /// <returns>A property set from a layer extension.</returns>
    public static IPropertySet GetPropertySetInLayerExtension(ILayer layer)
    {
        if (layer == null) throw new ArgumentNullException("layer");

        var layerExtensions = layer as ILayerExtensions;
        if (layerExtensions == null)
        {
            return null;
        }

        var propertySetExtension = layerExtensions.AsEnumerable().OfType<IPropertySet>().FirstOrDefault();
        return propertySetExtension;
    }

    /// <summary>
    /// Returns a property set from a layer extension. If not set on the layer,
    /// the property set is created and assigned to it.
    /// </summary>
    /// <param name="layer">The layer.</param>
    /// <returns>A property set from a layer extension.</returns>
    public static IPropertySet GetOrCreatePropertySetInLayerExtension(ILayer layer)
    {
        if (layer == null) throw new ArgumentNullException("layer");

        var propertySet = GetPropertySetInLayerExtension(layer);
        if (propertySet != null)
        {
            return propertySet;
        }

        var layerExtensions = layer as ILayerExtensions;
        if (layerExtensions == null)
        {
            return null;
        }

        propertySet = new PropertySetClass();
        layerExtensions.AddExtension(propertySet);
        return propertySet;
    }

    private static IEnumerable<object> AsEnumerable(this ILayerExtensions layerExtensions)
    {
        if (layerExtensions == null) throw new ArgumentNullException("layerExtensions");

        for (var i = 0; i < layerExtensions.ExtensionCount; i++)
        {
            yield return layerExtensions.get_Extension(i);
        }
    }

    private static IEnumerable<KeyValuePair<string, object>> AsEnumerable(this IPropertySet propertySet)
    {
        if (propertySet == null) throw new ArgumentNullException("propertySet");
        if (propertySet.Count == 0) yield break;

        object names;
        object values;

        propertySet.GetAllProperties(out names, out values);

        var nameArray = (string[])names;
        var valueArray = (object[])values;

        for (var i = 0; i < nameArray.Length; i++)
        {
            yield return new KeyValuePair<string, object>(nameArray[i], valueArray[i]);
        }
    }
}

कभी-कभी, आप ILayerExtension में एक विशेष कुंजी के साथ एक IPropertySet की तरह कुछ स्टोर करने से दूर हो सकते हैं। चूंकि यह एक सामान्य "ट्रिक" है, डेवलपर्स को एक जोड़ने से पहले IPropertySet के अस्तित्व की जांच करनी चाहिए।
जेम्स शेक

@ नाम: अच्छा टिप, मैं जवाब अपडेट कर दूंगा।
पेट्र क्रेब्स

पिछली बार जब मैंने Esri की जाँच की तो केवल IPersistStream को सम्मानित किया गया - परत एक्सटेंशन के लिए IPersistVariant को नहीं। मुझे यकीन नहीं है कि क्यों। मैंने IPersistVariant समर्थन को एक वृद्धि के रूप में पूछा है, लेकिन यह निश्चित नहीं है कि इसे कभी लागू किया गया था। वैसे भी, आप नमूना कोड के लिए रिची कारमाइकल के आईपीसिस्टस्ट्रीम पोस्ट का उपयोग करना चाह सकते हैं ।
कर्क कुक्केंडल

IPersistStream का उपयोग करने के बारे में मुझे पागल करने वाली बात यह है कि यह ऐड-इन के साथ काम नहीं करता है। आप जिस वस्तु को ILayerExtensions में जोड़ते हैं वह COM CoCreatable होना चाहिए।
जेम्स स्कैच

@ किर्क: यह सही है, मुझे याद है कि इसके कारण वीबी में परत विस्तार दृढ़ता को लागू नहीं किया जा सका। सुधारों के लिए धन्यवाद।
पेट्र क्रेब्स
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.