You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			152 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			152 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="ClientCultureInfo.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | //------------------------------------------------------------------------------ | ||
|  |   | ||
|  | namespace System.Web.Globalization { | ||
|  |     using System; | ||
|  |     using System.Collections; | ||
|  |     using System.Collections.Specialized; | ||
|  |     using System.Globalization; | ||
|  |     using System.Text; | ||
|  |     using System.Web.Util; | ||
|  |     using System.Web.Script.Serialization; | ||
|  | 
 | ||
|  |     internal class ClientCultureInfo { | ||
|  | 
 | ||
|  |         private static Hashtable cultureScriptBlockCache = Hashtable.Synchronized(new Hashtable()); | ||
|  |         private static readonly CultureInfo enUS = CultureInfo.GetCultureInfo(0x409); | ||
|  |         private static int eraNumber = 0; | ||
|  |         private static int eraName = 1; | ||
|  |         private static int eraStart = 2; | ||
|  |         private static int eraYearOffset = 3; | ||
|  | 
 | ||
|  |         public string name; | ||
|  |         public NumberFormatInfo numberFormat; | ||
|  |         public DateTimeFormatInfo dateTimeFormat; | ||
|  |         public object[] eras; | ||
|  |         private string _convertScript; | ||
|  |         private int _adjustment; | ||
|  | 
 | ||
|  |         private ClientCultureInfo(CultureInfo cultureInfo) { | ||
|  |             name = cultureInfo.Name; | ||
|  |             numberFormat = cultureInfo.NumberFormat; | ||
|  |             dateTimeFormat = cultureInfo.DateTimeFormat; | ||
|  |             var calendar = dateTimeFormat == null ? null : dateTimeFormat.Calendar; | ||
|  |             if (calendar != null) { | ||
|  |                 // Dev10 425049: Support Eras for gregorian based calendars | ||
|  |                 // with a simple year offset, and non-gregorian calendars. | ||
|  |                 // Era data is stored in binary resource "culture.nlp" in mscorlib, | ||
|  |                 // hard coded here for simplicity. | ||
|  |                 // era array has the following structure: | ||
|  |                 // [eraNumber1, eraName1, eraStartInTicks1, eraGregorianYearOffset1, eraNumber2, ...] | ||
|  |                 eras = new object[calendar.Eras.Length * 4]; | ||
|  |                 int i = 0; | ||
|  |                 foreach (int era in calendar.Eras) { | ||
|  |                     // era number | ||
|  |                     eras[i + eraNumber] = era; | ||
|  |                     // era name | ||
|  |                     eras[i + eraName] = dateTimeFormat.GetEraName(era); | ||
|  |                     // calendars with only one era will have a null tick count | ||
|  |                     // signifying that the era starts from the lowest datetime | ||
|  |                     // era begining in ticks (null = the oldest era) | ||
|  |                     // eras[i + eraStart] = null; | ||
|  |                     // era year offset from normal gregorian year | ||
|  |                     // some calendars dont have an offset, just a different name | ||
|  |                     // for the A.D. era (B.C. is not supported by normal calendar, | ||
|  |                     // so most calendars only have 1 era) | ||
|  |                     eras[i + eraYearOffset] = 0; | ||
|  |                     i += 4; | ||
|  |                 } | ||
|  |                 var calendarType = calendar.GetType(); | ||
|  |                 if (calendarType != typeof(GregorianCalendar)) { | ||
|  |                     if (calendarType == typeof(TaiwanCalendar)) { | ||
|  |                         // Only the current era is supported, so no tick count is needed | ||
|  |                         //eras[eraStart] = -1830384000000; | ||
|  |                         eras[eraYearOffset] = 1911; | ||
|  |                     } | ||
|  |                     else if (calendarType == typeof(KoreanCalendar)) { | ||
|  |                         // only one era to speak of, so no tick count is needed | ||
|  |                         //eras[eraStart] = -62135596800000; | ||
|  |                         eras[eraYearOffset] = -2333; | ||
|  |                     } | ||
|  |                     else if (calendarType == typeof(ThaiBuddhistCalendar)) { | ||
|  |                         // only one era to speak of, so no tick count is needed | ||
|  |                         //eras[eraStart] = -62135596800000; | ||
|  |                         eras[eraYearOffset] = -543; | ||
|  |                     } | ||
|  |                     else if (calendarType == typeof(JapaneseCalendar)) { | ||
|  |                         // there are multiple eras | ||
|  |                         eras[0 + eraStart] = 60022080000; | ||
|  |                         eras[0 + eraYearOffset] = 1988; | ||
|  |                         eras[4 + eraStart] = -1357603200000; | ||
|  |                         eras[4 + eraYearOffset] = 1925; | ||
|  |                         eras[8 + eraStart] = -1812153600000; | ||
|  |                         eras[8 + eraYearOffset] = 1911; | ||
|  |                         // oldest era is technically from this offset, but for simplicity | ||
|  |                         // it is counted from the lowest date time, so no tick count needed. | ||
|  |                         //eras[12 + eraStart] = -3218832000000; | ||
|  |                         eras[12 + eraYearOffset] = 1867; | ||
|  |                     } | ||
|  |                     else if (calendarType == typeof(HijriCalendar)) { | ||
|  |                         _convertScript = "Date.HijriCalendar.js"; | ||
|  |                         _adjustment = ((HijriCalendar)calendar).HijriAdjustment; | ||
|  |                     } | ||
|  |                     else if (calendarType == typeof(UmAlQuraCalendar)) { | ||
|  |                         _convertScript = "Date.UmAlQuraCalendar.js"; | ||
|  |                     } | ||
|  |                     // else { other calendars arent supported or have no era offsets just different names for A.D. | ||
|  |                 } | ||
|  |             } | ||
|  |              | ||
|  |         } | ||
|  | 
 | ||
|  |         internal Tuple<String, String> GetClientCultureScriptBlock() { | ||
|  |             return GetClientCultureScriptBlock(CultureInfo.CurrentCulture); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static Tuple<String, String> GetClientCultureScriptBlock(CultureInfo cultureInfo) { | ||
|  |             if (cultureInfo == null) { | ||
|  |                 return null; | ||
|  |             } | ||
|  | 
 | ||
|  |             // note: DateTimeFormat could be null since it is a virtual property, but DateTimeFormat.Calendar cannot be | ||
|  |             Type calendarType = cultureInfo.DateTimeFormat == null ? null : cultureInfo.DateTimeFormat.Calendar.GetType(); | ||
|  |             if (cultureInfo.Equals(enUS) && (calendarType == typeof(GregorianCalendar))) { | ||
|  |                 return null; | ||
|  |             } | ||
|  | 
 | ||
|  |             var key = new Tuple<CultureInfo, Type>(cultureInfo, calendarType); | ||
|  |             Tuple<String, String> cached = cultureScriptBlockCache[key] as Tuple<String, String>; | ||
|  |             if (cached == null) { | ||
|  |                 ClientCultureInfo clientCultureInfo = new ClientCultureInfo(cultureInfo); | ||
|  |                 string json = JavaScriptSerializer.SerializeInternal(BuildSerializeableCultureInfo(clientCultureInfo)); | ||
|  |                 if (json.Length > 0) { | ||
|  |                     string script = "var __cultureInfo = " + json + ";"; | ||
|  |                     if (clientCultureInfo._adjustment != 0) { | ||
|  |                         script += "\r\n__cultureInfo.dateTimeFormat.Calendar._adjustment = " + clientCultureInfo._adjustment.ToString(CultureInfo.InvariantCulture) + ";"; | ||
|  |                     } | ||
|  |                     cached = new Tuple<String, String>(script, clientCultureInfo._convertScript); | ||
|  |                 } | ||
|  |                 cultureScriptBlockCache[key] = cached; | ||
|  |             } | ||
|  |             return cached; | ||
|  |         } | ||
|  | 
 | ||
|  |         private static OrderedDictionary BuildSerializeableCultureInfo(ClientCultureInfo clientCultureInfo) { | ||
|  |             // It's safe to serialize the values set in the dictionary | ||
|  |             //  name is a string | ||
|  |             //  numberFormat is NumberFormatInfo which is a public type | ||
|  |             //  dateTimeFormat is a DateFormatInfo which is a public type | ||
|  |             //  eras is an object[] array that only contains strings or numbers | ||
|  |             var dictionary = new OrderedDictionary(); | ||
|  |             dictionary["name"] = clientCultureInfo.name; | ||
|  |             dictionary["numberFormat"] = clientCultureInfo.numberFormat; | ||
|  |             dictionary["dateTimeFormat"] = clientCultureInfo.dateTimeFormat; | ||
|  |             dictionary["eras"] = clientCultureInfo.eras; | ||
|  |             return dictionary; | ||
|  |         } | ||
|  |     } | ||
|  | } |