using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using System.IO;
using System.Web;
using System.Drawing;
using System.Net;
using System.Diagnostics;
using System.Text;
using SevenZip;
using Travis.Storage;
using Travis.Util;
using Travis.Http;

using Zanetti.UI;
using Zanetti.Data;

namespace Zanetti
{
    internal enum PrimitiveIndicator
    {
        Date,
        Open,
        High,
        Low,
        Close,
        Volume,
        CreditLong,
        CreditShort,
        LAST
    }
    [EnumDesc(typeof(ChartFormat))]
    internal enum ChartFormat
    {
        [EnumValue(Description = "日足")] Daily,
        [EnumValue(Description = "週足")] Weekly,
        [EnumValue(Description = "月足")] Monthly,
        [EnumValue(Description = "年足")] Yearly
#if DOJIMA
		, [EnumValue(Description = "半日足")] HalfDaily
#endif
    }

    internal enum FormatModifier
    {
        Nop,
        Mul100,
        Percent
    }
    internal enum CompanyInfoSite
    {
        Yahoo,
        Infoseek,
        Nikkei,
        Livedoor
    }


    /// <summary>
    /// Util の概要の説明です。
    /// </summary>
    internal class Util
    {
        //ユーザ定義銘柄のコード最小値
        public const int CustomBrandMinValue = 400;

        public static void ReportCriticalError(Exception ex)
        {
            SilentReportCriticalError(ex);
            //メッセージボックスで通知
            string dir = Env.GetAppDir();
            MessageBox.Show(String.Format("内部エラーが発生しました。" + dir + "error.logファイルにエラー位置が記録されました。\n{0}", ex.Message), "Error", MessageBoxButtons.OK, MessageBoxIcon.Stop);
        }
        public static void SilentReportCriticalError(Exception ex)
        {
            Debug.WriteLine(ex.Message);
            Debug.WriteLine(ex.StackTrace);
            //エラーファイルに追記
            string dir = Env.GetAppDir();
            StreamWriter sw = new StreamWriter(dir + "error.log", true);
            sw.WriteLine(DateTime.Now.ToString() + " : " + ex.Message);
            sw.WriteLine(ex.StackTrace);
            sw.Close();
        }

        public static MemoryStream HttpDownload(string url)
        {
            /* 2005/2/12より、ケンミレはクッキーを要求するようになった。WinHTTP経由だとこれに対応できない。
			 * スクリプトによるプロキシ設定はあきらめざるをえないが仕方ないか。
			if(HTTPConnection.IsWinHTTPAvailable && Env.Options.ProxyConfig.UseIESetting) {
				return new HTTPConnection(url).Open();
			}
			*/

            HttpWebRequest rq = (HttpWebRequest)WebRequest.Create(url);
            //rq.CookieContainer = _cookieContainer;
            rq.ProtocolVersion = new Version(1, 1);
            rq.Method = "GET";
            if (!Env.Options.ProxyConfig.UseIESetting)
                rq.Proxy = new WebProxy(Env.Options.ProxyConfig.Address, Env.Options.ProxyConfig.Port);
            HttpWebResponse rs = (HttpWebResponse)rq.GetResponse();

            //効率はわるいがWinHTTPが使えない環境用なのでまあいいだろう
            MemoryStream strm = new MemoryStream(0x18000);
            CopyStream(rs.GetResponseStream(), strm);
            strm.Close();
            return new MemoryStream(strm.ToArray());
        }

        public static Stream ExtractData(string url)
        {
            var execdir = Path.GetDirectoryName(Application.ExecutablePath) ?? "";
            SevenZipBase.SetLibraryPath(Path.Combine(execdir, IntPtr.Size == 4 ? "7z.dll" : "7z64.dll"));
            var tmp = Path.Combine(execdir, Path.GetFileName(url));
            var result = new MemoryStream();
            using (var ms = Util.HttpDownload(url))
            using (var file = File.Create(tmp))
            {
                ms.WriteTo(file);
                using (var extractor = new SevenZipExtractor(file))
                {
                    if (extractor.ArchiveFileData.Count != 1)
                        throw new ApplicationException("データファイルの展開に失敗しました。");
                    extractor.ExtractFile(0, result);
                }
            }
            File.Delete(tmp);
            result.Seek(0, SeekOrigin.Begin);
            return result;
        }

