예제 #1
0
        public MediumLevelRecordEditor(SubRecord sr, SubrecordStructure ss, dFormIDLookupS formIDLookup,
                                       dFormIDScan formIDScan, dLStringLookup strIDLookup)
        {
            InitializeComponent();
            Icon = Resources.tesv_ico;
            SuspendLayout();
            this.sr           = sr;
            this.ss           = ss;
            this.formIDLookup = formIDLookup;
            this.formIDScan   = formIDScan;
            this.strIDLookup  = strIDLookup;

            int offset = 0;

            byte[] data = sr.GetReadonlyData();
            boxes      = new List <TextBox>(ss.elements.Length);
            valueTypes = new List <ElementValueType>(ss.elements.Length);
            elements   = new List <Panel>();
            int groupOffset  = 0;
            int CurrentGroup = 0;

            try
            {
                for (int i = 0; i < ss.elements.Length; i++)
                {
                    if (ss.elements[i].optional && offset == data.Length)
                    {
                        AddElement(ss.elements[i]);
                    }
                    else
                    {
                        AddElement(ss.elements[i], ref offset, data, ref groupOffset, ref CurrentGroup);
                        if (ss.elements[i].repeat > 0)
                        {
                            repeatcount++;
                            if (offset < data.Length)
                            {
                                i--;
                            }
                        }
                    }
                }
                if (ss.elements[ss.elements.Length - 1].repeat > 0 && repeatcount > 0)
                {
                    AddElement(ss.elements[ss.elements.Length - 1]);
                }
            }
            catch
            {
                MessageBox.Show("The subrecord doesn't appear to conform to the expected structure.\n" +
                                "Saving is disabled, and the formatted information may be incorrect", "Warning");
                bSave.Enabled = false;
            }
            ResumeLayout();
        }
예제 #2
0
        public MediumLevelRecordEditor(SubRecord sr, SubrecordStructure ss, dFormIDLookupS formIDLookup, dFormIDScan formIDScan, dLStringLookup strIDLookup)
        {
            this.InitializeComponent();
            Icon = Resources.tesv_ico;
            SuspendLayout();
            this.sr = sr;
            this.ss = ss;
            this.formIDLookup = formIDLookup;
            this.formIDScan = formIDScan;
            this.strIDLookup = strIDLookup;

            int offset = 0;
            byte[] data = sr.GetReadonlyData();
            this.boxes = new List<TextBox>(ss.elements.Length);
            this.valueTypes = new List<ElementValueType>(ss.elements.Length);
            this.elements = new List<Panel>();
            int groupOffset = 0;
            int CurrentGroup = 0;
            try
            {
                for (int i = 0; i < ss.elements.Length; i++)
                {
                    if (ss.elements[i].optional > 0 && offset == data.Length)
                    {
                        this.AddElement(ss.elements[i]);
                    }
                    else
                    {
                        this.AddElement(ss.elements[i], ref offset, data, ref groupOffset, ref CurrentGroup);
                        if (ss.elements[i].repeat > 0)
                        {
                            this.repeatcount++;
                            if (offset < data.Length)
                            {
                                i--;
                            }
                        }
                    }
                }

                if (ss.elements[ss.elements.Length - 1].repeat > 0 && this.repeatcount > 0)
                {
                    this.AddElement(ss.elements[ss.elements.Length - 1]);
                }
            }
            catch
            {
                MessageBox.Show("The subrecord doesn't appear to conform to the expected structure.\n" + "Saving is disabled, and the formatted information may be incorrect", "Warning");
                this.bSave.Enabled = false;
            }

            ResumeLayout();
        }
