Hello!
Please let me know if I'm able to save some variables into current DWG?
I would like to provide availability for user to save some data which is different for each DWG file.
Thanks.
Solved! Go to Solution.
Lookup Xrecords / Xdata and Dictionary
http://through-the-interface.typepad.com/through_the_interface/2007/04/adding_xdata_to.html
http://through-the-interface.typepad.com/through_the_interface/2012/02/dynamic-net-in-autocad-2013.h...
Further to arcticad's reply, you'd typically store global data on the Named Objects Dictionary, and data related to a particular entity as xdata on that entity or as an xrecord in its extension dictionary.
But if you're storing 'variables' (sysvars), then you migh prefer to use the Autodesk.AutoCAD.Runtime.Variable class. You can specify whether the variable is per session, per user, per profile or per database.
I don't seem to find any information on how to use Autodesk.AutoCAD.Runtime.Variable
Can you point me in the direction.
Thanks.
I used LispVariable to store data in file so user could use them in Text Field.
Here are examples how to get and set Lisp Variables I used.
http://www.theswamp.org/index.php?topic=35714.0
Richard.
Thank you for the links.
Following the first, I have question - am I able to set my XData without selecting an entity?
Hi articad,
You can see this thread about custom Variables.
IMO, the main issue with Autodesk.AutoCAD.Runtim.Variable is the variable has to be registered in HKLM which most often requires administrator user account.
Use DBDictionary to save data (XRecord) in file.
Try to search for NamedObjectsDictionary (NOD) you should find many examples in this forum.
Richard
Do you have some example of code for XRecords and DBDictionary?
I'm searching, but it hard to find what I need.
Here're two little example to set and get xrecord data (ResultBuffer as for Xdata) in a named dictionary child of the root NamedObjectsDictionary (NOD).
/// <summary> /// Add or edit a Xrecord data in a named dictionary (the dictionary and xrecord are created if not already exist) /// </summary> /// <param name="dictName">The dictionary name</param> /// <param name="key">the xrecord key</param> /// <param name="resbuf">the xrecord data</param> public void SetXrecord(string dictName, string key, ResultBuffer resbuf) { Document doc = acadApp.DocumentManager.MdiActiveDocument; Database db = doc.Database; using (Transaction tr = db.TransactionManager.StartTransaction()) { DBDictionary NOD = (DBDictionary)tr.GetObject(db.NamedObjectsDictionaryId, OpenMode.ForRead); DBDictionary dict; if (NOD.Contains(dictName)) { dict = (DBDictionary)tr.GetObject(NOD.GetAt(dictName), OpenMode.ForWrite); } else { dict = new DBDictionary(); NOD.UpgradeOpen(); NOD.SetAt(dictName, dict); tr.AddNewlyCreatedDBObject(dict, true); } Xrecord xRec = new Xrecord(); xRec.Data = resbuf; dict.SetAt(key, xRec); tr.AddNewlyCreatedDBObject(xRec, true); tr.Commit(); } } /// <summary> /// Gets an xrecord data in a named dictionary /// </summary> /// <param name="dictName">The dictionary name</param> /// <param name="key">The xrecord key</param> /// <returns>The xrecord data or null if the dictionary or the xrecord do not exist</returns> public ResultBuffer GetXrecord(string dictName, string key) { Document doc = acadApp.DocumentManager.MdiActiveDocument; Database db = doc.Database; using (Transaction tr = db.TransactionManager.StartTransaction()) { DBDictionary NOD = (DBDictionary)tr.GetObject(db.NamedObjectsDictionaryId, OpenMode.ForRead); if (!NOD.Contains(dictName)) return null; DBDictionary dict = tr.GetObject(NOD.GetAt(dictName), OpenMode.ForRead) as DBDictionary; if (dict == null || !dict.Contains(key)) return null; Xrecord xRec = tr.GetObject(dict.GetAt(key), OpenMode.ForRead) as Xrecord; if (xRec == null) return null; return xRec.Data; } }
Thank you very much for help.
Please tell me one more thinkg. Am I able to get specified TypedValue from ResultBuffer?
If not I suppose that the best way would be to put all values into dictionary<int,string>. Am I right?
As for Xdata you have to use a ResultBuffer to set data in a Xrecord and the Xrecord.Data property is a ResultBuffer.
Assuming your xrecord contains a single data of type String, you can get the data this way:
ResultBuffer resbuf = GetXrecord(theDictName, theXrecKey);
string theFirstData =(string) resbuf.AsArray()[0].Value;
Please let me know one more thing.
Am I able to get all keys from my dictionary?
I would to allow user to select from which Key in my dictionary user want to see his variables.
In example:
SetXrecord("myDict", "keysFirst", rb);
SetXrecord("myDict", "keysSecond", rb);
SetXrecord("myDict", "keysThird", rb);
And now I want to list: keysFirst,keysSecond,keysThird.
Ok, I've got it:
List<string> _keys = new List<string>(); foreach (DBDictionaryEntry entry in dict) { _keys.Add(entry.Key); } return string.Join(",", _keys.ToArray());
One more to do 🙂 Removing keys 😛
I have this code:
public static void RemoveXrecord(string dictName, string key) { Document doc = Autodesk.AutoCAD.ApplicationServices.Application.DocumentManager.MdiActiveDocument; Database db = doc.Database; DocumentLock dl = doc.LockDocument(DocumentLockMode.ProtectedAutoWrite, null, null, true); using (dl) { using (Transaction tr = db.TransactionManager.StartTransaction()) { DBDictionary NOD = (DBDictionary)tr.GetObject(db.NamedObjectsDictionaryId, OpenMode.ForRead); if (!NOD.Contains(dictName)) return; DBDictionary dict = tr.GetObject(NOD.GetAt(dictName), OpenMode.ForWrite) as DBDictionary; if (dict == null || !dict.Contains(key)) return; dict.Remove(key); } } }
But key still exist after removing 😞
Once you work with Dictionaries very useful is ArxDbd utility for testing.
Can't find what you're looking for? Ask the community or share your knowledge.