        public static void CopyStream(Stream input, Stream output)
        {
            byte[] buffer = new byte[0x20000];
            int n = input.Read(buffer, 0, buffer.Length);
            while (n > 0)
            {
                output.Write(buffer, 0, n);
                n = input.Read(buffer, 0, buffer.Length);
            }
        }
        public static void StreamToFile(Stream input, string filename)
        {
            using (FileStream fs = new FileStream(filename, FileMode.Create, FileAccess.Write))
                CopyStream(input, fs);
        }

        //COLORREFに対応した整数を返す
        public static uint ToCOLORREF(Color c)
        {
            uint t = (uint)c.ToArgb();
            //COLORREFは0x00BBGGRR、ToArgbは0x00RRGGBB
            uint r = (t & 0x00FF0000) >> 16;
            uint b = (t & 0x000000FF) << 16;
            t &= 0x0000FF00;
            return t | r | b;
        }
        public static Color ToLightColor(Color c)
        {
            return Color.FromArgb((255 + c.R) / 2, (255 + c.G) / 2, (255 + c.B) / 2);
        }
        public static Color ToDarkColor(Color c)
        {
            return Color.FromArgb(c.R / 2, c.G / 2, c.B / 2);
        }
        //c1*v + c2*(1-v)
        public static Color MergeColor(Color c1, Color c2, double v)
        {
            double r1 = (double)c1.R;
            double g1 = (double)c1.G;
            double b1 = (double)c1.B;
            double r2 = (double)c2.R;
            double g2 = (double)c2.G;
            double b2 = (double)c2.B;

            return Color.FromArgb((int)(r1 * v + r2 * (1 - v)), (int)(g1 * v + g2 * (1 - v)), (int)(b1 * v + b2 * (1 - v)));
        }
        public static string FormatColor(Color col)
        {
            if (col.IsNamedColor)
                return col.Name;
            else
                return "#" + col.Name;
        }
        public static Color ParseColor(string src, Color def)
        {
            if (src.StartsWith("#"))
            {
                try
                {
                    return Color.FromArgb(Int32.Parse(src.Substring(1), System.Globalization.NumberStyles.HexNumber));
                }
                catch (Exception)
                {
                    return def;
                }
            }
            else
            {
                Color c = Color.FromName(src);
                return c;
            }
        }


        public static string GetDailyDataFileName(int code)
        {
            if (code < 1000)
                return Env.GetAppDir() + "data\\0" + code;
            else
                return Env.GetAppDir() + "data\\" + code;
        }

        public static bool IsDailyBased(ChartFormat fmt)
        {
#if DOJIMA
			return fmt==ChartFormat.Daily || fmt==ChartFormat.HalfDaily;
#else
            return fmt == ChartFormat.Daily;
#endif
        }

        private static string[] _dayOfWeek = { "日", "月", "火", "水", "木", "金", "土" };