예제 #3
0
        public void EditSelectedSubrecord()
        {
            var context = Selection;
            var rec     = Selection.Record as Record;

            if (rec == null)
            {
                return;
            }
            var p = GetPluginFromNode(rec);

            if (p == null)
            {
                return;
            }
            var sr = GetSelectedSubrecord();

            if (sr == null)
            {
                return;
            }

            if (OnEditSubrecord != null)
            {
                OnEditSubrecord(sr, false);
                return;
            }

            if (Properties.Settings.Default.UseOldSubRecordEditor)
            {
                new DataEdit(sr.Name, sr.GetData()).ShowDialog();
                if (!DataEdit.Canceled)
                {
                    sr.SetData(DataEdit.result);
                    sr.Name = DataEdit.resultName;
                }
            }
            else if (!Properties.Settings.Default.UseHexSubRecordEditor &&
                     sr.Structure != null &&
                     sr.Structure.elements != null &&
                     sr.Structure.elements.Length > 0 &&
                     sr.Structure.elements[0].type != ElementValueType.Blob && !sr.Structure.UseHexEditor)
            {
                Form re;
                try
                {
                    if (Properties.Settings.Default.UseOldSubRecordEditor)
                    {
                        var r            = context.Record;
                        var formIDLookup = new dFormIDLookupS(p.LookupFormIDS);
                        var formIDScan   = new dFormIDScan(FormIDScan);
                        var strIDLookup  = new dLStringLookup(p.LookupFormStrings);
                        re = new MediumLevelRecordEditor(sr, sr.Structure, formIDLookup, formIDScan, strIDLookup);
                    }
                    else
                    {
                        re = new NewMediumLevelRecordEditor(p, rec, sr, sr.Structure);
                    }
                }
                catch
                {
                    MessageBox.Show("Subrecord doesn't seem to conform to the expected structure.", Resources.ErrorText);
                    re = null;
                }
                if (re != null)
                {
                    if (DialogResult.OK == re.ShowDialog(this))
                    {
                        if (sr.Parent is Record)
                        {
                            sr.Parent.UpdateShortDescription();
                        }
                        listSubrecord.Refresh();
                        FireSubrecordChanged(sr);
                        FireDataChanged();
                    }
                    return;
                }
            }
            else
            {
                EditSelectedSubrecordHex();
            }
        }
예제 #4
0
 private string GetLocalizedString(dLStringLookup strLookup)
 {
     return(default(string));
 }
예제 #5
0
파일: Record.cs 프로젝트: rxantos/tesv-snip
 private string GetLocalizedString(dLStringLookup strLookup)
 {
     return default(string);
 }
