不再為項目重命名和修改命名空間而煩惱,簡單幾個字,但是開發加上測試大量項目,前前后后竟然跨越了1個月,汗。。。不過
真正的開發時間可能2-3天的樣子。
1.雖然說我們平常不會經常出現項目重命名的情況,但是一旦出現,修改起來還是一項比較大的工程,並且還不一定修改完整。
2.當團隊發展到一定程度的時候,基本上都有了自己固定的一些WEB/Winform開發框架和通用項目模版,這樣就會出現修改項目名稱,命名空間等結構的情況。
3.噠噠噠噠噠噠,不說第三了。親,沒了。@_@
1.自動重命名關聯的各種文件,並且支持自定義擴展。
2.自動檢測文件編碼格式,處理讀取不同編碼格式時出現的亂碼問題。(當項目內包含中文的時候)
3.自動修改sln和csproj文件中的引用關系,並且根據文件名稱修改文件夾名稱。
4.清理BIN目錄和OBJ目錄,已經項目產生的不必要的垃圾文件,可單獨使用。
5.每個csproj都是執行單獨修改,然后更新關聯引用,並記住修改過的引用。
6.輸入項目新名詞的文本框加入了選詞功能(文本框本身雙擊內容是全選的效果)。
7.自己試一下把。+_+
1.VS的觀察
更新前的VS項目解決方案
更新后的VS項目解決方案
2.系統資源管理器的觀察
更新前的文件夾結構
更新后的文件夾結構
3.軟件使用截圖
1.打開軟件顯示很簡單的打開解決方案的視圖。
2.選擇解決方案打開后,會自動加載該目下所有的csproj文件。
3.點擊【軟件設置】會出現其他需要修改關聯的文件后綴名,你也可以直接寫文件全名。
你也可以給輸入新名稱的文本框添加選詞分隔符,以達到你所需要的自動選詞效果。
1.文本框智能選詞功能的實現。
1 public delegate void AutoSelectTextBoxDoubleclick(object sender, MouseEventArgs e); 2 /// <summary> 3 /// 智能選擇文本控件,親,這個類copy走就可以使用喲 4 /// </summary> 5 public class AutoSelectTextBox : TextBox 6 { 7 const int WM_LBUTTONDBLCLK = 0x0203; 8 MouseEventArgs e; 9 10 /// <summary> 11 /// 是否啟用智能選擇 12 /// </summary> 13 [System.ComponentModel.Description("是否啟用智能選擇")] 14 public bool EnableAutoSelect 15 { 16 get; 17 set; 18 } 19 20 private List<char> splitChar; 21 22 /// <summary> 23 /// 選擇判斷分隔符,默認 . 號 24 /// </summary> 25 [System.ComponentModel.Description("判斷選擇分隔符,默認 . 號")] 26 [System.ComponentModel.Browsable(false)] 27 public List<char> SplitChar 28 { 29 get { return splitChar; } 30 set { splitChar = value; } 31 } 32 33 /// <summary> 34 /// 智能選擇文本框雙擊事件 35 /// </summary> 36 public event AutoSelectTextBoxDoubleclick AutoSelectTextMouseDoubleclick; 37 38 protected override void WndProc(ref Message m) 39 { 40 41 if (EnableAutoSelect && m.Msg == WM_LBUTTONDBLCLK) 42 { 43 Point p = this.PointToClient(MousePosition); 44 e = new MouseEventArgs(MouseButtons.Left, 2, p.X, p.Y, 0); 45 if (AutoSelectTextMouseDoubleclick != null) 46 { 47 AutoSelectTextMouseDoubleclick(this, e); 48 } 49 else 50 { 51 MouseDoubleClickAutoSelect(e); 52 } 53 return; 54 55 } 56 else 57 { 58 base.WndProc(ref m); 59 } 60 } 61 62 /// <summary> 63 /// 智能選擇實現 64 /// </summary> 65 private void MouseDoubleClickAutoSelect(MouseEventArgs e) 66 { 67 if (this.Text != "") 68 { 69 int pint = this.GetCharIndexFromPosition(e.Location); 70 int len = this.Text.Length; 71 int left = pint, right = pint; 72 73 while (left >= 0) 74 { 75 char lchar = this.Text[left]; 76 if (CheckSpiltChar(lchar)) 77 { 78 break; 79 } 80 left--; 81 } 82 while (right <= len - 1) 83 { 84 char rchar = this.Text[right]; 85 if (CheckSpiltChar(rchar)) 86 { 87 break; 88 } 89 right++; 90 } 91 //必須有字符可選 92 if (right - (left + 1) > 0) 93 { 94 this.Select(left + 1, right - (left + 1)); 95 } 96 } 97 98 } 99 100 101 102 /// <summary> 103 /// 檢查 104 /// </summary> 105 /// <param name="source"></param> 106 /// <returns></returns> 107 private bool CheckSpiltChar(char source) 108 { 109 if (SplitChar != null) 110 { 111 foreach (char c in SplitChar) 112 { 113 if (char.Equals(source, c)) 114 { 115 return true; 116 } 117 } 118 return false; 119 } 120 else 121 { 122 return char.Equals(source, '.'); 123 } 124 } 125 126 }
解釋:該自動完成選詞功能需要重寫文本框的消息機制,其中 WM_LBUTTONDBLCLK = 0x0203 為鼠標雙擊消息,然后重寫WndProc該事件。
為什么我重新寫雙擊消息,而不是用文本框本身的雙擊事件,是因為本身的雙擊事件會處理一些其他內容,並且會出現先選中全部內容然后再自動選詞的閃爍問題,
有興趣的同學可以試一下就知道了。
2.老調重彈自動識別文件編碼問題,大部分的解決方案都不太理想,基本都是讀取文件2-4位,判斷字節大小,來決定文件編碼類型。
這樣的思路只是其中之一,當我郁悶至極的時候突然想到ITextSharp控件(重量級語言高亮編輯器),找到源碼,發現了一塊編碼自動識別的功能。
下面代碼是ITextSharp控件中文件編碼識別的源碼,這個我發現比百度和google出來的效果好很多,基本上無識別錯誤情況,基本100%識別正確。
但是帶來的后果是要檢查一段內容前面很大一段內容,用性能換編碼識別精度。
1 public static StreamReader OpenStream(Stream fs, Encoding defaultEncoding) 2 { 3 if (fs == null) 4 throw new ArgumentNullException("fs"); 5 6 if (fs.Length >= 2) 7 { 8 // the autodetection of StreamReader is not capable of detecting the difference 9 // between ISO-8859-1 and UTF-8 without BOM. 10 int firstByte = fs.ReadByte(); 11 int secondByte = fs.ReadByte(); 12 switch ((firstByte << 8) | secondByte) 13 { 14 case 0x0000: // either UTF-32 Big Endian or a binary file; use StreamReader 15 case 0xfffe: // Unicode BOM (UTF-16 LE or UTF-32 LE) 16 case 0xfeff: // UTF-16 BE BOM 17 case 0xefbb: // start of UTF-8 BOM 18 // StreamReader autodetection works 19 fs.Position = 0; 20 return new StreamReader(fs); 21 default: 22 return AutoDetect(fs, (byte)firstByte, (byte)secondByte, defaultEncoding); 23 } 24 } 25 else 26 { 27 if (defaultEncoding != null) 28 { 29 return new StreamReader(fs, defaultEncoding); 30 } 31 else 32 { 33 return new StreamReader(fs); 34 } 35 } 36 } 37 38 static StreamReader AutoDetect(Stream fs, byte firstByte, byte secondByte, Encoding defaultEncoding) 39 { 40 int max = (int)Math.Min(fs.Length, 500000); // look at max. 500 KB 41 const int ASCII = 0; 42 const int Error = 1; 43 const int UTF8 = 2; 44 const int UTF8Sequence = 3; 45 int state = ASCII; 46 int sequenceLength = 0; 47 byte b; 48 for (int i = 0; i < max; i++) 49 { 50 if (i == 0) 51 { 52 b = firstByte; 53 } 54 else if (i == 1) 55 { 56 b = secondByte; 57 } 58 else 59 { 60 b = (byte)fs.ReadByte(); 61 } 62 if (b < 0x80) 63 { 64 // normal ASCII character 65 if (state == UTF8Sequence) 66 { 67 state = Error; 68 break; 69 } 70 } 71 else if (b < 0xc0) 72 { 73 // 10xxxxxx : continues UTF8 byte sequence 74 if (state == UTF8Sequence) 75 { 76 --sequenceLength; 77 if (sequenceLength < 0) 78 { 79 state = Error; 80 break; 81 } 82 else if (sequenceLength == 0) 83 { 84 state = UTF8; 85 } 86 } 87 else 88 { 89 state = Error; 90 break; 91 } 92 } 93 else if (b >= 0xc2 && b < 0xf5) 94 { 95 // beginning of byte sequence 96 if (state == UTF8 || state == ASCII) 97 { 98 state = UTF8Sequence; 99 if (b < 0xe0) 100 { 101 sequenceLength = 1; // one more byte following 102 } 103 else if (b < 0xf0) 104 { 105 sequenceLength = 2; // two more bytes following 106 } 107 else 108 { 109 sequenceLength = 3; // three more bytes following 110 } 111 } 112 else 113 { 114 state = Error; 115 break; 116 } 117 } 118 else 119 { 120 // 0xc0, 0xc1, 0xf5 to 0xff are invalid in UTF-8 (see RFC 3629) 121 state = Error; 122 break; 123 } 124 } 125 fs.Position = 0; 126 switch (state) 127 { 128 case ASCII: 129 case Error: 130 // when the file seems to be ASCII or non-UTF8, 131 // we read it using the user-specified encoding so it is saved again 132 // using that encoding. 133 if (IsUnicode(defaultEncoding)) 134 { 135 // the file is not Unicode, so don't read it using Unicode even if the 136 // user has choosen Unicode as the default encoding. 137 138 // If we don't do this, SD will end up always adding a Byte Order Mark 139 // to ASCII files. 140 defaultEncoding = Encoding.Default; // use system encoding instead 141 } 142 return new StreamReader(fs, defaultEncoding); 143 default: 144 return new StreamReader(fs); 145 } 146 }
ITextSharp文件識別基本思路:
1.傳入需要讀取文件流,並且傳入一個你認為合適的編碼方式,當然你可以傳入NULL。
2.讀取文件前面2個字節(這個是大部分解析編碼需要做的事情),讀取進來的字節向左移8位,然后與第二個字節進行 | 運算,主要用途是
檢測StreamReader是否在iso - 8859 - 1和utf - 8之間的編碼范圍,並且不包含BOM。
忘記了C#這些運算的可以看看,http://www.cnblogs.com/NatureSex/archive/2011/04/21/2023768.html
3.當位移出來的結果等於0x0000,0xfffe,0xfeff,0xefbb時,是StreamReader可以進行自動讀取范圍內的。如果不在范圍內時再進行讀取偏移檢測,
具體怎么檢測的,補一下系統編碼,任何有興趣的可以細看。
下載地址:VS重命名工具 喜歡的朋友記得推薦一下喲
源碼就不放了,文件沒有混淆也沒有加密,自己反編譯統統都有。