        //内部形式の日付のフォーマット
        public static string FormatFullDate(int date)
        {
            DateTime d = new DateTime(date / 10000, (date % 10000) / 100, (date % 100));
            return String.Format("{0}年{1:d2}月{2:d2}日({3})", d.Year, d.Month, d.Day, _dayOfWeek[(int)d.DayOfWeek]);
        }
        public static string FormatFullDate(int date, ChartFormat format)
        {
            DateTime d = new DateTime(date / 10000, (date % 10000) / 100, (date % 100));
            if (IsDailyBased(format))
                return String.Format("{0}年{1:d2}月{2:d2}日({3})", d.Year, d.Month, d.Day, _dayOfWeek[(int)d.DayOfWeek]);
            else if (format == ChartFormat.Weekly)
            {
                d = d.AddDays(5); //金曜日の位置で決めるのが自然
                return String.Format("{0}年{1:d2}月{2}週", d.Year, d.Month, (d.Day - 1) / 7 + 1);
            }
            else if (format == ChartFormat.Yearly)
            {
                return String.Format("{0}年", d.Year);
            }
            else // Monthly
                return String.Format("{0}年{1:d2}月", d.Year, d.Month);
        }
        public static string FormatShortDate(int date)
        {
            DateTime d = new DateTime(date / 10000, (date % 10000) / 100, (date % 100));
            return String.Format("{0}/{1:d2}/{2:d2}", d.Year, d.Month, d.Day);
        }
        public static DayOfWeek GetDayOfWeek(int date)
        {
            DateTime d = new DateTime(date / 10000, (date % 10000) / 100, (date % 100));
            return d.DayOfWeek;
        }

        //len文字の右詰文字
        public static string FormatFixedLenValue(double value, int len, string format, FormatModifier mod)
        {
            if (Double.IsNaN(value))
                return new string(' ', len - 1) + '-';
            else
            {
                if (mod == FormatModifier.Mul100 || mod == FormatModifier.Percent) value *= 100;
                string t = Math.Abs(value) < 100000 ? value.ToString(format) : value.ToString("F0"); //あまりでかい数に小数点つけても仕方ない
                if (mod == FormatModifier.Percent) t += "%";
                if (t.Length < len) t = new string(' ', len - t.Length) + t;

                return t;
            }
        }
        //一般的な数値フォーマット
        public static string FormatValue(double value, string fs, FormatModifier mod)
        {
            switch (mod)
            {
                case FormatModifier.Nop:
                    return value.ToString(fs);
                case FormatModifier.Mul100:
                    return (value * 100).ToString(fs);
                case FormatModifier.Percent:
                    return (value * 100).ToString(fs) + "%";
                default:
                    return "format error";
            }
        }

        //データが利用可能と思われる最終日付を取得
        public static DateTime GuessLatestTradeDate()
        {
            DateTime dt = DateTime.UtcNow.AddHours(9); //日本時間を常に取得
            return GuessLatestTradeDate(dt);
        }
        //last以前で最後の日付を取得
        public static DateTime GuessLatestTradeDate(DateTime last)
        {
            while (!IsMarketOpenDate(last)) last = last.AddDays(-1);
            return new DateTime(last.Year, last.Month, last.Day, 0, 0, 0);
        }
        public static int DateToInt(DateTime dt)
        {
            return dt.Year * 10000 + dt.Month * 100 + dt.Day;
        }
        public static int DateToInt(int year, int month, int day)
        {
            return year * 10000 + month * 100 + day;
        }
        public static DateTime IntToDate(int dt)
        {
            return new DateTime(dt / 10000, (dt % 10000) / 100, (dt % 100));
        }

        /// <summary>
        /// 指定された日付に市場が開いているかを返す。
        /// </summary>
        /// <param name="d">日付を指定する。</param>
        /// <returns></returns>
        public static bool IsMarketOpenDate(DateTime d)
        {
            // 土日
            if (d.DayOfWeek == DayOfWeek.Saturday ||
                d.DayOfWeek == DayOfWeek.Sunday)
                return false;
            // 年末年始
            if ((d.Month == 12 && d.Day == 31) ||
                (d.Month == 1 && d.Day <= 3))
                return false;
            return !IsHoliday(d);
        }