예제 #6
0
        public static void GetFormattedData(this SubRecord rec, StringBuilder s)
        {
            SubrecordStructure ss = rec.Structure;

            if (ss == null)
            {
                return;
            }

            var            p             = rec.GetPlugin();
            dFormIDLookupI formIDLookup  = p.LookupFormID;
            dLStringLookup strLookup     = p.LookupFormStrings;
            dFormIDLookupR formIDLookupR = p.GetRecordByID;

            var recdata = rec.GetReadonlyData();
            int offset  = 0;

            s.AppendFormat("{0} ({1})", ss.name, ss.desc);
            s.AppendLine();
            try
            {
                for (int eidx = 0, elen = 1; eidx < ss.elements.Length; eidx += elen)
                {
                    var  sselem = ss.elements[eidx];
                    bool repeat = sselem.repeat > 0;
                    elen = sselem.repeat > 1 ? sselem.repeat : 1;
                    do
                    {
                        for (int eoff = 0; eoff < elen && offset < recdata.Length; ++eoff)
                        {
                            sselem = ss.elements[eidx + eoff];

                            if (offset == recdata.Length && eidx == ss.elements.Length - 1 && sselem.optional > 0)
                            {
                                break;
                            }

                            if (!sselem.notininfo)
                            {
                                s.Append(sselem.name).Append(": ");
                            }

                            switch (sselem.type)
                            {
                            case ElementValueType.Int:
                            {
                                string tmps = TypeConverter.h2si(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]).ToString();
                                if (!sselem.notininfo)
                                {
                                    if (sselem.hexview)
                                    {
                                        s.Append(TypeConverter.h2i(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]).ToString("X8"));
                                    }
                                    else
                                    {
                                        s.Append(tmps);
                                    }

                                    if (sselem.options != null && sselem.options.Length > 0)
                                    {
                                        for (int k = 0; k < sselem.options.Length; k += 2)
                                        {
                                            if (tmps == sselem.options[k + 1])
                                            {
                                                s.AppendFormat(" ({0})", sselem.options[k]);
                                            }
                                        }
                                    }
                                    else if (sselem.flags != null && sselem.flags.Length > 0)
                                    {
                                        uint val  = TypeConverter.h2i(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]);
                                        var  tmp2 = new StringBuilder();
                                        for (int k = 0; k < sselem.flags.Length; k++)
                                        {
                                            if ((val & (1 << k)) != 0)
                                            {
                                                if (tmp2.Length > 0)
                                                {
                                                    tmp2.Append(", ");
                                                }

                                                tmp2.Append(sselem.flags[k]);
                                            }
                                        }

                                        if (tmp2.Length > 0)
                                        {
                                            s.AppendFormat(" ({0})", tmp2);
                                        }
                                    }
                                }

                                offset += 4;
                            }

                            break;

                            case ElementValueType.UInt:
                            {
                                string tmps = TypeConverter.h2i(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]).ToString();
                                if (!sselem.notininfo)
                                {
                                    if (sselem.hexview)
                                    {
                                        s.Append(TypeConverter.h2i(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]).ToString("X8"));
                                    }
                                    else
                                    {
                                        s.Append(tmps);
                                    }

                                    if (sselem.options != null && sselem.options.Length > 0)
                                    {
                                        for (int k = 0; k < sselem.options.Length; k += 2)
                                        {
                                            if (tmps == sselem.options[k + 1])
                                            {
                                                s.AppendFormat(" ({0})", sselem.options[k]);
                                            }
                                        }
                                    }
                                    else if (sselem.flags != null && sselem.flags.Length > 0)
                                    {
                                        uint val  = TypeConverter.h2i(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]);
                                        var  tmp2 = new StringBuilder();
                                        for (int k = 0; k < sselem.flags.Length; k++)
                                        {
                                            if ((val & (1 << k)) != 0)
                                            {
                                                if (tmp2.Length > 0)
                                                {
                                                    tmp2.Append(", ");
                                                }

                                                tmp2.Append(sselem.flags[k]);
                                            }
                                        }

                                        if (tmp2.Length > 0)
                                        {
                                            s.AppendFormat(" ({0})", tmp2);
                                        }
                                    }
                                }

                                offset += 4;
                            }

                            break;

                            case ElementValueType.Short:
                            {
                                string tmps = TypeConverter.h2ss(recdata[offset], recdata[offset + 1]).ToString();
                                if (!sselem.notininfo)
                                {
                                    if (sselem.hexview)
                                    {
                                        s.Append(TypeConverter.h2ss(recdata[offset], recdata[offset + 1]).ToString("X4"));
                                    }
                                    else
                                    {
                                        s.Append(tmps);
                                    }

                                    if (sselem.options != null && sselem.options.Length > 0)
                                    {
                                        for (int k = 0; k < sselem.options.Length; k += 2)
                                        {
                                            if (tmps == sselem.options[k + 1])
                                            {
                                                s.AppendFormat(" ({0})", sselem.options[k]);
                                            }
                                        }
                                    }
                                    else if (sselem.flags != null && sselem.flags.Length > 0)
                                    {
                                        uint val  = TypeConverter.h2s(recdata[offset], recdata[offset + 1]);
                                        var  tmp2 = new StringBuilder();
                                        for (int k = 0; k < sselem.flags.Length; k++)
                                        {
                                            if ((val & (1 << k)) != 0)
                                            {
                                                if (tmp2.Length > 0)
                                                {
                                                    tmp2.Append(", ");
                                                }

                                                tmp2.Append(sselem.flags[k]);
                                            }
                                        }

                                        if (tmp2.Length > 0)
                                        {
                                            s.AppendFormat(" ({0})", tmp2);
                                        }
                                    }
                                }

                                offset += 2;
                            }

                            break;

                            case ElementValueType.UShort:
                            {
                                string tmps = TypeConverter.h2s(recdata[offset], recdata[offset + 1]).ToString();
                                if (!sselem.notininfo)
                                {
                                    if (sselem.hexview)
                                    {
                                        s.Append(TypeConverter.h2s(recdata[offset], recdata[offset + 1]).ToString("X4"));
                                    }
                                    else
                                    {
                                        s.Append(tmps);
                                    }

                                    if (sselem.options != null && sselem.options.Length > 0)
                                    {
                                        for (int k = 0; k < sselem.options.Length; k += 2)
                                        {
                                            if (tmps == sselem.options[k + 1])
                                            {
                                                s.Append(" (").Append(sselem.options[k]).Append(")");
                                            }
                                        }
                                    }
                                    else if (sselem.flags != null && sselem.flags.Length > 0)
                                    {
                                        uint val  = TypeConverter.h2s(recdata[offset], recdata[offset + 1]);
                                        var  tmp2 = new StringBuilder();
                                        for (int k = 0; k < sselem.flags.Length; k++)
                                        {
                                            if ((val & (1 << k)) != 0)
                                            {
                                                if (tmp2.Length > 0)
                                                {
                                                    tmp2.Append(", ");
                                                }

                                                tmp2.Append(sselem.flags[k]);
                                            }
                                        }

                                        if (tmp2.Length > 0)
                                        {
                                            s.AppendFormat(" ({0})", tmp2);
                                        }
                                    }
                                }

                                offset += 2;
                            }

                            break;

                            case ElementValueType.Byte:
                            {
                                string tmps = recdata[offset].ToString();
                                if (!sselem.notininfo)
                                {
                                    if (sselem.hexview)
                                    {
                                        s.Append(recdata[offset].ToString("X2"));
                                    }
                                    else
                                    {
                                        s.Append(tmps);
                                    }

                                    if (sselem.options != null && sselem.options.Length > 0)
                                    {
                                        for (int k = 0; k < sselem.options.Length; k += 2)
                                        {
                                            if (tmps == sselem.options[k + 1])
                                            {
                                                s.AppendFormat(" ({0})", sselem.options[k]);
                                            }
                                        }
                                    }
                                    else if (sselem.flags != null && sselem.flags.Length > 0)
                                    {
                                        int val  = recdata[offset];
                                        var tmp2 = new StringBuilder();
                                        for (int k = 0; k < sselem.flags.Length; k++)
                                        {
                                            if ((val & (1 << k)) != 0)
                                            {
                                                if (tmp2.Length > 0)
                                                {
                                                    tmp2.Append(", ");
                                                }

                                                tmp2.Append(sselem.flags[k]);
                                            }
                                        }

                                        if (tmp2.Length > 0)
                                        {
                                            s.AppendFormat(" ({0})", tmp2);
                                        }
                                    }
                                }

                                offset++;
                            }

                            break;

                            case ElementValueType.SByte:
                            {
                                string tmps = ((sbyte)recdata[offset]).ToString();
                                if (!sselem.notininfo)
                                {
                                    if (sselem.hexview)
                                    {
                                        s.Append(recdata[offset].ToString("X2"));
                                    }
                                    else
                                    {
                                        s.Append(tmps);
                                    }

                                    if (sselem.options != null && sselem.options.Length > 0)
                                    {
                                        for (int k = 0; k < sselem.options.Length; k += 2)
                                        {
                                            if (tmps == sselem.options[k + 1])
                                            {
                                                s.AppendFormat(" ({0})", sselem.options[k]);
                                            }
                                        }
                                    }
                                    else if (sselem.flags != null && sselem.flags.Length > 0)
                                    {
                                        int val  = recdata[offset];
                                        var tmp2 = new StringBuilder();
                                        for (int k = 0; k < sselem.flags.Length; k++)
                                        {
                                            if ((val & (1 << k)) != 0)
                                            {
                                                if (tmp2.Length > 0)
                                                {
                                                    tmp2.Append(", ");
                                                }

                                                tmp2.Append(sselem.flags[k]);
                                            }
                                        }

                                        if (tmp2.Length > 0)
                                        {
                                            s.AppendFormat(" ({0})", tmp2);
                                        }
                                    }
                                }

                                offset++;
                            }

                            break;

                            case ElementValueType.FormID:
                            {
                                uint id = TypeConverter.h2i(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]);
                                if (!sselem.notininfo)
                                {
                                    s.Append(id.ToString("X8"));
                                }

                                if (id != 0 && formIDLookup != null)
                                {
                                    s.Append(": ").Append(formIDLookup(id));
                                }

                                offset += 4;
                            }

                            break;

                            case ElementValueType.Float:
                                if (!sselem.notininfo)
                                {
                                    s.Append(TypeConverter.h2f(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]));
                                }

                                offset += 4;
                                break;

                            case ElementValueType.String:
                                if (!sselem.notininfo)
                                {
                                    while (recdata[offset] != 0)
                                    {
                                        s.Append((char)recdata[offset++]);
                                    }
                                }
                                else
                                {
                                    while (recdata[offset] != 0)
                                    {
                                        offset++;
                                    }
                                }

                                offset++;
                                break;

                            case ElementValueType.Blob:
                                if (!sselem.notininfo)
                                {
                                    s.Append(TypeConverter.GetHexData(recdata, offset, recdata.Length - offset));
                                }

                                offset += recdata.Length - offset;
                                break;

                            case ElementValueType.BString:
                            {
                                int len = TypeConverter.h2s(recdata[offset], recdata[offset + 1]);
                                if (!sselem.notininfo)
                                {
                                    s.Append(TESVSnip.Framework.Services.Encoding.Instance.GetString(recdata, offset + 2, len));
                                }

                                offset += 2 + len;
                            }

                            break;

                            case ElementValueType.IString:
                            {
                                int len = TypeConverter.h2si(recdata[offset], recdata[offset + 1], recdata[offset + 2], recdata[offset + 3]);
                                if (!sselem.notininfo)
                                {
                                    s.Append(TESVSnip.Framework.Services.Encoding.Instance.GetString(recdata, offset + 4, len));
                                }

                                offset += 4 + len;
                            }

                            break;

                            case ElementValueType.LString:
                            {
                                // Try to guess if string or string index.  Do not know if the external string checkbox is set or not in this code
                                int    left     = recdata.Length - offset;
                                var    data     = new ArraySegment <byte>(recdata, offset, left);
                                bool   isString = TypeConverter.IsLikelyString(data);
                                uint   id       = TypeConverter.h2i(data);
                                string lvalue   = strLookup(id);
                                if (!string.IsNullOrEmpty(lvalue) || !isString)
                                {
                                    if (!sselem.notininfo)
                                    {
                                        s.Append(id.ToString("X8"));
                                    }

                                    if (strLookup != null)
                                    {
                                        s.Append(": ").Append(lvalue);
                                    }

                                    offset += 4;
                                }
                                else
                                {
                                    if (!sselem.notininfo)
                                    {
                                        while (recdata[offset] != 0)
                                        {
                                            s.Append((char)recdata[offset++]);
                                        }
                                    }
                                    else
                                    {
                                        while (recdata[offset] != 0)
                                        {
                                            offset++;
                                        }
                                    }

                                    offset++;
                                }
                            }

                            break;

                            case ElementValueType.Str4:
                            {
                                if (!sselem.notininfo)
                                {
                                    s.Append(TESVSnip.Framework.Services.Encoding.Instance.GetString(recdata, offset, 4));
                                }

                                offset += 4;
                            }

                            break;

                            default:
                                throw new ApplicationException();
                            }

                            if (!sselem.notininfo)
                            {
                                s.AppendLine();
                            }
                        }
                    }while (repeat && offset < recdata.Length);
                }

                if (offset < recdata.Length)
                {
                    s.AppendLine();
                    s.AppendLine("Remaining Data: ");
                    s.Append(TypeConverter.GetHexData(recdata, offset, recdata.Length - offset));
                }
            }
            catch
            {
                s.AppendLine("Warning: Subrecord doesn't seem to match the expected structure");
            }
        }
