Можете ли вы контролировать порядок, в котором класс TagBuilder отображает атрибуты? - PullRequest
5 голосов
/ 12 августа 2010

Я знаю, что это отчасти навязчиво, но есть ли способ контролировать порядок, в котором класс TagBuilder отображает атрибуты тега HTML при вызове ToString()?

т.е. так что

var tb = new TagBuilder("meta");            
tb.Attributes.Add("http-equiv", "Content-Type");            
tb.Attributes.Add("content", "text/html; charset=utf-8");    
tb.ToString(TagRenderMode.SelfClosing)

вернется

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />

не

<meta content="text/html; charset=utf-8" http-equiv="Content-Type" />

Изменение порядка добавления атрибутов не меняет его, похоже, оно отображается в алфавитном порядке

Ответы [ 3 ]

3 голосов
/ 12 августа 2010

Попробуйте использовать этот класс, который наследует TagBuilder и переопределяет метод ToString, создает SortedDictionary из атрибутов и использует этот словарь для визуализации.

    public class MyTagBuilder : TagBuilder
    {
        //required to inherit from TagBuilder
        public MyTagBuilder(string tagName) : base(tagName){}

        //new hides the original ToString(TagRenderMode renderMode) 
        //The only changes in this method is that all calls to GetAttributesString
        //have been changed to GetMyAttributesString 
        public new string ToString(TagRenderMode renderMode)
        {
            switch (renderMode)
            {
                case TagRenderMode.StartTag:
                    return string.Format(CultureInfo.InvariantCulture, "<{0}{1}>", new object[] { this.TagName, this.GetMyAttributesString() });

                case TagRenderMode.EndTag:
                    return string.Format(CultureInfo.InvariantCulture, "</{0}>", new object[] { this.TagName });

                case TagRenderMode.SelfClosing:
                    return string.Format(CultureInfo.InvariantCulture, "<{0}{1} />", new object[] { this.TagName, this.GetMyAttributesString() });
            }
            return string.Format(CultureInfo.InvariantCulture, "<{0}{1}>{2}</{0}>", new object[] { this.TagName, this.GetMyAttributesString(), this.InnerHtml });
        }

        //Implement GetMyAttributesString where the Attributes are changed to a SortedDictionary
        private string GetMyAttributesString()
        {
            var builder = new StringBuilder();
            var myDictionary = new SortedDictionary<string, string>();     //new
            foreach (KeyValuePair<string, string> pair in this.Attributes) //new
            {                                                              //new
                myDictionary.Add(pair.Key, pair.Value);                    //new
            }                                                              //new 
            //foreach (KeyValuePair<string, string> pair in this.Attributes)
            foreach (KeyValuePair<string, string> pair in myDictionary)    //changed
            {
                string key = pair.Key;
                if (!string.Equals(key, "id", StringComparison.Ordinal) || !string.IsNullOrEmpty(pair.Value))
                {
                    string str2 = HttpUtility.HtmlAttributeEncode(pair.Value);
                    builder.AppendFormat(CultureInfo.InvariantCulture, " {0}=\"{1}\"", new object[] { key, str2 });
                }
            }
            return builder.ToString();
        }
    }
1 голос
/ 12 августа 2010

Я дизассемблировал TagBuilder.ToString() с помощью Reflector, и это ключевой бит кода:

foreach (KeyValuePair<string, string> pair in this.Attributes)
{
    string key = pair.Key;
    string str2 = HttpUtility.HtmlAttributeEncode(pair.Value);
    builder.AppendFormat(CultureInfo.InvariantCulture, " {0}=\"{1}\"", new object[] { key, str2 });
}

Так что я бы сказал, что нет - this.Attributes - это интерфейс IDictionary<string,string> при перечислении этогопорядок, в котором возвращаются товары, не определен ", в соответствии с MSDN.

0 голосов
/ 12 января 2018

Я не хотел переопределять весь этот код для изменения поведения сортировки, поэтому я изменил свойство Attributes на обычный несортированный словарь с отражением вместо

private class MyTagBuilder: TagBuilder
{
    private static readonly MethodInfo tagBuilderAttrSetMethod = typeof(TagBuilder).GetProperty(nameof(Attributes)).SetMethod;

    public MyTagBuilder(string tagName) : base(tagName)
    {
        // TagBuilder internally uses SortedDictionary, render attributes according to the order they are added instead
        tagBuilderAttrSetMethod.Invoke(this, new object[] { new Dictionary<string, string>() });
    }
}
...