        static bool IsHoliday(DateTime d)
        {
            // 動かない休日
            if ((d.Month == 1 && d.Day == 1) ||
                (d.Month == 2 && d.Day == 11) ||
                (d.Month == 4 && d.Day == 29) ||
                (d.Month == 5 && (d.Day >= 3 && d.Day <= 5)) ||
                (d.Month == 11 && (d.Day == 3 || d.Day == 23)) ||
                (d.Month == 12 && d.Day == 23))
                return true;
            if (d.Year >= 2016 && d.Month == 8 && d.Day == 11) // 山の日
                return true;
            // 春分と秋分(1980~2099年に対応)
            if (d.Month == 3 &&
                d.Day == (int)(20.8431 + 0.242194 * (d.Year - 1980) - (d.Year - 1980) / 4))
                return true;
            if (d.Month == 9 &&
                d.Day == (int)(23.2488 + 0.242194 * (d.Year - 1980) - (d.Year - 1980) / 4))
                return true;
            // ハッピーマンデー
            if (d.Year < 2000)
            {
                if ((d.Month == 1 && d.Day == 15) ||
                    (d.Month == 10 && d.Day == 10))
                    return true;
            }
            else
            {
                if ((d.Month == 1 || d.Month == 10) &&
                    (d.Day >= 8 && d.Day <= 14) && d.DayOfWeek == DayOfWeek.Monday)
                    return true;
            }
            if (d.Year < 2003)
            {
                if ((d.Month == 7 && d.Day == 20) ||
                    (d.Month == 9 && d.Day == 15))
                    return true;
            }
            else
            {
                if ((d.Month == 7 || d.Month == 9) &&
                    (d.Day >= 15 && d.Day <= 21) && d.DayOfWeek == DayOfWeek.Monday)
                    return true;
                // シルバーウィーク
                if (d.Month == 9 && (d.Day == 21 || d.Day == 22) && d.DayOfWeek == DayOfWeek.Tuesday)
                    return IsHoliday(d.AddDays(1));
            }
            // 5月4日が国民の祝日になったので、振替休日が二日以上繰り越す。
            if (d.Year > 2007 && d.Month == 5 && d.Day == 6)
                return d.DayOfWeek >= DayOfWeek.Monday &&
                    d.DayOfWeek <= DayOfWeek.Wednesday;
            // 振り替え休日
            if (d.DayOfWeek == DayOfWeek.Monday)
                return IsHoliday(d.AddDays(-1));
            return false;
        }

        public static double Yobine(MarketType mt, double v)
        {
            //呼値を計算
            double r;
            if (mt == MarketType.B)
                return 1;
            /* 旧JASDAQ版
				if(v <= 1000)
					r = 1;
				else if(v < 10000)
					r = 10;
				else if(v < 100000)
					r = 100;
				else if(v < 1000000)
					r = 1000;
				else if(v < 10000000)
					r = 10000;
				else
					r = 50000;
			}
			else {
			*/
            if (v <= 2000)
                r = 1;
            else if (v <= 3000)
                r = 5;
            else if (v <= 30000)
                r = 10;
            else if (v <= 50000)
                r = 50;
            else if (v <= 100000)
                r = 100;
            else if (v <= 1000000)
                r = 1000;
            else if (v <= 20000000)
                r = 10000;
            else if (v <= 30000000)
                r = 50000;
            else
                r = 100000;
            return r;
        }
        //呼値の整数倍になるように丸める
        public static double RoundToYobine(MarketType mt, double v)
        {
            double y = Yobine(mt, v);
            if (y == 1) return v; //ショートカット
            double n = Math.Round(v / y);
            return n * y;
        }

        public static void Warning(IWin32Window parent, string msg)
        {
            MessageBox.Show(parent, msg, Env.Constants.AppTitle, MessageBoxButtons.OK, MessageBoxIcon.Warning);
        }
        public static void Warning(string msg)
        {
            MessageBox.Show(msg, Env.Constants.AppTitle, MessageBoxButtons.OK, MessageBoxIcon.Warning);
        }
        public static void Information(IWin32Window parent, string msg)
        {
            MessageBox.Show(parent, msg, Env.Constants.AppTitle, MessageBoxButtons.OK, MessageBoxIcon.Information);
        }
        public static DialogResult AskUserYesNo(IWin32Window parent, string msg)
        {
            return MessageBox.Show(parent, msg, Env.Constants.AppTitle, MessageBoxButtons.YesNo, MessageBoxIcon.Question);
        }