예제 #7
0
        public static void GetFormattedData(this SubRecord rec, RTFBuilder s)
        {
            SubrecordStructure ss = rec.Structure;

            if (ss == null || ss.elements == null)
            {
                s.Append("String:\t").AppendLine(rec.GetStrData()).AppendLine();
                s.Append("Hex: \t").AppendLine(rec.GetHexData());
                s.AppendPara();
                return;
            }

            bool addTerminatingParagraph = false;

            try
            {
                var p = rec.GetPlugin();

                dFormIDLookupI formIDLookup  = p.LookupFormID;
                dLStringLookup strLookup     = p.LookupFormStrings;
                dFormIDLookupR formIDLookupR = p.GetRecordByID;

                // Table of items
                var table = new List <List <RTFCellDefinition> >();

                // set up elements
                float maxWidth          = 0;
                int   maxFirstCellWidth = 0;

                var elems = rec.EnumerateElements(true).Where(x => x.Structure != null && !x.Structure.notininfo).ToList();
                if (elems.Count == 0)
                {
                    return;
                }

                foreach (var element in elems)
                {
                    Size sz    = s.MeasureText(element.Structure.name);
                    int  width = Math.Max(sz.Width / 11, 10);

                    // approximate convert pixels to twips as the rtflib has crap documentation
                    if (width > maxFirstCellWidth)
                    {
                        maxFirstCellWidth = width;
                    }
                }

                foreach (var element in elems)
                {
                    var row = new List <RTFCellDefinition>();
                    table.Add(row);
                    var  sselem     = element.Structure;
                    bool hasOptions = sselem.options != null && sselem.options.Length > 0;
                    bool hasFlags   = sselem.flags != null && sselem.flags.Length > 1;

                    // setup borders for header
                    var value    = element.Value;
                    var nameCell = new RTFCellDefinition(maxFirstCellWidth, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty);
                    row.Add(nameCell);
                    switch (sselem.type)
                    {
                    case ElementValueType.FormID:
                        row.Add(new RTFCellDefinition(12, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        row.Add(new RTFCellDefinition(30, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));

                        // Optional Add cell for
                        break;

                    case ElementValueType.LString:
                        row.Add(new RTFCellDefinition(12, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        row.Add(new RTFCellDefinition(30, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        break;

                    case ElementValueType.BString:
                    case ElementValueType.IString:
                    case ElementValueType.String:
                        row.Add(new RTFCellDefinition(42, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        break;

                    case ElementValueType.Int:
                    case ElementValueType.UInt:
                    case ElementValueType.Byte:
                    case ElementValueType.SByte:
                    case ElementValueType.Short:
                    case ElementValueType.UShort:
                    case ElementValueType.Float:
                        row.Add(new RTFCellDefinition(20, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        row.Add(new RTFCellDefinition(30, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        break;

                    case ElementValueType.Blob:
                        row.Add(new RTFCellDefinition(42, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        break;

                    case ElementValueType.Str4:
                        row.Add(new RTFCellDefinition(42, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        break;

                    default:
                        row.Add(new RTFCellDefinition(42, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty));
                        break;
                    }

                    maxWidth = Math.Max(maxWidth, row.Sum(x => x.CellWidthRaw));
                }

                var rowWidth = (int)(maxWidth * 100.0f);
                var pd       = new Padding {
                    All = 50
                };

                var hdrd   = new RTFRowDefinition(rowWidth, RTFAlignment.TopLeft, RTFBorderSide.Default, 15, SystemColors.WindowText, pd);
                var hdrcds = new[] { new RTFCellDefinition(rowWidth, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 15, Color.DarkGray, Padding.Empty) };

                addTerminatingParagraph = true;
                s.Reset();
                using (IRTFRow ie = s.CreateRow(hdrd, hdrcds))
                {
                    foreach (var item in ie)
                    {
                        var rb = item.Content;
                        item.Content.FontSize(s.DefaultFontSize + 1).FontStyle(FontStyle.Bold).ForeColor(KnownColor.DarkCyan).AppendFormat("{0} ({1})", ss.name, ss.desc);
                    }
                }

                for (int rowIdx = 0; rowIdx < elems.Count; ++rowIdx)
                {
                    var rd     = new RTFRowDefinition(rowWidth, RTFAlignment.TopLeft, RTFBorderSide.Default, 15, SystemColors.WindowText, pd);
                    var cds    = table[rowIdx];
                    var elem   = elems[rowIdx];
                    var sselem = elem.Structure;
                    var value  = elem.Value;

                    string recprefix  = null;
                    Record record     = null;
                    string strValue   = null; // value to display
                    string strDesc    = null; // first description
                    string strDesc2   = null; // second description
                    bool   hasOptions = sselem.options != null && sselem.options.Length > 0;
                    bool   hasFlags   = sselem.flags != null && sselem.flags.Length > 1;

                    if (!string.IsNullOrWhiteSpace(elem.Structure.funcr))
                    {
                        if (elem.Type == ElementValueType.Float)
                        {
                            value = PyInterpreter.ExecuteFunction <float>(elem, FunctionOperation.ForReading);
                        }
                        else if (elem.Type == ElementValueType.Int)
                        {
                            value = PyInterpreter.ExecuteFunction <int>(elem, FunctionOperation.ForReading);
                        }
                        else if (elem.Type == ElementValueType.Short)
                        {
                            value = PyInterpreter.ExecuteFunction <short>(elem, FunctionOperation.ForReading);
                        }
                        else if (elem.Type == ElementValueType.UShort)
                        {
                            value = PyInterpreter.ExecuteFunction <ushort>(elem, FunctionOperation.ForReading);
                        }
                        else if (elem.Type == ElementValueType.UInt)
                        {
                            value = PyInterpreter.ExecuteFunction <uint>(elem, FunctionOperation.ForReading);
                        }
                    }

                    // Pre row write caching to avoid expensive duplicate calls between cells
                    switch (sselem.type)
                    {
                    case ElementValueType.FormID:
                    {
                        var id = (uint)value;
                        strValue = id.ToString("X8");
                        if (id != 0)
                        {
                            record = formIDLookupR != null?formIDLookupR(id) : null;
                        }

                        if (record != null)
                        {
                            var pref = record.GetPlugin();
                            recprefix = pref != null?string.Format("{0}@{1}", pref.Name, record.Name) : record.Name;

                            strDesc = record.DescriptiveName;
                            var full = record.SubRecords.FirstOrDefault(x => x.Name == "FULL");
                            if (full != null)
                            {
                                // split the cell 2 in 2 if full name found
                                var    data     = new ArraySegment <byte>(full.GetReadonlyData());
                                bool   isString = TypeConverter.IsLikelyString(data);
                                string lvalue   = isString ? full.GetStrData() : strLookup != null?strLookup(TypeConverter.h2i(data)) : null;

                                if (!string.IsNullOrEmpty(lvalue))
                                {
                                    var  first = cds[cds.Count - 1];
                                    Size sz    = s.MeasureText(lvalue);
                                    int  width = Math.Min(40, Math.Max(sz.Width / 12, 10));

                                    // approximate convert pixels to twips as the rtflib has crap documentation
                                    var second = new RTFCellDefinition(width, RTFAlignment.MiddleLeft, RTFBorderSide.Default, 0, Color.DarkGray, Padding.Empty);
                                    cds.Add(second);
                                    strDesc2 = lvalue;
                                }
                            }
                        }
                    }

                    break;

                    case ElementValueType.LString:
                    {
                        if (elem.Type == ElementValueType.String)
                        {
                            strValue = string.Empty;
                            strDesc  = value.ToString();
                        }
                        else if (TypeConverter.IsLikelyString(elem.Data))
                        {
                            strValue = string.Empty;
                            strDesc  = TypeConverter.GetString(elem.Data);
                        }
                        else
                        {
                            uint id = TypeConverter.h2i(elem.Data);
                            strValue = id.ToString("X8");
                            strDesc  = strLookup != null?strLookup(id) : null;
                        }
                    }

                    break;

                    case ElementValueType.Blob:
                        strValue = TypeConverter.GetHexData(elem.Data);
                        break;

                    case ElementValueType.SByte:
                    case ElementValueType.Int:
                    case ElementValueType.Short:
                    {
                        if (sselem.hexview || hasFlags)
                        {
                            if (sselem.hexviewwithdec)
                            {
                                strValue = string.Format(string.Format("{{0:X{0}}}", elem.Data.Count * 2), value) + string.Format(" : {0}", value);
                            }
                            else
                            {
                                strValue = string.Format(string.Format("{{0:X{0}}}", elem.Data.Count * 2), value);
                            }
                        }
                        else
                        {
                            strValue = value == null ? string.Empty : value.ToString();
                        }

                        if (hasOptions)
                        {
                            int intVal = Convert.ToInt32(value);
                            for (int k = 0; k < sselem.options.Length; k += 2)
                            {
                                int intValOption;
                                if (int.TryParse(sselem.options[k + 1], out intValOption) && intVal == intValOption)
                                {
                                    strDesc = sselem.options[k];
                                }
                            }
                        }
                        else if (hasFlags)
                        {
                            int intVal = Convert.ToInt32(value);
                            var tmp2   = new StringBuilder();
                            for (int k = 0; k < sselem.flags.Length; k++)
                            {
                                if ((intVal & (1 << k)) != 0)
                                {
                                    if (tmp2.Length > 0)
                                    {
                                        tmp2.Append(", ");
                                    }

                                    tmp2.Append(sselem.flags[k]);
                                }
                            }

                            strDesc = tmp2.ToString();
                        }
                    }

                    break;

                    case ElementValueType.UInt:
                    case ElementValueType.Byte:
                    case ElementValueType.UShort:
                    {
                        if (sselem.hexview || hasFlags)
                        {
                            if (sselem.hexviewwithdec)
                            {
                                strValue = string.Format(string.Format("{{0:X{0}}}", elem.Data.Count * 2), value) + string.Format(" : {0}", value);
                            }
                            else
                            {
                                strValue = string.Format(string.Format("{{0:X{0}}}", elem.Data.Count * 2), value);
                            }
                        }
                        else
                        {
                            strValue = value == null ? string.Empty : value.ToString();
                        }

                        if (hasOptions)
                        {
                            uint intVal = Convert.ToUInt32(value);
                            for (int k = 0; k < sselem.options.Length; k += 2)
                            {
                                if (intVal == uint.Parse(sselem.options[k + 1]))
                                {
                                    strDesc = sselem.options[k];
                                }
                            }
                        }
                        else if (hasFlags)
                        {
                            uint intVal = Convert.ToUInt32(value);
                            var  tmp2   = new StringBuilder();
                            for (int k = 0; k < sselem.flags.Length; k++)
                            {
                                if ((intVal & (1 << k)) != 0)
                                {
                                    if (tmp2.Length > 0)
                                    {
                                        tmp2.Append(", ");
                                    }

                                    tmp2.Append(sselem.flags[k]);
                                }
                            }

                            strDesc = tmp2.ToString();
                        }
                    }

                    break;

                    case ElementValueType.Str4:
                        strValue = TypeConverter.GetString(elem.Data);
                        break;

                    case ElementValueType.BString:
                        strValue = TypeConverter.GetBString(elem.Data);
                        break;

                    case ElementValueType.IString:
                        strValue = TypeConverter.GetIString(elem.Data);
                        break;

                    default:
                        strValue = value == null ? string.Empty : value.ToString();
                        break;
                    }

                    // Now create row and fill in cells
                    using (IRTFRow ie = s.CreateRow(rd, cds))
                    {
                        int colIdx = 0;
                        IEnumerator <IBuilderContent> ie2 = ie.GetEnumerator();
                        for (bool ok = ie2.MoveNext(); ok; ok = ie2.MoveNext(), ++colIdx)
                        {
                            using (var item = ie2.Current)
                            {
                                var rb = item.Content;
                                if (colIdx == 0)
                                {
                                    // name
                                    rb.FontStyle(FontStyle.Bold).Append(sselem.name);
                                }
                                else if (colIdx == 1)
                                {
                                    // value
                                    switch (sselem.type)
                                    {
                                    case ElementValueType.FormID:
                                        if (((uint)value) == 0)
                                        {
                                            rb.Append(strValue);
                                        }
                                        else if (record != null)
                                        {
                                            RTFRenderer.AppendLink(rb, strValue, record.GetLink());
                                        }
                                        else if (!string.IsNullOrEmpty(sselem.FormIDType))
                                        {
                                            RTFRenderer.AppendLink(rb, strValue, string.Format("{0}:{1}", sselem.FormIDType, strValue));
                                        }
                                        else
                                        {
                                            RTFRenderer.AppendLink(rb, strValue, string.Format("XXXX:{0}", strValue));
                                        }

                                        break;

                                    default:
                                        rb.Append(strValue);
                                        break;
                                    }
                                }
                                else if (colIdx == 2)
                                {
                                    // desc
                                    if (!string.IsNullOrEmpty(strDesc))
                                    {
                                        rb.Append(strDesc);
                                    }
                                }
                                else if (colIdx == 3)
                                {
                                    // desc2
                                    if (!string.IsNullOrEmpty(strDesc2))
                                    {
                                        rb.Append(strDesc2);
                                    }
                                }
                            }
                        }
                    }
                }
            }
            catch
            {
                s.AppendLine("Warning: Subrecord doesn't seem to match the expected structure");
            }
            finally
            {
                if (addTerminatingParagraph)
                {
                    s.Reset();
                    s.AppendPara();
                }
            }
        }