        public static void AddMenuBar(Menu parent)
        {
            ZMenuItem mi = new ZMenuItem();
            mi.Text = "-";
            mi.Index = parent.MenuItems.Count;
            parent.MenuItems.Add(mi);
        }
        public static void AddMenuItem(Menu parent, string text, EventHandler handler)
        {
            ZMenuItem mi = new ZMenuItem();
            mi.Text = text;
            mi.Index = parent.MenuItems.Count;
            mi.Click += handler;
            parent.MenuItems.Add(mi);
        }

        public static double ParseDouble(string value, double def)
        {
            try
            {
                if (value == null || value.Length == 0)
                    return def;
                else
                    return Double.Parse(value);
            }
            catch (Exception)
            {
                return def;
            }
        }
        public static double[] ParseDoubles(string value)
        {
            string[] t = value.Split(',');
            double[] r = new double[t.Length];
            for (int i = 0; i < t.Length; i++)
                r[i] = Double.Parse(t[i]);
            return r;
        }
        public static int ParseInt(string value)
        {
            try
            {
                return Int32.Parse(value);
            }
            catch (Exception)
            {
                Debug.WriteLine("Int32 parse error " + value);
                throw;
            }
        }
        public static int ParseInt(string value, int def)
        {
            try
            {
                if (value == null || value.Length == 0)
                    return def;
                else
                    return Int32.Parse(value);
            }
            catch (Exception)
            {
                Debug.WriteLine("Int32 parse error " + value);
                return def;
            }
        }
        public static int ParseHexInt(string value, int def)
        {
            try
            {
                if (value == null || value.Length == 0)
                    return def;
                else
                    return Int32.Parse(value, System.Globalization.NumberStyles.HexNumber); //先頭に0xがついていてはだめ
            }
            catch (Exception)
            {
                Debug.WriteLine("Int32 parse error " + value);
                return def;
            }
        }
        public static float ParseFloat(string value, float def)
        {
            try
            {
                if (value == null || value.Length == 0)
                    return def;
                else
                    return Single.Parse(value);
            }
            catch (Exception)
            {
                return def;
            }
        }
        public static bool ParseBool(string value)
        {
            return value == "True";
        }
        public static bool ParseBool(string value, bool def)
        {
            if (value == "True" || value == "true")
                return true;
            else if (value == "False" || value == "false")
                return false;
            else
                return def;
        }
        public static HeightConfig ParseHeightConfig(string value, HeightConfig def)
        {
            try
            {
                if (value == null || value.Length == 0) return def; //shortcut
                return (HeightConfig)Enum.Parse(typeof(HeightConfig), value);
            }
            catch (Exception)
            {
                return def;
            }
        }

        public static int IndexOf(object[] values, object value)
        {
            for (int i = 0; i < values.Length; i++)
                if (values[i] == value) return i;
            return -1;
        }

        public static string LoadMandatoryAttr(StorageNode node, string name)
        {
            string r = node[name];
            if (r == null || r.Length == 0)
                throw new FormatException(String.Format("{0}に必須アトリビュート{1}がないか、空です。", node.Name, name));
            return r;
        }
        public static string LoadMandatoryText(StorageNode node, string name)
        {
            StorageNode r = node.FindChildNode(name);
            if (r == null)
                throw new FormatException(String.Format("{0}に必須要素{1}がありません。", node.Name, name));
            else if (r.TextValue == null)
                throw new FormatException(String.Format("{0}の必須要素{1}が空です。", node.Name, name));
            return r.TextValue;
        }
        public static string LoadOptionalText(StorageNode node, string name, string def)
        {
            StorageNode r = node.FindChildNode(name);
            if (r == null)
                return def;
            else if (r.TextValue == null)
                return def;
            else
                return r.TextValue;
        }

        public static int SafeArgLength(Array a)
        {
            return a == null ? 0 : a.Length;
        }

        public static Keys ParseKey(string s)
        {
            if (s.Length == 0)
                return Keys.None;
            /*
			 !!もしここの処理が遅ければ自前で書きなおせる
			else if(s.Length==1) {
				char ch = s[0];
				if('0'<=ch && ch<='9')
					return Keys.D0 + (ch - '0');
			}
			*/

            return (Keys)Enum.Parse(typeof(Keys), s);
        }
        public static string FormatShortcut(Keys key)
        {
            if (key == Keys.None) return "";

            Keys modifiers = key & Keys.Modifiers;
            StringBuilder b = new StringBuilder();
            if ((modifiers & Keys.Control) != Keys.None)
            {
                b.Append("Ctrl");
            }
            if ((modifiers & Keys.Shift) != Keys.None)
            {
                if (b.Length > 0) b.Append('+');
                b.Append("Shift");
            }
            if ((modifiers & Keys.Alt) != Keys.None)
            {
                if (b.Length > 0) b.Append('+');
                b.Append("Alt");
            }
            if (b.Length > 0)
                b.Append('+');

            b.Append(KeyString(key & Keys.KeyCode));
            return b.ToString();
        }
        public static string KeyString(Keys key)
        {
            int ik = (int)key;
            if ((int)Keys.D0 <= ik && ik <= (int)Keys.D9)
                return new string((char)('0' + (ik - (int)Keys.D0)), 1);
            else
            {
                switch (key)
                {
                    case Keys.None:
                        return "";
                    case Keys.Prior:
                        return "PageUp";
                    case Keys.Next:
                        return "PageDown";
                    //Oemほにゃららがうざったい
                    case Keys.OemBackslash:
                        return "Backslash";
                    case Keys.OemCloseBrackets:
                        return "CloseBrackets";
                    case Keys.Oemcomma:
                        return "Comma";
                    case Keys.OemMinus:
                        return "Minus";
                    case Keys.OemOpenBrackets:
                        return "OpenBrackets";
                    case Keys.OemPeriod:
                        return "Period";
                    case Keys.OemPipe:
                        return "Pipe";
                    case Keys.Oemplus:
                        return "Plus";
                    case Keys.OemQuestion:
                        return "Question";
                    case Keys.OemQuotes:
                        return "Quotes";
                    case Keys.OemSemicolon:
                        return "Semicolon";
                    case Keys.Oemtilde:
                        return "Tilde";
                    default:
                        return key.ToString();
                }
            }
        }

        public static void Swap(ref int x, ref int y)
        {
            int t = x;
            x = y;
            y = t;
        }

    }

    internal abstract class Trans
    {
        protected double _a;
        protected double _b;

        public Trans(double a, double b)
        {
            _a = a;
            _b = b;
        }
        public double A
        {
            get
            {
                return _a;
            }
        }
        public double B
        {
            get
            {
                return _b;
            }
        }
        public static Trans Solve(double x1, double y1, double x2, double y2, bool logscale, bool inverseupdown)
        {
            if (logscale)
            {
                if (inverseupdown)
                {
                    return LogTrans.SolveUpsideDown(x1, y1, x2, y2);
                }
                else
                {
                    return LogTrans.Solve(x1, y1, x2, y2);
                }
            }
            else
            {
                if (inverseupdown)
                {
                    return LinearTrans.SolveUpsideDown(x1, y1, x2, y2);
                }
                else
                {
                    return LinearTrans.Solve(x1, y1, x2, y2);
                }
            }
        }
        public abstract double TransValue(double x);
        public abstract double Inverse(double y);
        public abstract double Inverse(float y);//☆Fibonacci 
    }

    //y = ax + b の変換をする
    internal class LinearTrans : Trans
    {
        public LinearTrans(double a, double b) : base(a, b)
        {
        }

        public override double TransValue(double x)
        {
            return _a * x + _b;
        }
        public override double Inverse(double y)
        {
            return (y - _b) / _a;
        }
        public override double Inverse(float y)//☆Fibonacci 
        {
            return (y - _b) / _a;
        }

        //0除算や精度はあまり気にせず連立1次方程式を解く
        public static LinearTrans Solve(double x1, double y1, double x2, double y2)
        {
            double a = (y1 - y2) / (x1 - x2);
            return new LinearTrans(a, y1 - a * x1);
        }
        public static LinearTrans SolveUpsideDown(double x1, double y1, double x2, double y2)
        {
            double a = -(y1 - y2) / (x1 - x2);
            return new LinearTrans(a, y1 - a * x2);
        }
    }

    internal class LogTrans : Trans
    {
        public LogTrans(double a, double b) : base(a, b)
        {
        }
        public override double TransValue(double x)
        {
            // ボリバンなどがはみ出てもトリミング
            if (x < 1)
                x = 1;
            return _a * Math.Log10(x) + _b;
        }
        public override double Inverse(double y)
        {
            return Math.Pow(10, (y - _b) / _a);
        }
        public override double Inverse(float y)//☆Fibonacci 
        {
            return Math.Pow(10, (y - _b) / _a);
        }
        public static LogTrans Solve(double x1, double y1, double x2, double y2)
        {
            // 負の部分はトリミング
            if (x2 < 1)
            {
                y2 = LinearTrans.Solve(x1, y1, x2, y2).TransValue(1);
                x2 = 1;
            }

            double a = (y1 - y2) / Math.Log10(x1 / x2);
            double b = y1 - a * Math.Log10(x1);

            return new LogTrans(a, b);
        }
        public static LogTrans SolveUpsideDown(double x1, double y1, double x2, double y2)
        {
            // 負の部分はトリミング
            if (x2 < 1)
            {
                y2 = LinearTrans.Solve(x1, y1, x2, y2).TransValue(1);
                x2 = 1;
            }

            double a = -(y1 - y2) / Math.Log10(x1 / x2);
            double b = y1 - a * Math.Log10(x2);

            return new LogTrans(a, b);
        }
    }


    //時間のかかる動作のときに途中経過を知らせるための定数 SendMessageを使う
    internal class AsyncConst
    {
        public const int WM_ASYNCPROCESS = Win32.WM_USER + 1;
        public const int LPARAM_PROGRESS_SUCCESSFUL = 1;
        public const int LPARAM_PROGRESS_FAILURE = 2;
        public const int LPARAM_FINISHED = 3;
        public const int LPARAM_ERROR = 4;
    }

    //Look & Feelを変更するための仕組み
    internal class ThemeUtil
    {
        public enum Theme
        {
            Unspecified,
            Luna
        }

        private static Theme _theme;

        public static Theme CurrentTheme
        {
            get
            {
                return _theme;
            }
        }

        [DllImport("uxtheme.dll", CharSet = CharSet.Unicode)]
        private static extern int GetCurrentThemeName(char[] filename, int filenamelen, char[] colorbuff, int colornamelen, char[] sizebuff, int sizebufflen);

        private static void SpecifyThemeUnderWinXP()
        {
            try
            {
                char[] fn = new char[256];
                char[] cb = new char[256];
                char[] sz = new char[256];
                int r = GetCurrentThemeName(fn, 256, cb, 256, sz, 256);
                if (r == 0)
                {
                    string theme_name = new string(fn);
                    if (theme_name.IndexOf("Luna") != -1)
                        _theme = Theme.Luna;
                }
                //Debug.WriteLine(String.Format("FN={0} Color={1} Size={2}", new string(fn), new string(cb), new string(sz)));
            }
            catch (Exception)
            {
            }
        }

        public static void Init()
        {
            _theme = Theme.Unspecified;
            OperatingSystem os = System.Environment.OSVersion;
            if (os.Platform == PlatformID.Win32NT && os.Version.CompareTo(new Version(5, 1)) >= 0)
                SpecifyThemeUnderWinXP();
        }

        public static Color TabPaneBackColor
        {
            get
            {
                if (_theme == Theme.Luna)
                    return Color.FromKnownColor(KnownColor.ControlLightLight);
                else
                    return Color.FromKnownColor(KnownColor.Control);
            }
        }


    }

}