1 // OpenTween - Client of Twitter
2 // Copyright (c) 2007-2011 kiri_feather (@kiri_feather) <kiri.feather@gmail.com>
3 // (c) 2008-2011 Moz (@syo68k)
4 // (c) 2008-2011 takeshik (@takeshik) <http://www.takeshik.org/>
5 // (c) 2010-2011 anis774 (@anis774) <http://d.hatena.ne.jp/anis774/>
6 // (c) 2010-2011 fantasticswallow (@f_swallow) <http://twitter.com/f_swallow>
7 // (c) 2011 Egtra (@egtra) <http://dev.activebasic.com/egtra/>
8 // (c) 2013 kim_upsilon (@kim_upsilon) <https://upsilo.net/~upsilon/>
9 // All rights reserved.
11 // This file is part of OpenTween.
13 // This program is free software; you can redistribute it and/or modify it
14 // under the terms of the GNU General Public License as published by the Free
15 // Software Foundation; either version 3 of the License, or (at your option)
18 // This program is distributed in the hope that it will be useful, but
19 // WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
20 // or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
23 // You should have received a copy of the GNU General Public License along
24 // with this program. If not, see <http://www.gnu.org/licenses/>, or write to
25 // the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor,
26 // Boston, MA 02110-1301, USA.
28 using System.Diagnostics;
32 using System.Runtime.CompilerServices;
33 using System.Runtime.Serialization;
34 using System.Runtime.Serialization.Json;
36 using System.Text.RegularExpressions;
37 using System.Threading;
38 using System.Threading.Tasks;
41 using System.Xml.Linq;
43 using System.Reflection;
44 using System.Collections.Generic;
46 using System.Windows.Forms;
51 public class Twitter : IDisposable
53 #region Regexp from twitter-text-js
55 // The code in this region code block incorporates works covered by
56 // the following copyright and permission notices:
58 // Copyright 2011 Twitter, Inc.
60 // Licensed under the Apache License, Version 2.0 (the "License"); you
61 // may not use this work except in compliance with the License. You
62 // may obtain a copy of the License in the LICENSE file, or at:
64 // http://www.apache.org/licenses/LICENSE-2.0
66 // Unless required by applicable law or agreed to in writing, software
67 // distributed under the License is distributed on an "AS IS" BASIS,
68 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
69 // implied. See the License for the specific language governing
70 // permissions and limitations under the License.
73 private const string LATIN_ACCENTS = @"\u00c0-\u00d6\u00d8-\u00f6\u00f8-\u00ff\u0100-\u024f\u0253\u0254\u0256\u0257\u0259\u025b\u0263\u0268\u026f\u0272\u0289\u028b\u02bb\u1e00-\u1eff";
74 private const string NON_LATIN_HASHTAG_CHARS = @"\u0400-\u04ff\u0500-\u0527\u1100-\u11ff\u3130-\u3185\uA960-\uA97F\uAC00-\uD7AF\uD7B0-\uD7FF";
75 //private const string CJ_HASHTAG_CHARACTERS = @"\u30A1-\u30FA\uFF66-\uFF9F\uFF10-\uFF19\uFF21-\uFF3A\uFF41-\uFF5A\u3041-\u3096\u3400-\u4DBF\u4E00-\u9FFF\u20000-\u2A6DF\u2A700-\u2B73F\u2B740-\u2B81F\u2F800-\u2FA1F";
76 private const string CJ_HASHTAG_CHARACTERS = @"\u30A1-\u30FA\u30FC\u3005\uFF66-\uFF9F\uFF10-\uFF19\uFF21-\uFF3A\uFF41-\uFF5A\u3041-\u309A\u3400-\u4DBF\p{IsCJKUnifiedIdeographs}";
77 private const string HASHTAG_BOUNDARY = @"^|$|\s|「|」|。|\.|!";
78 private const string HASHTAG_ALPHA = "[a-z_" + LATIN_ACCENTS + NON_LATIN_HASHTAG_CHARS + CJ_HASHTAG_CHARACTERS + "]";
79 private const string HASHTAG_ALPHANUMERIC = "[a-z0-9_" + LATIN_ACCENTS + NON_LATIN_HASHTAG_CHARS + CJ_HASHTAG_CHARACTERS + "]";
80 private const string HASHTAG_TERMINATOR = "[^a-z0-9_" + LATIN_ACCENTS + NON_LATIN_HASHTAG_CHARS + CJ_HASHTAG_CHARACTERS + "]";
81 public const string HASHTAG = "(" + HASHTAG_BOUNDARY + ")(#|#)(" + HASHTAG_ALPHANUMERIC + "*" + HASHTAG_ALPHA + HASHTAG_ALPHANUMERIC + "*)(?=" + HASHTAG_TERMINATOR + "|" + HASHTAG_BOUNDARY + ")";
83 private const string url_valid_preceding_chars = @"(?:[^A-Za-z0-9@@$##\ufffe\ufeff\uffff\u202a-\u202e]|^)";
84 public const string url_invalid_without_protocol_preceding_chars = @"[-_./]$";
85 private const string url_invalid_domain_chars = @"\!'#%&'\(\)*\+,\\\-\.\/:;<=>\?@\[\]\^_{|}~\$\u2000-\u200a\u0009-\u000d\u0020\u0085\u00a0\u1680\u180e\u2028\u2029\u202f\u205f\u3000\ufffe\ufeff\uffff\u202a-\u202e";
86 private const string url_valid_domain_chars = @"[^" + url_invalid_domain_chars + "]";
87 private const string url_valid_subdomain = @"(?:(?:" + url_valid_domain_chars + @"(?:[_-]|" + url_valid_domain_chars + @")*)?" + url_valid_domain_chars + @"\.)";
88 private const string url_valid_domain_name = @"(?:(?:" + url_valid_domain_chars + @"(?:-|" + url_valid_domain_chars + @")*)?" + url_valid_domain_chars + @"\.)";
89 private const string url_valid_GTLD = @"(?:(?:aero|asia|biz|cat|com|coop|edu|gov|info|int|jobs|mil|mobi|museum|name|net|org|pro|tel|travel|xxx)(?=[^0-9a-zA-Z]|$))";
90 private const string url_valid_CCTLD = @"(?:(?:ac|ad|ae|af|ag|ai|al|am|an|ao|aq|ar|as|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|bi|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|co|cr|cs|cu|cv|cx|cy|cz|dd|de|dj|dk|dm|do|dz|ec|ee|eg|eh|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|in|io|iq|ir|is|it|je|jm|jo|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mk|ml|mm|mn|mo|mp|mq|mr|ms|mt|mu|mv|mw|mx|my|mz|na|nc|ne|nf|ng|ni|nl|no|np|nr|nu|nz|om|pa|pe|pf|pg|ph|pk|pl|pm|pn|pr|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|ss|st|su|sv|sy|sz|tc|td|tf|tg|th|tj|tk|tl|tm|tn|to|tp|tr|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|ye|yt|za|zm|zw)(?=[^0-9a-zA-Z]|$))";
91 private const string url_valid_punycode = @"(?:xn--[0-9a-z]+)";
92 private const string url_valid_domain = @"(?<domain>" + url_valid_subdomain + "*" + url_valid_domain_name + "(?:" + url_valid_GTLD + "|" + url_valid_CCTLD + ")|" + url_valid_punycode + ")";
93 public const string url_valid_ascii_domain = @"(?:(?:[a-z0-9" + LATIN_ACCENTS + @"]+)\.)+(?:" + url_valid_GTLD + "|" + url_valid_CCTLD + "|" + url_valid_punycode + ")";
94 public const string url_invalid_short_domain = "^" + url_valid_domain_name + url_valid_CCTLD + "$";
95 private const string url_valid_port_number = @"[0-9]+";
97 private const string url_valid_general_path_chars = @"[a-z0-9!*';:=+,.$/%#\[\]\-_~|&" + LATIN_ACCENTS + "]";
98 private const string url_balance_parens = @"(?:\(" + url_valid_general_path_chars + @"+\))";
99 private const string url_valid_path_ending_chars = @"(?:[+\-a-z0-9=_#/" + LATIN_ACCENTS + "]|" + url_balance_parens + ")";
100 private const string pth = "(?:" +
102 url_valid_general_path_chars + "*" +
103 "(?:" + url_balance_parens + url_valid_general_path_chars + "*)*" +
104 url_valid_path_ending_chars +
105 ")|(?:@" + url_valid_general_path_chars + "+/)" +
107 private const string qry = @"(?<query>\?[a-z0-9!?*'();:&=+$/%#\[\]\-_.,~|]*[a-z0-9_&=#/])?";
108 public const string rgUrl = @"(?<before>" + url_valid_preceding_chars + ")" +
109 "(?<url>(?<protocol>https?://)?" +
110 "(?<domain>" + url_valid_domain + ")" +
111 "(?::" + url_valid_port_number + ")?" +
112 "(?<path>/" + pth + "*)?" +
119 /// Twitter API のステータスページのURL
121 public const string ServiceAvailabilityStatusUrl = "https://status.io.watchmouse.com/7617";
124 /// ツイートへのパーマリンクURLを判定する正規表現
126 public static readonly Regex StatusUrlRegex = new Regex(@"https?://([^.]+\.)?twitter\.com/(#!/)?(?<ScreenName>[a-zA-Z0-9_]+)/status(es)?/(?<StatusId>[0-9]+)(/photo)?", RegexOptions.IgnoreCase);
129 /// FavstarやaclogなどTwitter関連サービスのパーマリンクURLからステータスIDを抽出する正規表現
131 public static readonly Regex ThirdPartyStatusUrlRegex = new Regex(@"https?://(?:[^.]+\.)?(?:
132 favstar\.fm/users/[a-zA-Z0-9_]+/status/ # Favstar
133 | favstar\.fm/t/ # Favstar (short)
134 | aclog\.koba789\.com/i/ # aclog
135 | frtrt\.net/solo_status\.php\?status= # RtRT
136 )(?<StatusId>[0-9]+)", RegexOptions.IgnoreCase | RegexOptions.IgnorePatternWhitespace);
138 delegate void GetIconImageDelegate(PostClass post);
139 private readonly object LockObj = new object();
140 private List<long> followerId = new List<long>();
141 private bool _GetFollowerResult = false;
142 private long[] noRTId = new long[0];
143 private bool _GetNoRetweetResult = false;
145 private int _followersCount = 0;
146 private int _friendsCount = 0;
147 private int _statusesCount = 0;
148 private string _location = "";
149 private string _bio = "";
152 private string _uname;
154 private bool _tinyUrlResolve;
155 private bool _restrictFavCheck;
157 private bool _readOwnPost;
158 private List<string> _hashList = new List<string>();
160 //max_idで古い発言を取得するために保持(lists分は個別タブで管理)
161 private long minHomeTimeline = long.MaxValue;
162 private long minMentions = long.MaxValue;
163 private long minDirectmessage = long.MaxValue;
164 private long minDirectmessageSent = long.MaxValue;
166 //private FavoriteQueue favQueue;
168 private HttpTwitter twCon = new HttpTwitter();
170 //private List<PostClass> _deletemessages = new List<PostClass>();
172 public TwitterApiAccessLevel AccessLevel
176 return MyCommon.TwitterApiInfo.AccessLevel;
180 protected void ResetApiStatus()
182 MyCommon.TwitterApiInfo.Reset();
185 public string Authenticate(string username, string password)
187 this.ResetApiStatus();
193 res = twCon.AuthUserAndPass(username, password, ref content);
197 return "Err:" + ex.Message;
200 var err = this.CheckStatusCode(res, content);
201 if (err != null) return err;
203 _uname = username.ToLower();
204 if (AppendSettingDialog.Instance.UserstreamStartup) this.ReconnectUserStream();
208 public string StartAuthentication(ref string pinPageUrl)
213 this.ResetApiStatus();
216 res = twCon.AuthGetRequestToken(ref pinPageUrl);
220 return "Err:" + "Failed to access auth server.";
226 public string Authenticate(string pinCode)
228 this.ResetApiStatus();
233 res = twCon.AuthGetAccessToken(pinCode);
237 return "Err:" + "Failed to access auth acc server.";
240 var err = this.CheckStatusCode(res, null);
241 if (err != null) return err;
243 _uname = Username.ToLower();
244 if (AppendSettingDialog.Instance.UserstreamStartup) this.ReconnectUserStream();
248 public void ClearAuthInfo()
250 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Invalid;
251 this.ResetApiStatus();
252 twCon.ClearAuthInfo();
255 public void VerifyCredentials()
261 res = twCon.VerifyCredentials(ref content);
268 if (res == HttpStatusCode.OK)
270 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Valid;
274 user = TwitterUser.ParseJson(content);
276 catch(SerializationException)
280 twCon.AuthenticatedUserId = user.Id;
284 public void Initialize(string token, string tokenSecret, string username, long userId)
287 if (string.IsNullOrEmpty(token) || string.IsNullOrEmpty(tokenSecret) || string.IsNullOrEmpty(username))
289 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Invalid;
291 this.ResetApiStatus();
292 twCon.Initialize(token, tokenSecret, username, userId);
293 _uname = username.ToLower();
294 if (AppendSettingDialog.Instance.UserstreamStartup) this.ReconnectUserStream();
297 public string PreProcessUrl(string orgData)
301 //var IDNConveter = new IdnMapping();
302 var href = "<a href=\"";
306 if (orgData.IndexOf(href, posl2, StringComparison.Ordinal) > -1)
310 posl1 = orgData.IndexOf(href, posl2, StringComparison.Ordinal);
311 posl1 += href.Length;
312 posl2 = orgData.IndexOf("\"", posl1, StringComparison.Ordinal);
313 urlStr = orgData.Substring(posl1, posl2 - posl1);
315 if (!urlStr.StartsWith("http://") && !urlStr.StartsWith("https://") && !urlStr.StartsWith("ftp://"))
320 var replacedUrl = MyCommon.IDNEncode(urlStr);
321 if (replacedUrl == null) continue;
322 if (replacedUrl == urlStr) continue;
324 orgData = orgData.Replace("<a href=\"" + urlStr, "<a href=\"" + replacedUrl);
335 private string GetPlainText(string orgData)
337 return WebUtility.HtmlDecode(Regex.Replace(orgData, "(?<tagStart><a [^>]+>)(?<text>[^<]+)(?<tagEnd></a>)", "${text}"));
340 // htmlの簡易サニタイズ(詳細表示に不要なタグの除去)
342 private string SanitizeHtml(string orgdata)
344 var retdata = orgdata;
346 retdata = Regex.Replace(retdata, "<(script|object|applet|image|frameset|fieldset|legend|style).*" +
347 "</(script|object|applet|image|frameset|fieldset|legend|style)>", "", RegexOptions.IgnoreCase);
349 retdata = Regex.Replace(retdata, "<(frame|link|iframe|img)>", "", RegexOptions.IgnoreCase);
354 private string AdjustHtml(string orgData)
356 var retStr = orgData;
357 //var m = Regex.Match(retStr, "<a [^>]+>[#|#](?<1>[a-zA-Z0-9_]+)</a>");
362 // _hashList.Add("#" + m.Groups(1).Value);
366 retStr = Regex.Replace(retStr, "<a [^>]*href=\"/", "<a href=\"https://twitter.com/");
367 retStr = retStr.Replace("<a href=", "<a target=\"_self\" href=");
368 retStr = Regex.Replace(retStr, @"(\r\n?|\n)", "<br>"); // CRLF, CR, LF は全て <br> に置換する
370 //半角スペースを置換(Thanks @anis774)
374 ret = EscapeSpace(ref retStr);
377 return SanitizeHtml(retStr);
380 private bool EscapeSpace(ref string html)
382 //半角スペースを置換(Thanks @anis774)
384 for (int i = 0; i < html.Length; i++)
395 if ((!isTag) && (html[i] == ' '))
397 html = html.Remove(i, 1);
398 html = html.Insert(i, " ");
405 private struct PostInfo
407 public string CreatedAt;
410 public string UserId;
411 public PostInfo(string Created, string IdStr, string txt, string uid)
418 public bool Equals(PostInfo dst)
420 if (this.CreatedAt == dst.CreatedAt && this.Id == dst.Id && this.Text == dst.Text && this.UserId == dst.UserId)
431 static private PostInfo _prev = new PostInfo("", "", "", "");
432 private bool IsPostRestricted(TwitterStatus status)
434 var _current = new PostInfo("", "", "", "");
436 _current.CreatedAt = status.CreatedAt;
437 _current.Id = status.IdStr;
438 if (status.Text == null)
444 _current.Text = status.Text;
446 _current.UserId = status.User.IdStr;
448 if (_current.Equals(_prev))
452 _prev.CreatedAt = _current.CreatedAt;
453 _prev.Id = _current.Id;
454 _prev.Text = _current.Text;
455 _prev.UserId = _current.UserId;
460 public string PostStatus(string postStr, long? reply_to)
462 if (MyCommon._endingFlag) return "";
464 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
466 if (Regex.Match(postStr, "^DM? +(?<id>[a-zA-Z0-9_]+) +(?<body>.+)", RegexOptions.IgnoreCase | RegexOptions.Singleline).Success)
468 return SendDirectMessage(postStr);
475 res = twCon.UpdateStatus(postStr, reply_to, ref content);
479 return "Err:" + ex.Message;
482 // 投稿に成功していても404が返ることがあるらしい: https://dev.twitter.com/discussions/1213
483 if (res == HttpStatusCode.NotFound) return "";
485 var err = this.CheckStatusCode(res, content);
486 if (err != null) return err;
488 TwitterStatus status;
491 status = TwitterStatus.ParseJson(content);
493 catch(SerializationException ex)
495 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
496 return "Err:Json Parse Error(DataContractJsonSerializer)";
500 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
501 return "Err:Invalid Json!";
503 _followersCount = status.User.FollowersCount;
504 _friendsCount = status.User.FriendsCount;
505 _statusesCount = status.User.StatusesCount;
506 _location = status.User.Location;
507 _bio = status.User.Description;
509 if (IsPostRestricted(status))
511 return "OK:Delaying?";
516 public string PostStatusWithMedia(string postStr, long? reply_to, FileInfo mediaFile)
518 if (MyCommon._endingFlag) return "";
520 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
526 res = twCon.UpdateStatusWithMedia(postStr, reply_to, mediaFile, ref content);
530 return "Err:" + ex.Message;
533 // 投稿に成功していても404が返ることがあるらしい: https://dev.twitter.com/discussions/1213
534 if (res == HttpStatusCode.NotFound) return "";
536 var err = this.CheckStatusCode(res, content);
537 if (err != null) return err;
539 TwitterStatus status;
542 status = TwitterStatus.ParseJson(content);
544 catch(SerializationException ex)
546 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
547 return "Err:Json Parse Error(DataContractJsonSerializer)";
551 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
552 return "Err:Invalid Json!";
554 _followersCount = status.User.FollowersCount;
555 _friendsCount = status.User.FriendsCount;
556 _statusesCount = status.User.StatusesCount;
557 _location = status.User.Location;
558 _bio = status.User.Description;
560 if (IsPostRestricted(status))
562 return "OK:Delaying?";
567 public string SendDirectMessage(string postStr)
569 if (MyCommon._endingFlag) return "";
571 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
573 if (this.AccessLevel == TwitterApiAccessLevel.Read || this.AccessLevel == TwitterApiAccessLevel.ReadWrite)
575 return "Auth Err:try to re-authorization.";
578 var mc = Regex.Match(postStr, "^DM? +(?<id>[a-zA-Z0-9_]+) +(?<body>.+)", RegexOptions.IgnoreCase | RegexOptions.Singleline);
584 res = twCon.SendDirectMessage(mc.Groups["body"].Value, mc.Groups["id"].Value, ref content);
588 return "Err:" + ex.Message;
591 var err = this.CheckStatusCode(res, content);
592 if (err != null) return err;
594 TwitterDirectMessage status;
597 status = TwitterDirectMessage.ParseJson(content);
599 catch(SerializationException ex)
601 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
602 return "Err:Json Parse Error(DataContractJsonSerializer)";
606 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
607 return "Err:Invalid Json!";
609 _followersCount = status.Sender.FollowersCount;
610 _friendsCount = status.Sender.FriendsCount;
611 _statusesCount = status.Sender.StatusesCount;
612 _location = status.Sender.Location;
613 _bio = status.Sender.Description;
618 public string RemoveStatus(long id)
620 if (MyCommon._endingFlag) return "";
622 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
627 res = twCon.DestroyStatus(id);
631 return "Err:" + ex.Message;
634 return this.CheckStatusCode(res, null) ?? "";
637 public string PostRetweet(long id, bool read)
639 if (MyCommon._endingFlag) return "";
640 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
644 var post = TabInformations.GetInstance()[id];
647 return "Err:Target isn't found.";
649 if (TabInformations.GetInstance()[id].RetweetedId != null)
651 target = TabInformations.GetInstance()[id].RetweetedId.Value; //再RTの場合は元発言をRT
658 res = twCon.RetweetStatus(target, ref content);
662 return "Err:" + ex.Message;
665 var err = this.CheckStatusCode(res, content);
666 if (err != null) return err;
668 TwitterStatus status;
671 status = TwitterStatus.ParseJson(content);
673 catch(SerializationException ex)
675 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
676 return "Err:Json Parse Error(DataContractJsonSerializer)";
680 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
681 return "Err:Invalid Json!";
685 post = CreatePostsFromStatusData(status);
686 if (post == null) return "Invalid Json!";
691 if (TabInformations.GetInstance().ContainsKey(post.StatusId)) return "";
694 if (post.RetweetedId == null) return "Invalid Json!";
700 if (_readOwnPost) post.IsRead = true;
703 TabInformations.GetInstance().AddPost(post);
708 public string RemoveDirectMessage(long id, PostClass post)
710 if (MyCommon._endingFlag) return "";
712 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
714 if (this.AccessLevel == TwitterApiAccessLevel.Read || this.AccessLevel == TwitterApiAccessLevel.ReadWrite)
716 return "Auth Err:try to re-authorization.";
720 // _deletemessages.Add(post)
726 res = twCon.DestroyDirectMessage(id);
730 return "Err:" + ex.Message;
733 return this.CheckStatusCode(res, null) ?? "";
736 public string PostFollowCommand(string screenName)
738 if (MyCommon._endingFlag) return "";
740 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
746 res = twCon.CreateFriendships(screenName, ref content);
750 return "Err:" + ex.Message;
753 return this.CheckStatusCode(res, content) ?? "";
756 public string PostRemoveCommand(string screenName)
758 if (MyCommon._endingFlag) return "";
760 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
766 res = twCon.DestroyFriendships(screenName, ref content);
770 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
773 return this.CheckStatusCode(res, content) ?? "";
776 public string PostCreateBlock(string screenName)
778 if (MyCommon._endingFlag) return "";
780 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
786 res = twCon.CreateBlock(screenName, ref content);
790 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
793 return this.CheckStatusCode(res, content) ?? "";
796 public string PostDestroyBlock(string screenName)
798 if (MyCommon._endingFlag) return "";
800 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
806 res = twCon.DestroyBlock(screenName, ref content);
810 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
813 return this.CheckStatusCode(res, content) ?? "";
816 public string PostReportSpam(string screenName)
818 if (MyCommon._endingFlag) return "";
820 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
826 res = twCon.ReportSpam(screenName, ref content);
830 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
833 return this.CheckStatusCode(res, content) ?? "";
836 public string GetFriendshipInfo(string screenName, ref bool isFollowing, ref bool isFollowed)
838 if (MyCommon._endingFlag) return "";
840 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
846 res = twCon.ShowFriendships(_uname, screenName, ref content);
850 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
853 var err = this.CheckStatusCode(res, content);
854 if (err != null) return err;
858 var friendship = TwitterFriendship.ParseJson(content);
859 isFollowing = friendship.Relationship.Source.Following;
860 isFollowed = friendship.Relationship.Source.FollowedBy;
863 catch(SerializationException ex)
865 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
866 return "Err:Json Parse Error(DataContractJsonSerializer)";
870 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
871 return "Err:Invalid Json!";
875 public string GetUserInfo(string screenName, ref TwitterUser user)
877 if (MyCommon._endingFlag) return "";
879 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
887 res = twCon.ShowUserInfo(screenName, ref content);
891 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
894 var err = this.CheckStatusCode(res, content);
895 if (err != null) return err;
899 user = TwitterUser.ParseJson(content);
901 catch (SerializationException ex)
903 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
904 return "Err:Json Parse Error(DataContractJsonSerializer)";
908 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
909 return "Err:Invalid Json!";
914 public string GetStatus_Retweeted_Count(long StatusId, ref int retweeted_count)
916 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
918 if (MyCommon._endingFlag) return "";
924 res = twCon.ShowStatuses(StatusId, ref content);
928 return "Err:" + ex.Message;
931 var err = this.CheckStatusCode(res, content);
932 if (err != null) return err;
934 TwitterStatus status;
937 status = TwitterStatus.ParseJson(content);
939 catch (SerializationException ex)
941 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
942 return "Json Parse Error(DataContractJsonSerializer)";
946 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
947 return "Invalid Json!";
949 retweeted_count = status.RetweetCount;
953 public string PostFavAdd(long id)
955 if (MyCommon._endingFlag) return "";
957 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
959 //if (this.favQueue == null) this.favQueue = new FavoriteQueue(this)
961 //if (this.favQueue.Contains(id)) this.favQueue.Remove(id)
967 res = twCon.CreateFavorites(id, ref content);
971 //this.favQueue.Add(id)
972 //return "Err:->FavoriteQueue:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
973 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
976 var err = this.CheckStatusCode(res, content);
977 if (err != null) return err;
979 if (!_restrictFavCheck) return "";
981 //http://twitter.com/statuses/show/id.xml APIを発行して本文を取得
985 res = twCon.ShowStatuses(id, ref content);
989 return "Err:" + ex.Message;
992 err = this.CheckStatusCode(res, content);
993 if (err != null) return err;
995 TwitterStatus status;
998 status = TwitterStatus.ParseJson(content);
1000 catch (SerializationException ex)
1002 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
1003 return "Err:Json Parse Error(DataContractJsonSerializer)";
1005 catch (Exception ex)
1007 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
1008 return "Err:Invalid Json!";
1010 if (status.Favorited == true)
1016 return "NG(Restricted?)";
1020 public string PostFavRemove(long id)
1022 if (MyCommon._endingFlag) return "";
1024 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
1026 //if (this.favQueue == null) this.favQueue = new FavoriteQueue(this)
1028 //if (this.favQueue.Contains(id))
1029 // this.favQueue.Remove(id)
1037 res = twCon.DestroyFavorites(id, ref content);
1041 return "Err:" + ex.Message;
1044 return this.CheckStatusCode(res, content) ?? "";
1047 public TwitterUser PostUpdateProfile(string name, string url, string location, string description)
1049 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid)
1050 throw new WebApiException("AccountState invalid");
1056 res = twCon.UpdateProfile(name, url, location, description, ref content);
1060 throw new WebApiException("Err:" + ex.Message, content, ex);
1063 var err = this.CheckStatusCode(res, content);
1065 throw new WebApiException(err, content);
1069 return TwitterUser.ParseJson(content);
1071 catch (SerializationException e)
1073 var ex = new WebApiException("Err:Json Parse Error(DataContractJsonSerializer)", content, e);
1074 MyCommon.TraceOut(ex);
1079 var ex = new WebApiException("Err:Invalid Json!", content, e);
1080 MyCommon.TraceOut(ex);
1085 public string PostUpdateProfileImage(string filename)
1087 if (MyCommon._endingFlag) return "";
1089 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
1095 res = twCon.UpdateProfileImage(new FileInfo(filename), ref content);
1099 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
1102 return this.CheckStatusCode(res, content) ?? "";
1105 public string Username
1109 return twCon.AuthenticatedUsername;
1117 return twCon.AuthenticatedUserId;
1121 public string Password
1125 return twCon.Password;
1129 private static MyCommon.ACCOUNT_STATE _accountState = MyCommon.ACCOUNT_STATE.Valid;
1130 public static MyCommon.ACCOUNT_STATE AccountState
1134 return _accountState;
1138 _accountState = value;
1142 public bool TinyUrlResolve
1146 _tinyUrlResolve = value;
1150 public bool RestrictFavCheck
1154 _restrictFavCheck = value;
1159 public string GetTweenBinary(string strVer)
1164 if (!(new HttpVarious()).GetDataToFile("http://tween.sourceforge.jp/Tween" + strVer + ".gz?" + DateTime.Now.ToString("yyMMddHHmmss") + Environment.TickCount.ToString(),
1165 Path.Combine(MyCommon.settingPath, "TweenNew.exe")))
1167 return "Err:Download failed";
1170 if (!Directory.Exists(Path.Combine(MyCommon.settingPath, "en")))
1172 Directory.CreateDirectory(Path.Combine(MyCommon.settingPath, "en"));
1174 if (!(new HttpVarious()).GetDataToFile("http://tween.sourceforge.jp/TweenResEn" + strVer + ".gz?" + DateTime.Now.ToString("yyMMddHHmmss") + Environment.TickCount.ToString(),
1175 Path.Combine(Path.Combine(MyCommon.settingPath, "en"), "Tween.resourcesNew.dll")))
1177 return "Err:Download failed";
1179 //その他言語圏のリソース。取得失敗しても継続
1182 if (!Thread.CurrentThread.CurrentUICulture.IsNeutralCulture)
1184 var idx = Thread.CurrentThread.CurrentUICulture.Name.LastIndexOf('-');
1187 curCul = Thread.CurrentThread.CurrentUICulture.Name.Substring(0, idx);
1191 curCul = Thread.CurrentThread.CurrentUICulture.Name;
1196 curCul = Thread.CurrentThread.CurrentUICulture.Name;
1198 if (!string.IsNullOrEmpty(curCul) && curCul != "en" && curCul != "ja")
1200 if (!Directory.Exists(Path.Combine(MyCommon.settingPath, curCul)))
1202 Directory.CreateDirectory(Path.Combine(MyCommon.settingPath, curCul));
1204 if (!(new HttpVarious()).GetDataToFile("http://tween.sourceforge.jp/TweenRes" + curCul + strVer + ".gz?" + DateTime.Now.ToString("yyMMddHHmmss") + Environment.TickCount.ToString(),
1205 Path.Combine(Path.Combine(MyCommon.settingPath, curCul), "Tween.resourcesNew.dll")))
1207 //return "Err:Download failed";
1212 if (!Thread.CurrentThread.CurrentCulture.IsNeutralCulture)
1214 var idx = Thread.CurrentThread.CurrentCulture.Name.LastIndexOf('-');
1217 curCul2 = Thread.CurrentThread.CurrentCulture.Name.Substring(0, idx);
1221 curCul2 = Thread.CurrentThread.CurrentCulture.Name;
1226 curCul2 = Thread.CurrentThread.CurrentCulture.Name;
1228 if (!string.IsNullOrEmpty(curCul2) && curCul2 != "en" && curCul2 != curCul)
1230 if (!Directory.Exists(Path.Combine(MyCommon.settingPath, curCul2)))
1232 Directory.CreateDirectory(Path.Combine(MyCommon.settingPath, curCul2));
1234 if (!(new HttpVarious()).GetDataToFile("http://tween.sourceforge.jp/TweenRes" + curCul2 + strVer + ".gz?" + DateTime.Now.ToString("yyMMddHHmmss") + Environment.TickCount.ToString(),
1235 Path.Combine(Path.Combine(MyCommon.settingPath, curCul2), "Tween.resourcesNew.dll")))
1237 //return "Err:Download failed";
1242 if (!(new HttpVarious()).GetDataToFile("http://tween.sourceforge.jp/TweenUp3.gz?" + DateTime.Now.ToString("yyMMddHHmmss") + Environment.TickCount.ToString(),
1243 Path.Combine(MyCommon.settingPath, "TweenUp3.exe")))
1245 return "Err:Download failed";
1248 if (!(new HttpVarious()).GetDataToFile("http://tween.sourceforge.jp/TweenDll" + strVer + ".gz?" + DateTime.Now.ToString("yyMMddHHmmss") + Environment.TickCount.ToString(),
1249 Path.Combine(MyCommon.settingPath, "TweenNew.XmlSerializers.dll")))
1251 return "Err:Download failed";
1257 return "Err:Download failed";
1262 public bool ReadOwnPost
1266 return _readOwnPost;
1270 _readOwnPost = value;
1274 public int FollowersCount
1278 return _followersCount;
1282 public int FriendsCount
1286 return _friendsCount;
1290 public int StatusesCount
1294 return _statusesCount;
1298 public string Location
1314 public string GetTimelineApi(bool read,
1315 MyCommon.WORKERTYPE gType,
1319 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
1321 if (MyCommon._endingFlag) return "";
1325 var count = AppendSettingDialog.Instance.CountApi;
1326 if (gType == MyCommon.WORKERTYPE.Reply) count = AppendSettingDialog.Instance.CountApiReply;
1327 if (AppendSettingDialog.Instance.UseAdditionalCount)
1329 if (more && AppendSettingDialog.Instance.MoreCountApi != 0)
1331 count = AppendSettingDialog.Instance.MoreCountApi;
1333 else if (startup && AppendSettingDialog.Instance.FirstCountApi != 0 && gType == MyCommon.WORKERTYPE.Timeline)
1335 count = AppendSettingDialog.Instance.FirstCountApi;
1340 if (gType == MyCommon.WORKERTYPE.Timeline)
1344 res = twCon.HomeTimeline(count, this.minHomeTimeline, null, ref content);
1348 res = twCon.HomeTimeline(count, null, null, ref content);
1355 res = twCon.Mentions(count, this.minMentions, null, ref content);
1359 res = twCon.Mentions(count, null, null, ref content);
1365 return "Err:" + ex.Message;
1368 var err = this.CheckStatusCode(res, content);
1369 if (err != null) return err;
1371 if (gType == MyCommon.WORKERTYPE.Timeline)
1373 return CreatePostsFromJson(content, gType, null, read, count, ref this.minHomeTimeline);
1377 return CreatePostsFromJson(content, gType, null, read, count, ref this.minMentions);
1381 public string GetUserTimelineApi(bool read,
1387 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
1389 if (MyCommon._endingFlag) return "";
1394 if (count == 0) count = 20;
1397 if (string.IsNullOrEmpty(userName))
1399 var target = tab.User;
1400 if (string.IsNullOrEmpty(target)) return "";
1402 res = twCon.UserTimeline(null, target, count, null, null, ref content);
1408 res = twCon.UserTimeline(null, userName, count, tab.OldestId, null, ref content);
1412 res = twCon.UserTimeline(null, userName, count, null, null, ref content);
1418 return "Err:" + ex.Message;
1421 if (res == HttpStatusCode.Unauthorized)
1422 return "Err:@" + userName + "'s Tweets are protected.";
1424 var err = this.CheckStatusCode(res, content);
1425 if (err != null) return err;
1427 TwitterStatus[] items;
1430 items = TwitterStatus.ParseJsonArray(content);
1432 catch(SerializationException ex)
1434 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
1435 return "Json Parse Error(DataContractJsonSerializer)";
1439 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
1440 return "Invalid Json!";
1443 foreach (var status in items)
1445 var item = CreatePostsFromStatusData(status);
1446 if (item == null) continue;
1447 if (item.StatusId < tab.OldestId) tab.OldestId = item.StatusId;
1449 if (item.IsMe && !read && _readOwnPost) item.IsRead = true;
1450 if (tab != null) item.RelTabName = tab.TabName;
1451 //非同期アイコン取得&StatusDictionaryに追加
1452 TabInformations.GetInstance().AddPost(item);
1458 public string GetStatusApi(bool read,
1462 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
1464 if (MyCommon._endingFlag) return "";
1470 res = twCon.ShowStatuses(id, ref content);
1474 return "Err:" + ex.Message;
1477 if (res == HttpStatusCode.Forbidden)
1478 return "Err:protected user's tweet";
1480 var err = this.CheckStatusCode(res, content);
1481 if (err != null) return err;
1483 TwitterStatus status;
1486 status = TwitterStatus.ParseJson(content);
1488 catch(SerializationException ex)
1490 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
1491 return "Json Parse Error(DataContractJsonSerializer)";
1495 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
1496 return "Invalid Json!";
1499 var item = CreatePostsFromStatusData(status);
1500 if (item == null) return "Err:Can't create post";
1502 if (item.IsMe && !read && _readOwnPost) item.IsRead = true;
1508 public string GetStatusApi(bool read,
1512 PostClass post = null;
1513 var r = this.GetStatusApi(read, id, ref post);
1515 if (string.IsNullOrEmpty(r))
1517 if (tab != null) post.RelTabName = tab.TabName;
1518 //非同期アイコン取得&StatusDictionaryに追加
1519 TabInformations.GetInstance().AddPost(post);
1525 private PostClass CreatePostsFromStatusData(TwitterStatus status)
1527 var post = new PostClass();
1528 TwitterEntities entities;
1530 post.StatusId = status.Id;
1531 if (status.RetweetedStatus != null)
1533 var retweeted = status.RetweetedStatus;
1535 post.CreatedAt = MyCommon.DateTimeParse(retweeted.CreatedAt);
1538 post.RetweetedId = retweeted.Id;
1540 post.TextFromApi = retweeted.Text;
1541 entities = retweeted.MergedEntities;
1542 //Source取得(htmlの場合は、中身を取り出し)
1543 post.Source = retweeted.Source;
1545 post.InReplyToStatusId = retweeted.InReplyToStatusId;
1546 post.InReplyToUser = retweeted.InReplyToScreenName;
1547 post.InReplyToUserId = status.InReplyToUserId;
1550 var tc = TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites);
1551 post.IsFav = tc.Contains(retweeted.Id);
1553 if (retweeted.Coordinates != null) post.PostGeo = new PostClass.StatusGeo { Lng = retweeted.Coordinates.Coordinates[0], Lat = retweeted.Coordinates.Coordinates[1] };
1556 var user = retweeted.User;
1558 if (user == null || user.ScreenName == null || status.User.ScreenName == null) return null;
1560 post.UserId = user.Id;
1561 post.ScreenName = user.ScreenName;
1562 post.Nickname = user.Name.Trim();
1563 post.ImageUrl = user.ProfileImageUrlHttps;
1564 post.IsProtect = user.Protected;
1567 post.RetweetedBy = status.User.ScreenName;
1568 post.RetweetedByUserId = status.User.Id;
1569 post.IsMe = post.RetweetedBy.ToLower().Equals(_uname);
1573 post.CreatedAt = MyCommon.DateTimeParse(status.CreatedAt);
1575 post.TextFromApi = status.Text;
1576 entities = status.MergedEntities;
1577 //Source取得(htmlの場合は、中身を取り出し)
1578 post.Source = status.Source;
1579 post.InReplyToStatusId = status.InReplyToStatusId;
1580 post.InReplyToUser = status.InReplyToScreenName;
1581 post.InReplyToUserId = status.InReplyToUserId;
1583 if (status.Coordinates != null) post.PostGeo = new PostClass.StatusGeo { Lng = status.Coordinates.Coordinates[0], Lat = status.Coordinates.Coordinates[1] };
1586 var user = status.User;
1588 if (user == null || user.ScreenName == null) return null;
1590 post.UserId = user.Id;
1591 post.ScreenName = user.ScreenName;
1592 post.Nickname = user.Name.Trim();
1593 post.ImageUrl = user.ProfileImageUrlHttps;
1594 post.IsProtect = user.Protected;
1595 post.IsMe = post.ScreenName.ToLower().Equals(_uname);
1598 var tc = TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites);
1599 post.IsFav = tc.Contains(post.StatusId) && TabInformations.GetInstance()[post.StatusId].IsFav;
1602 string textFromApi = post.TextFromApi;
1603 post.Text = CreateHtmlAnchor(textFromApi, post.ReplyToList, entities, post.Media);
1604 post.TextFromApi = textFromApi;
1605 post.TextFromApi = this.ReplaceTextFromApi(post.TextFromApi, entities);
1606 post.TextFromApi = WebUtility.HtmlDecode(post.TextFromApi);
1607 post.TextFromApi = post.TextFromApi.Replace("<3", "\u2661");
1612 post.IsReply = post.ReplyToList.Contains(_uname);
1613 post.IsExcludeReply = false;
1621 if (followerId.Count > 0) post.IsOwl = !followerId.Contains(post.UserId);
1628 private string CreatePostsFromJson(string content, MyCommon.WORKERTYPE gType, TabClass tab, bool read, int count, ref long minimumId)
1630 TwitterStatus[] items;
1633 items = TwitterStatus.ParseJsonArray(content);
1635 catch(SerializationException ex)
1637 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
1638 return "Json Parse Error(DataContractJsonSerializer)";;
1642 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
1643 return "Invalid Json!";
1646 foreach (var status in items)
1648 PostClass post = null;
1649 post = CreatePostsFromStatusData(status);
1650 if (post == null) continue;
1652 if (minimumId > post.StatusId) minimumId = post.StatusId;
1658 if (TabInformations.GetInstance().ContainsKey(post.StatusId)) continue;
1662 if (TabInformations.GetInstance().ContainsKey(post.StatusId, tab.TabName)) continue;
1667 if (post.RetweetedByUserId != null && this.noRTId.Contains(post.RetweetedByUserId.Value)) continue;
1670 if (post.IsMe && !read && _readOwnPost) post.IsRead = true;
1672 if (tab != null) post.RelTabName = tab.TabName;
1673 //非同期アイコン取得&StatusDictionaryに追加
1674 TabInformations.GetInstance().AddPost(post);
1680 private string CreatePostsFromSearchJson(string content, TabClass tab, bool read, int count, ref long minimumId, bool more)
1682 TwitterSearchResult items;
1685 items = TwitterSearchResult.ParseJson(content);
1687 catch (SerializationException ex)
1689 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
1690 return "Json Parse Error(DataContractJsonSerializer)";
1692 catch (Exception ex)
1694 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
1695 return "Invalid Json!";
1697 foreach (var result in items.Statuses)
1699 PostClass post = null;
1700 post = CreatePostsFromStatusData(result);
1704 // Search API は相変わらずぶっ壊れたデータを返すことがあるため、必要なデータが欠如しているものは取得し直す
1705 var ret = this.GetStatusApi(read, result.Id, ref post);
1706 if (!string.IsNullOrEmpty(ret)) continue;
1709 if (minimumId > post.StatusId) minimumId = post.StatusId;
1710 if (!more && post.StatusId > tab.SinceId) tab.SinceId = post.StatusId;
1716 if (TabInformations.GetInstance().ContainsKey(post.StatusId)) continue;
1720 if (TabInformations.GetInstance().ContainsKey(post.StatusId, tab.TabName)) continue;
1725 if ((post.IsMe && !read) && this._readOwnPost) post.IsRead = true;
1727 if (tab != null) post.RelTabName = tab.TabName;
1728 //非同期アイコン取得&StatusDictionaryに追加
1729 TabInformations.GetInstance().AddPost(post);
1735 public string GetListStatus(bool read,
1740 if (MyCommon._endingFlag) return "";
1745 if (AppendSettingDialog.Instance.UseAdditionalCount)
1747 count = AppendSettingDialog.Instance.ListCountApi;
1750 if (more && AppendSettingDialog.Instance.MoreCountApi != 0)
1752 count = AppendSettingDialog.Instance.MoreCountApi;
1754 else if (startup && AppendSettingDialog.Instance.FirstCountApi != 0)
1756 count = AppendSettingDialog.Instance.FirstCountApi;
1760 count = AppendSettingDialog.Instance.CountApi;
1766 count = AppendSettingDialog.Instance.CountApi;
1772 res = twCon.GetListsStatuses(tab.ListInfo.UserId, tab.ListInfo.Id, count, tab.OldestId, null, AppendSettingDialog.Instance.IsListStatusesIncludeRts, ref content);
1776 res = twCon.GetListsStatuses(tab.ListInfo.UserId, tab.ListInfo.Id, count, null, null, AppendSettingDialog.Instance.IsListStatusesIncludeRts, ref content);
1781 return "Err:" + ex.Message;
1784 var err = this.CheckStatusCode(res, content);
1785 if (err != null) return err;
1787 return CreatePostsFromJson(content, MyCommon.WORKERTYPE.List, tab, read, count, ref tab.OldestId);
1791 /// startStatusId からリプライ先の発言を辿る。発言は posts 以外からは検索しない。
1793 /// <returns>posts の中から検索されたリプライチェインの末端</returns>
1794 internal static PostClass FindTopOfReplyChain(IDictionary<Int64, PostClass> posts, Int64 startStatusId)
1796 if (!posts.ContainsKey(startStatusId))
1797 throw new ArgumentException("startStatusId (" + startStatusId + ") が posts の中から見つかりませんでした。");
1799 var nextPost = posts[startStatusId];
1800 while (nextPost.InReplyToStatusId != null)
1802 if (!posts.ContainsKey(nextPost.InReplyToStatusId.Value))
1804 nextPost = posts[nextPost.InReplyToStatusId.Value];
1810 public string GetRelatedResult(bool read, TabClass tab)
1813 var relPosts = new Dictionary<Int64, PostClass>();
1814 if (tab.RelationTargetPost.TextFromApi.Contains("@") && tab.RelationTargetPost.InReplyToStatusId == null)
1817 var p = TabInformations.GetInstance()[tab.RelationTargetPost.StatusId];
1818 if (p != null && p.InReplyToStatusId != null)
1820 tab.RelationTargetPost = p;
1824 rslt = this.GetStatusApi(read, tab.RelationTargetPost.StatusId, ref p);
1825 if (!string.IsNullOrEmpty(rslt)) return rslt;
1826 tab.RelationTargetPost = p;
1829 relPosts.Add(tab.RelationTargetPost.StatusId, tab.RelationTargetPost.Clone());
1831 // in_reply_to_status_id を使用してリプライチェインを辿る
1832 var nextPost = FindTopOfReplyChain(relPosts, tab.RelationTargetPost.StatusId);
1834 while (nextPost.InReplyToStatusId != null && loopCount++ <= 20)
1836 var inReplyToId = nextPost.InReplyToStatusId.Value;
1838 var inReplyToPost = TabInformations.GetInstance()[inReplyToId];
1839 if (inReplyToPost != null)
1841 inReplyToPost = inReplyToPost.Clone();
1845 var errorText = this.GetStatusApi(read, inReplyToId, ref inReplyToPost);
1846 if (!string.IsNullOrEmpty(errorText))
1853 relPosts.Add(inReplyToPost.StatusId, inReplyToPost);
1855 nextPost = FindTopOfReplyChain(relPosts, nextPost.StatusId);
1858 //MRTとかに対応のためツイート内にあるツイートを指すURLを取り込む
1859 var text = tab.RelationTargetPost.Text;
1860 var ma = Twitter.StatusUrlRegex.Matches(text).Cast<Match>()
1861 .Concat(Twitter.ThirdPartyStatusUrlRegex.Matches(text).Cast<Match>());
1862 foreach (var _match in ma)
1865 if (Int64.TryParse(_match.Groups["StatusId"].Value, out _statusId))
1867 if (relPosts.ContainsKey(_statusId))
1871 var _post = TabInformations.GetInstance()[_statusId];
1874 this.GetStatusApi(read, _statusId, ref p);
1882 relPosts.Add(p.StatusId, p);
1886 relPosts.Values.ToList().ForEach(p =>
1888 if (p.IsMe && !read && this._readOwnPost)
1893 p.RelTabName = tab.TabName;
1894 TabInformations.GetInstance().AddPost(p);
1900 public string GetSearch(bool read,
1904 if (MyCommon._endingFlag) return "";
1909 long? sinceId = null;
1911 if (AppendSettingDialog.Instance.UseAdditionalCount &&
1912 AppendSettingDialog.Instance.SearchCountApi != 0)
1914 count = AppendSettingDialog.Instance.SearchCountApi;
1918 count = AppendSettingDialog.Instance.CountApi;
1922 maxId = tab.OldestId - 1;
1926 sinceId = tab.SinceId;
1931 // TODO:一時的に40>100件に 件数変更UI作成の必要あり
1932 res = twCon.Search(tab.SearchWords, tab.SearchLang, count, maxId, sinceId, ref content);
1936 return "Err:" + ex.Message;
1940 case HttpStatusCode.BadRequest:
1941 return "Invalid query";
1942 case HttpStatusCode.NotFound:
1943 return "Invalid query";
1944 case HttpStatusCode.PaymentRequired: //API Documentには420と書いてあるが、該当コードがないので402にしてある
1945 return "Search API Limit?";
1946 case HttpStatusCode.OK:
1949 return "Err:" + res.ToString() + "(" + MethodBase.GetCurrentMethod().Name + ")";
1952 if (!TabInformations.GetInstance().ContainsTab(tab)) return "";
1954 return this.CreatePostsFromSearchJson(content, tab, read, count, ref tab.OldestId, more);
1957 private string CreateDirectMessagesFromJson(string content, MyCommon.WORKERTYPE gType, bool read)
1959 TwitterDirectMessage[] item;
1962 if (gType == MyCommon.WORKERTYPE.UserStream)
1964 item = new[] { TwitterStreamEventDirectMessage.ParseJson(content).DirectMessage };
1968 item = TwitterDirectMessage.ParseJsonArray(content);
1971 catch(SerializationException ex)
1973 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
1974 return "Json Parse Error(DataContractJsonSerializer)";
1978 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
1979 return "Invalid Json!";
1982 foreach (var message in item)
1984 var post = new PostClass();
1987 post.StatusId = message.Id;
1988 if (gType != MyCommon.WORKERTYPE.UserStream)
1990 if (gType == MyCommon.WORKERTYPE.DirectMessegeRcv)
1992 if (minDirectmessage > post.StatusId) minDirectmessage = post.StatusId;
1996 if (minDirectmessageSent > post.StatusId) minDirectmessageSent = post.StatusId;
2003 if (TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.DirectMessage).Contains(post.StatusId)) continue;
2007 post.CreatedAt = MyCommon.DateTimeParse(message.CreatedAt);
2009 var textFromApi = message.Text;
2011 post.Text = CreateHtmlAnchor(textFromApi, post.ReplyToList, message.Entities, post.Media);
2012 post.TextFromApi = this.ReplaceTextFromApi(textFromApi, message.Entities);
2013 post.TextFromApi = WebUtility.HtmlDecode(post.TextFromApi);
2014 post.TextFromApi = post.TextFromApi.Replace("<3", "\u2661");
2019 if (gType == MyCommon.WORKERTYPE.UserStream)
2021 if (twCon.AuthenticatedUsername.Equals(message.Recipient.ScreenName, StringComparison.CurrentCultureIgnoreCase))
2023 user = message.Sender;
2029 user = message.Recipient;
2036 if (gType == MyCommon.WORKERTYPE.DirectMessegeRcv)
2038 user = message.Sender;
2044 user = message.Recipient;
2050 post.UserId = user.Id;
2051 post.ScreenName = user.ScreenName;
2052 post.Nickname = user.Name.Trim();
2053 post.ImageUrl = user.ProfileImageUrlHttps;
2054 post.IsProtect = user.Protected;
2058 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2059 MessageBox.Show("Parse Error(CreateDirectMessagesFromJson)");
2064 if (post.IsMe && !read && _readOwnPost) post.IsRead = true;
2065 post.IsReply = false;
2066 post.IsExcludeReply = false;
2069 TabInformations.GetInstance().AddPost(post);
2076 public string GetDirectMessageApi(bool read,
2077 MyCommon.WORKERTYPE gType,
2080 if (MyCommon._endingFlag) return "";
2082 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
2084 if (this.AccessLevel == TwitterApiAccessLevel.Read || this.AccessLevel == TwitterApiAccessLevel.ReadWrite)
2086 return "Auth Err:try to re-authorization.";
2094 if (gType == MyCommon.WORKERTYPE.DirectMessegeRcv)
2098 res = twCon.DirectMessages(20, minDirectmessage, null, ref content);
2102 res = twCon.DirectMessages(20, null, null, ref content);
2109 res = twCon.DirectMessagesSent(20, minDirectmessageSent, null, ref content);
2113 res = twCon.DirectMessagesSent(20, null, null, ref content);
2119 return "Err:" + ex.Message;
2122 var err = this.CheckStatusCode(res, content);
2123 if (err != null) return err;
2125 return CreateDirectMessagesFromJson(content, gType, read);
2128 static int page_ = 1;
2129 public string GetFavoritesApi(bool read,
2130 MyCommon.WORKERTYPE gType,
2133 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
2135 if (MyCommon._endingFlag) return "";
2137 var count = AppendSettingDialog.Instance.CountApi;
2138 if (AppendSettingDialog.Instance.UseAdditionalCount &&
2139 AppendSettingDialog.Instance.FavoritesCountApi != 0)
2141 count = AppendSettingDialog.Instance.FavoritesCountApi;
2144 // 前ページ取得の場合はページカウンタをインクリメント、それ以外の場合はページカウンタリセット
2158 res = twCon.Favorites(count, page_, ref content);
2162 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2165 var err = this.CheckStatusCode(res, content);
2166 if (err != null) return err;
2168 TwitterStatus[] item;
2171 item = TwitterStatus.ParseJsonArray(content);
2173 catch(SerializationException ex)
2175 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
2176 return "Json Parse Error(DataContractJsonSerializer)";
2180 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2181 return "Invalid Json!";
2184 foreach (var status in item)
2186 var post = new PostClass();
2187 TwitterEntities entities;
2191 post.StatusId = status.Id;
2195 if (TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites).Contains(post.StatusId)) continue;
2198 if (status.RetweetedStatus != null)
2200 var retweeted = status.RetweetedStatus;
2201 post.CreatedAt = MyCommon.DateTimeParse(retweeted.CreatedAt);
2204 post.RetweetedId = post.StatusId;
2206 post.TextFromApi = retweeted.Text;
2207 entities = retweeted.MergedEntities;
2208 //Source取得(htmlの場合は、中身を取り出し)
2209 post.Source = retweeted.Source;
2211 post.InReplyToStatusId = retweeted.InReplyToStatusId;
2212 post.InReplyToUser = retweeted.InReplyToScreenName;
2213 post.InReplyToUserId = retweeted.InReplyToUserId;
2217 var user = retweeted.User;
2218 post.UserId = user.Id;
2219 post.ScreenName = user.ScreenName;
2220 post.Nickname = user.Name.Trim();
2221 post.ImageUrl = user.ProfileImageUrlHttps;
2222 post.IsProtect = user.Protected;
2225 post.RetweetedBy = status.User.ScreenName;
2226 post.IsMe = post.RetweetedBy.ToLower().Equals(_uname);
2230 post.CreatedAt = MyCommon.DateTimeParse(status.CreatedAt);
2233 post.TextFromApi = status.Text;
2234 entities = status.MergedEntities;
2235 //Source取得(htmlの場合は、中身を取り出し)
2236 post.Source = status.Source;
2237 post.InReplyToStatusId = status.InReplyToStatusId;
2238 post.InReplyToUser = status.InReplyToScreenName;
2239 post.InReplyToUserId = status.InReplyToUserId;
2244 var user = status.User;
2245 post.UserId = user.Id;
2246 post.ScreenName = user.ScreenName;
2247 post.Nickname = user.Name.Trim();
2248 post.ImageUrl = user.ProfileImageUrlHttps;
2249 post.IsProtect = user.Protected;
2250 post.IsMe = post.ScreenName.ToLower().Equals(_uname);
2253 string textFromApi = post.TextFromApi;
2254 post.Text = CreateHtmlAnchor(textFromApi, post.ReplyToList, entities, post.Media);
2255 post.TextFromApi = textFromApi;
2256 post.TextFromApi = this.ReplaceTextFromApi(post.TextFromApi, entities);
2257 post.TextFromApi = WebUtility.HtmlDecode(post.TextFromApi);
2258 post.TextFromApi = post.TextFromApi.Replace("<3", "\u2661");
2263 post.IsReply = post.ReplyToList.Contains(_uname);
2264 post.IsExcludeReply = false;
2272 if (followerId.Count > 0) post.IsOwl = !followerId.Contains(post.UserId);
2279 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2283 TabInformations.GetInstance().AddPost(post);
2290 private string ReplaceTextFromApi(string text, TwitterEntities entities)
2292 if (entities != null)
2294 if (entities.Urls != null)
2296 foreach (var m in entities.Urls)
2298 if (!string.IsNullOrEmpty(m.DisplayUrl)) text = text.Replace(m.Url, m.DisplayUrl);
2301 if (entities.Media != null)
2303 foreach (var m in entities.Media)
2305 if (!string.IsNullOrEmpty(m.DisplayUrl)) text = text.Replace(m.Url, m.DisplayUrl);
2315 /// <exception cref="WebApiException"/>
2316 public void RefreshFollowerIds()
2318 if (MyCommon._endingFlag) return;
2321 var newFollowerIds = new List<long>();
2324 var ret = this.GetFollowerIdsApi(ref cursor);
2325 newFollowerIds.AddRange(ret.Ids);
2326 cursor = ret.NextCursor;
2327 } while (cursor != 0);
2329 this.followerId = newFollowerIds;
2330 TabInformations.GetInstance().RefreshOwl(this.followerId);
2332 this._GetFollowerResult = true;
2335 public bool GetFollowersSuccess
2339 return _GetFollowerResult;
2343 private TwitterIds GetFollowerIdsApi(ref long cursor)
2345 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid)
2346 throw new WebApiException("AccountState invalid");
2352 res = twCon.FollowerIds(cursor, ref content);
2356 throw new WebApiException("Err:" + e.Message + "(" + MethodBase.GetCurrentMethod().Name + ")", e);
2359 var err = this.CheckStatusCode(res, content);
2361 throw new WebApiException(err, content);
2365 var ret = TwitterIds.ParseJson(content);
2367 if (ret.Ids == null)
2369 var ex = new WebApiException("Err: ret.id == null (GetFollowerIdsApi)", content);
2370 MyCommon.ExceptionOut(ex);
2376 catch(SerializationException e)
2378 var ex = new WebApiException("Err:Json Parse Error(DataContractJsonSerializer)", content, e);
2379 MyCommon.TraceOut(ex);
2384 var ex = new WebApiException("Err:Invalid Json!", content, e);
2385 MyCommon.TraceOut(ex);
2391 /// RT 非表示ユーザーを更新します
2393 /// <exception cref="WebApiException"/>
2394 public void RefreshNoRetweetIds()
2396 if (MyCommon._endingFlag) return;
2398 this.noRTId = this.NoRetweetIdsApi();
2400 this._GetNoRetweetResult = true;
2403 private long[] NoRetweetIdsApi()
2405 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid)
2406 throw new WebApiException("AccountState invalid");
2412 res = twCon.NoRetweetIds(ref content);
2416 throw new WebApiException("Err:" + e.Message + "(" + MethodBase.GetCurrentMethod().Name + ")", e);
2419 var err = this.CheckStatusCode(res, content);
2421 throw new WebApiException(err, content);
2425 return MyCommon.CreateDataFromJson<long[]>(content);
2427 catch(SerializationException e)
2429 var ex = new WebApiException("Err:Json Parse Error(DataContractJsonSerializer)", content, e);
2430 MyCommon.TraceOut(ex);
2435 var ex = new WebApiException("Err:Invalid Json!", content, e);
2436 MyCommon.TraceOut(ex);
2441 public bool GetNoRetweetSuccess
2445 return _GetNoRetweetResult;
2450 /// t.co の文字列長などの設定情報を取得します
2452 /// <exception cref="WebApiException"/>
2453 public TwitterConfiguration ConfigurationApi()
2459 res = twCon.GetConfiguration(ref content);
2463 throw new WebApiException("Err:" + e.Message + "(" + MethodBase.GetCurrentMethod().Name + ")", e);
2466 var err = this.CheckStatusCode(res, content);
2468 throw new WebApiException(err, content);
2472 return TwitterConfiguration.ParseJson(content);
2474 catch(SerializationException e)
2476 var ex = new WebApiException("Err:Json Parse Error(DataContractJsonSerializer)", content, e);
2477 MyCommon.TraceOut(ex);
2482 var ex = new WebApiException("Err:Invalid Json!", content, e);
2483 MyCommon.TraceOut(ex);
2488 public string GetListsApi()
2490 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
2493 IEnumerable<ListElement> lists;
2498 res = twCon.GetLists(this.Username, ref content);
2500 catch (Exception ex)
2502 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2505 var err = this.CheckStatusCode(res, content);
2506 if (err != null) return err;
2510 lists = TwitterList.ParseJsonArray(content)
2511 .Select(x => new ListElement(x, this));
2513 catch (SerializationException ex)
2515 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
2516 return "Err:Json Parse Error(DataContractJsonSerializer)";
2518 catch (Exception ex)
2520 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2521 return "Err:Invalid Json!";
2526 res = twCon.GetListsSubscriptions(this.Username, ref content);
2528 catch (Exception ex)
2530 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2533 err = this.CheckStatusCode(res, content);
2534 if (err != null) return err;
2538 lists = lists.Concat(TwitterList.ParseJsonArray(content)
2539 .Select(x => new ListElement(x, this)));
2541 catch (SerializationException ex)
2543 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
2544 return "Err:Json Parse Error(DataContractJsonSerializer)";
2546 catch (Exception ex)
2548 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2549 return "Err:Invalid Json!";
2552 TabInformations.GetInstance().SubscribableLists = lists.ToList();
2556 public string DeleteList(string list_id)
2563 res = twCon.DeleteListID(this.Username, list_id, ref content);
2567 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2570 var err = this.CheckStatusCode(res, content);
2571 if (err != null) return err;
2576 public string EditList(string list_id, string new_name, bool isPrivate, string description, ref ListElement list)
2583 res = twCon.UpdateListID(this.Username, list_id, new_name, isPrivate, description, ref content);
2587 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2590 var err = this.CheckStatusCode(res, content);
2591 if (err != null) return err;
2595 var le = TwitterList.ParseJson(content);
2596 var newList = new ListElement(le, this);
2597 list.Description = newList.Description;
2598 list.Id = newList.Id;
2599 list.IsPublic = newList.IsPublic;
2600 list.MemberCount = newList.MemberCount;
2601 list.Name = newList.Name;
2602 list.SubscriberCount = newList.SubscriberCount;
2603 list.Slug = newList.Slug;
2604 list.Nickname = newList.Nickname;
2605 list.Username = newList.Username;
2606 list.UserId = newList.UserId;
2609 catch(SerializationException ex)
2611 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
2612 return "Err:Json Parse Error(DataContractJsonSerializer)";
2616 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2617 return "Err:Invalid Json!";
2622 public string GetListMembers(string list_id, List<UserInfo> lists, ref long cursor)
2624 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
2630 res = twCon.GetListMembers(this.Username, list_id, cursor, ref content);
2634 return "Err:" + ex.Message;
2637 var err = this.CheckStatusCode(res, content);
2638 if (err != null) return err;
2642 var users = TwitterUsers.ParseJson(content);
2643 Array.ForEach<TwitterUser>(
2645 u => lists.Add(new UserInfo(u)));
2646 cursor = users.NextCursor;
2649 catch(SerializationException ex)
2651 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
2652 return "Err:Json Parse Error(DataContractJsonSerializer)";
2656 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2657 return "Err:Invalid Json!";
2661 public string CreateListApi(string listName, bool isPrivate, string description)
2663 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
2669 res = twCon.CreateLists(listName, isPrivate, description, ref content);
2673 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2676 var err = this.CheckStatusCode(res, content);
2677 if (err != null) return err;
2681 var le = TwitterList.ParseJson(content);
2682 TabInformations.GetInstance().SubscribableLists.Add(new ListElement(le, this));
2685 catch(SerializationException ex)
2687 MyCommon.TraceOut(ex.Message + Environment.NewLine + content);
2688 return "Err:Json Parse Error(DataContractJsonSerializer)";
2692 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
2693 return "Err:Invalid Json!";
2697 public string ContainsUserAtList(string listId, string user, ref bool value)
2701 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return "";
2708 res = this.twCon.ShowListMember(listId, user, ref content);
2712 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2715 if (res == HttpStatusCode.NotFound)
2721 var err = this.CheckStatusCode(res, content);
2722 if (err != null) return err;
2726 var u = TwitterUser.ParseJson(content);
2737 public string AddUserToList(string listId, string user)
2744 res = twCon.CreateListMembers(listId, user, ref content);
2748 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2751 var err = this.CheckStatusCode(res, content);
2752 if (err != null) return err;
2757 public string RemoveUserToList(string listId, string user)
2764 res = twCon.DeleteListMembers(listId, user, ref content);
2768 return "Err:" + ex.Message + "(" + MethodBase.GetCurrentMethod().Name + ")";
2771 var err = this.CheckStatusCode(res, content);
2772 if (err != null) return err;
2779 public int fromIndex { get; set; }
2780 public int toIndex { get; set; }
2781 public range(int fromIndex, int toIndex)
2783 this.fromIndex = fromIndex;
2784 this.toIndex = toIndex;
2787 public async Task<string> CreateHtmlAnchorAsync(string Text, List<string> AtList, Dictionary<string, string> media)
2789 if (Text == null) return null;
2790 var retStr = Text.Replace(">", "<<<<<tweenだいなり>>>>>").Replace("<", "<<<<<tweenしょうなり>>>>>");
2792 //const string url_valid_domain = "(?<domain>(?:[^\p{P}\s][\.\-_](?=[^\p{P}\s])|[^\p{P}\s]){1,}\.[a-z]{2,}(?::[0-9]+)?)"
2793 //const string url_valid_general_path_chars = "[a-z0-9!*';:=+$/%#\[\]\-_,~]"
2794 //const string url_balance_parens = "(?:\(" + url_valid_general_path_chars + "+\))"
2795 //const string url_valid_url_path_ending_chars = "(?:[a-z0-9=_#/\-\+]+|" + url_balance_parens + ")"
2796 //const string pth = "(?:" + url_balance_parens +
2797 // "|@" + url_valid_general_path_chars + "+/" +
2798 // "|[.,]?" + url_valid_general_path_chars + "+" +
2800 //const string pth2 = "(/(?:" +
2801 // pth + "+" + url_valid_url_path_ending_chars + "|" +
2802 // pth + "+" + url_valid_url_path_ending_chars + "?|" +
2803 // url_valid_url_path_ending_chars +
2805 //const string qry = "(?<query>\?[a-z0-9!*'();:&=+$/%#\[\]\-_.,~]*[a-z0-9_&=#])?"
2806 //const string rgUrl = "(?<before>(?:[^\""':!=#]|^|\:/))" +
2807 // "(?<url>(?<protocol>https?://)" +
2808 // url_valid_domain +
2812 //const string rgUrl = "(?<before>(?:[^\""':!=#]|^|\:/))" +
2813 // "(?<url>(?<protocol>https?://|www\.)" +
2814 // url_valid_domain +
2819 retStr = await new Regex(rgUrl, RegexOptions.IgnoreCase).ReplaceAsync(retStr, async mu =>
2821 var sb = new StringBuilder(mu.Result("${before}<a href=\""));
2822 //if (mu.Result("${protocol}").StartsWith("w", StringComparison.OrdinalIgnoreCase))
2823 // sb.Append("http://");
2825 var url = mu.Result("${url}");
2826 var title = await ShortUrl.Instance.ExpandUrlStrAsync(url);
2827 sb.Append(url + "\" title=\"" + MyCommon.ConvertToReadableUrl(title) + "\">").Append(url).Append("</a>");
2828 if (media != null && !media.ContainsKey(url)) media.Add(url, title);
2829 return sb.ToString();
2833 retStr = Regex.Replace(retStr,
2834 @"(^|[^a-zA-Z0-9_/])([@@]+)([a-zA-Z0-9_]{1,20}/[a-zA-Z][a-zA-Z0-9\p{IsLatin-1Supplement}\-]{0,79})",
2835 "$1$2<a href=\"/$3\">$3</a>");
2837 var m = Regex.Match(retStr, "(^|[^a-zA-Z0-9_])[@@]([a-zA-Z0-9_]{1,20})");
2840 if (!AtList.Contains(m.Result("$2").ToLower())) AtList.Add(m.Result("$2").ToLower());
2844 retStr = Regex.Replace(retStr,
2845 "(^|[^a-zA-Z0-9_/])([@@])([a-zA-Z0-9_]{1,20})",
2846 "$1$2<a href=\"/$3\">$3</a>");
2849 var anchorRange = new List<range>();
2850 for (int i = 0; i < retStr.Length; i++)
2852 var index = retStr.IndexOf("<a ", i);
2853 if (index > -1 && index < retStr.Length)
2856 var toIndex = retStr.IndexOf("</a>", index);
2859 anchorRange.Add(new range(index, toIndex + 3));
2864 //retStr = Regex.Replace(retStr,
2865 // "(^|[^a-zA-Z0-9/&])([##])([0-9a-zA-Z_]*[a-zA-Z_]+[a-zA-Z0-9_\xc0-\xd6\xd8-\xf6\xf8-\xff]*)",
2866 // new MatchEvaluator(Function(mh As Match)
2867 // foreach (var rng in anchorRange)
2869 // if (mh.Index >= rng.fromIndex &&
2870 // mh.Index <= rng.toIndex) return mh.Result("$0");
2872 // if (IsNumeric(mh.Result("$3"))) return mh.Result("$0");
2875 // _hashList.Add("#" + mh.Result("$3"))
2877 // return mh.Result("$1") + "<a href=\"" + _protocol + "twitter.com/search?q=%23" + mh.Result("$3") + "\">" + mh.Result("$2$3") + "</a>";
2879 // RegexOptions.IgnoreCase)
2880 retStr = Regex.Replace(retStr,
2882 new MatchEvaluator(mh =>
2884 foreach (var rng in anchorRange)
2886 if (mh.Index >= rng.fromIndex &&
2887 mh.Index <= rng.toIndex) return mh.Result("$0");
2891 _hashList.Add("#" + mh.Result("$3"));
2893 return mh.Result("$1") + "<a href=\"https://twitter.com/search?q=%23" + mh.Result("$3") + "\">" + mh.Result("$2$3") + "</a>";
2895 RegexOptions.IgnoreCase);
2898 retStr = Regex.Replace(retStr, "(^|[^a-zA-Z0-9_/&##@@>=.~])(sm|nm)([0-9]{1,10})", "$1<a href=\"http://www.nicovideo.jp/watch/$2$3\">$2$3</a>");
2900 retStr = retStr.Replace("<<<<<tweenだいなり>>>>>", ">").Replace("<<<<<tweenしょうなり>>>>>", "<");
2902 //retStr = AdjustHtml(ShortUrl.Resolve(PreProcessUrl(retStr), true)) //IDN置換、短縮Uri解決、@リンクを相対→絶対にしてtarget属性付与
2903 retStr = AdjustHtml(PreProcessUrl(retStr)); //IDN置換、短縮Uri解決、@リンクを相対→絶対にしてtarget属性付与
2907 public async Task<string> CreateHtmlAnchorAsync(string text, List<string> AtList, TwitterEntities entities, Dictionary<string, string> media)
2909 if (entities != null)
2911 if (entities.Urls != null)
2913 foreach (var ent in entities.Urls)
2915 ent.ExpandedUrl = await ShortUrl.Instance.ExpandUrlStrAsync(ent.ExpandedUrl)
2916 .ConfigureAwait(false);
2918 if (media != null && !media.ContainsKey(ent.Url))
2919 media.Add(ent.Url, ent.ExpandedUrl);
2922 if (entities.Hashtags != null)
2926 this._hashList.AddRange(entities.Hashtags.Select(x => "#" + x.Text));
2929 if (entities.UserMentions != null)
2931 foreach (var ent in entities.UserMentions)
2933 var screenName = ent.ScreenName.ToLower();
2934 if (!AtList.Contains(screenName))
2935 AtList.Add(screenName);
2938 if (entities.Media != null)
2940 foreach (var ent in entities.Media)
2942 if (media != null && !media.ContainsKey(ent.Url))
2943 media.Add(ent.Url, ent.MediaUrl);
2948 text = TweetFormatter.AutoLinkHtml(text, entities);
2950 text = Regex.Replace(text, "(^|[^a-zA-Z0-9_/&##@@>=.~])(sm|nm)([0-9]{1,10})", "$1<a href=\"http://www.nicovideo.jp/watch/$2$3\">$2$3</a>");
2951 text = PreProcessUrl(text); //IDN置換
2957 public string CreateHtmlAnchor(string text, List<string> AtList, TwitterEntities entities, Dictionary<string, string> media)
2959 return this.CreateHtmlAnchorAsync(text, AtList, entities, media).Result;
2963 private void CreateSource(PostClass post)
2965 if (string.IsNullOrEmpty(post.Source)) return;
2967 if (post.Source.StartsWith("<"))
2969 if (!post.Source.Contains("</a>"))
2971 post.Source += "</a>";
2973 var mS = Regex.Match(post.Source, ">(?<source>.+)<");
2976 post.SourceHtml = ShortUrl.Instance.ExpandUrlHtml(PreProcessUrl(post.Source));
2977 post.Source = WebUtility.HtmlDecode(mS.Result("${source}"));
2982 post.SourceHtml = "";
2987 if (post.Source == "web")
2989 post.SourceHtml = Properties.Resources.WebSourceString;
2991 else if (post.Source == "Keitai Mail")
2993 post.SourceHtml = Properties.Resources.KeitaiMailSourceString;
2997 post.SourceHtml = post.Source;
3002 public TwitterApiStatus GetInfoApi()
3004 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid) return null;
3006 if (MyCommon._endingFlag) return null;
3012 res = twCon.RateLimitStatus(ref content);
3016 this.ResetApiStatus();
3020 var err = this.CheckStatusCode(res, content);
3021 if (err != null) return null;
3025 MyCommon.TwitterApiInfo.UpdateFromJson(content);
3026 return MyCommon.TwitterApiInfo;
3028 catch (Exception ex)
3030 MyCommon.TraceOut(ex, MethodBase.GetCurrentMethod().Name + " " + content);
3031 MyCommon.TwitterApiInfo.Reset();
3037 /// ブロック中のユーザーを更新します
3039 /// <exception cref="WebApiException"/>
3040 public void RefreshBlockIds()
3042 if (MyCommon._endingFlag) return;
3045 var newBlockIds = new List<long>();
3048 var ret = this.GetBlockIdsApi(cursor);
3049 newBlockIds.AddRange(ret.Ids);
3050 cursor = ret.NextCursor;
3051 } while (cursor != 0);
3053 newBlockIds.Remove(this.UserId); // 元のソースにあったので一応残しておく
3055 TabInformations.GetInstance().BlockIds = newBlockIds;
3058 public TwitterIds GetBlockIdsApi(long cursor)
3060 if (Twitter.AccountState != MyCommon.ACCOUNT_STATE.Valid)
3061 throw new WebApiException("AccountState invalid");
3067 res = twCon.GetBlockUserIds(ref content, cursor);
3071 throw new WebApiException("Err:" + e.Message + "(" + MethodBase.GetCurrentMethod().Name + ")", e);
3074 var err = this.CheckStatusCode(res, content);
3076 throw new WebApiException(err, content);
3080 return TwitterIds.ParseJson(content);
3082 catch(SerializationException e)
3084 var ex = new WebApiException("Err:Json Parse Error(DataContractJsonSerializer)", content, e);
3085 MyCommon.TraceOut(ex);
3090 var ex = new WebApiException("Err:Invalid Json!", content, e);
3091 MyCommon.TraceOut(ex);
3097 /// ミュート中のユーザーIDを更新します
3099 /// <exception cref="WebApiException"/>
3100 public async Task RefreshMuteUserIdsAsync()
3102 if (MyCommon._endingFlag) return;
3104 var ids = await TwitterIds.GetAllItemsAsync(this.GetMuteUserIdsApiAsync)
3105 .ConfigureAwait(false);
3107 TabInformations.GetInstance().MuteUserIds = ids.ToList();
3110 public async Task<TwitterIds> GetMuteUserIdsApiAsync(long cursor)
3116 var res = await Task.Run(() => twCon.GetMuteUserIds(ref content, cursor))
3117 .ConfigureAwait(false);
3119 var err = this.CheckStatusCode(res, content);
3121 throw new WebApiException(err, content);
3123 return TwitterIds.ParseJson(content);
3125 catch (WebException ex)
3127 var ex2 = new WebApiException("Err:" + ex.Message, ex);
3128 MyCommon.TraceOut(ex2);
3131 catch (SerializationException ex)
3133 var ex2 = new WebApiException("Err:Json Parse Error(DataContractJsonSerializer)", content, ex);
3134 MyCommon.TraceOut(ex2);
3139 public string[] GetHashList()
3144 hashArray = _hashList.ToArray();
3150 public string AccessToken
3154 return twCon.AccessToken;
3158 public string AccessTokenSecret
3162 return twCon.AccessTokenSecret;
3166 [MethodImpl(MethodImplOptions.NoInlining)] // 呼び出し元の取得に必要
3167 private string CheckStatusCode(HttpStatusCode httpStatus, string responseText)
3169 if (httpStatus == HttpStatusCode.OK)
3171 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Valid;
3175 // 404エラーの挙動が変なので無視: https://dev.twitter.com/discussions/1213
3176 if (httpStatus == HttpStatusCode.NotFound) return null;
3178 var callerMethod = new StackTrace(false).GetFrame(1).GetMethod();
3179 var callerMethodName = callerMethod != null
3183 if (string.IsNullOrWhiteSpace(responseText))
3185 if (httpStatus == HttpStatusCode.Unauthorized)
3186 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Invalid;
3188 return "Err:" + httpStatus + "(" + callerMethodName + ")";
3193 var errors = TwitterError.ParseJson(responseText).Errors;
3194 if (errors == null || !errors.Any())
3196 return "Err:" + httpStatus + "(" + callerMethodName + ")";
3199 foreach (var error in errors)
3201 if (error.Code == TwitterErrorCode.InvalidToken ||
3202 error.Code == TwitterErrorCode.SuspendedAccount)
3204 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Invalid;
3208 return "Err:" + string.Join(",", errors.Select(x => x.ToString())) + "(" + callerMethodName + ")";
3210 catch (SerializationException) { }
3212 return "Err:" + httpStatus + "(" + callerMethodName + ")";
3215 #region "UserStream"
3216 private string trackWord_ = "";
3217 public string TrackWord
3228 private bool allAtReply_ = false;
3229 public bool AllAtReply
3237 allAtReply_ = value;
3241 public event Action NewPostFromStream;
3242 public event Action UserStreamStarted;
3243 public event Action UserStreamStopped;
3244 public event Action<long> PostDeleted;
3245 public event Action<FormattedEvent> UserStreamEventReceived;
3246 private DateTime _lastUserstreamDataReceived;
3247 private TwitterUserstream userStream;
3249 public class FormattedEvent
3251 public MyCommon.EVENTTYPE Eventtype { get; set; }
3252 public DateTime CreatedAt { get; set; }
3253 public string Event { get; set; }
3254 public string Username { get; set; }
3255 public string Target { get; set; }
3256 public Int64 Id { get; set; }
3257 public bool IsMe { get; set; }
3260 public List<FormattedEvent> storedEvent_ = new List<FormattedEvent>();
3261 public List<FormattedEvent> StoredEvent
3265 return storedEvent_;
3269 storedEvent_ = value;
3273 private class EventTypeTableElement
3276 public MyCommon.EVENTTYPE Type;
3278 public EventTypeTableElement(string name, MyCommon.EVENTTYPE type)
3285 private EventTypeTableElement[] EventTable = {
3286 new EventTypeTableElement("favorite", MyCommon.EVENTTYPE.Favorite),
3287 new EventTypeTableElement("unfavorite", MyCommon.EVENTTYPE.Unfavorite),
3288 new EventTypeTableElement("follow", MyCommon.EVENTTYPE.Follow),
3289 new EventTypeTableElement("list_member_added", MyCommon.EVENTTYPE.ListMemberAdded),
3290 new EventTypeTableElement("list_member_removed", MyCommon.EVENTTYPE.ListMemberRemoved),
3291 new EventTypeTableElement("block", MyCommon.EVENTTYPE.Block),
3292 new EventTypeTableElement("unblock", MyCommon.EVENTTYPE.Unblock),
3293 new EventTypeTableElement("user_update", MyCommon.EVENTTYPE.UserUpdate),
3294 new EventTypeTableElement("deleted", MyCommon.EVENTTYPE.Deleted),
3295 new EventTypeTableElement("list_created", MyCommon.EVENTTYPE.ListCreated),
3296 new EventTypeTableElement("list_destroyed", MyCommon.EVENTTYPE.ListDestroyed),
3297 new EventTypeTableElement("list_updated", MyCommon.EVENTTYPE.ListUpdated),
3298 new EventTypeTableElement("unfollow", MyCommon.EVENTTYPE.Unfollow),
3299 new EventTypeTableElement("list_user_subscribed", MyCommon.EVENTTYPE.ListUserSubscribed),
3300 new EventTypeTableElement("list_user_unsubscribed", MyCommon.EVENTTYPE.ListUserUnsubscribed),
3303 public MyCommon.EVENTTYPE EventNameToEventType(string EventName)
3305 return (from tbl in EventTable where tbl.Name.Equals(EventName) select tbl.Type).FirstOrDefault();
3308 public bool IsUserstreamDataReceived
3312 return DateTime.Now.Subtract(this._lastUserstreamDataReceived).TotalSeconds < 31;
3316 private void userStream_StatusArrived(string line)
3318 this._lastUserstreamDataReceived = DateTime.Now;
3319 if (string.IsNullOrEmpty(line)) return;
3325 using (var jsonReader = JsonReaderWriterFactory.CreateJsonReader(Encoding.UTF8.GetBytes(line), XmlDictionaryReaderQuotas.Max))
3327 var xElm = XElement.Load(jsonReader);
3328 if (xElm.Element("friends") != null)
3330 Debug.WriteLine("friends");
3333 else if (xElm.Element("delete") != null)
3335 Debug.WriteLine("delete");
3337 if (xElm.Element("delete").Element("direct_message") != null &&
3338 xElm.Element("delete").Element("direct_message").Element("id") != null)
3341 long.TryParse(xElm.Element("delete").Element("direct_message").Element("id").Value, out id);
3342 if (PostDeleted != null)
3347 else if (xElm.Element("delete").Element("status") != null &&
3348 xElm.Element("delete").Element("status").Element("id") != null)
3351 long.TryParse(xElm.Element("delete").Element("status").Element("id").Value, out id);
3352 if (PostDeleted != null)
3359 MyCommon.TraceOut("delete:" + line);
3362 for (int i = this.StoredEvent.Count - 1; i >= 0; i--)
3364 var sEvt = this.StoredEvent[i];
3365 if (sEvt.Id == id && (sEvt.Event == "favorite" || sEvt.Event == "unfavorite"))
3367 this.StoredEvent.RemoveAt(i);
3372 else if (xElm.Element("limit") != null)
3374 Debug.WriteLine(line);
3377 else if (xElm.Element("event") != null)
3379 Debug.WriteLine("event: " + xElm.Element("event").Value);
3380 CreateEventFromJson(line);
3383 else if (xElm.Element("direct_message") != null)
3385 Debug.WriteLine("direct_message");
3388 else if (xElm.Element("scrub_geo") != null)
3392 TabInformations.GetInstance().ScrubGeoReserve(long.Parse(xElm.Element("scrub_geo").Element("user_id").Value),
3393 long.Parse(xElm.Element("scrub_geo").Element("up_to_status_id").Value));
3397 MyCommon.TraceOut("scrub_geo:" + line);
3405 CreateDirectMessagesFromJson(line, MyCommon.WORKERTYPE.UserStream, false);
3410 CreatePostsFromJson("[" + line + "]", MyCommon.WORKERTYPE.Timeline, null, false, 0, ref dummy);
3413 catch(NullReferenceException)
3415 MyCommon.TraceOut("NullRef StatusArrived: " + line);
3418 if (NewPostFromStream != null)
3420 NewPostFromStream();
3424 private void CreateEventFromJson(string content)
3426 TwitterStreamEvent eventData = null;
3429 eventData = TwitterStreamEvent.ParseJson(content);
3431 catch(SerializationException ex)
3433 MyCommon.TraceOut(ex, "Event Serialize Exception!" + Environment.NewLine + content);
3437 MyCommon.TraceOut(ex, "Event Exception!" + Environment.NewLine + content);
3440 var evt = new FormattedEvent();
3441 evt.CreatedAt = MyCommon.DateTimeParse(eventData.CreatedAt);
3442 evt.Event = eventData.Event;
3443 evt.Username = eventData.Source.ScreenName;
3444 evt.IsMe = evt.Username.ToLower().Equals(this.Username.ToLower());
3445 evt.Eventtype = EventNameToEventType(evt.Event);
3446 switch (eventData.Event)
3448 case "access_revoked":
3451 if (eventData.Target.ScreenName.ToLower().Equals(_uname))
3453 if (!this.followerId.Contains(eventData.Source.Id)) this.followerId.Add(eventData.Source.Id);
3457 return; //Block後のUndoをすると、SourceとTargetが逆転したfollowイベントが帰ってくるため。
3462 evt.Target = "@" + eventData.Target.ScreenName;
3466 var tweetEvent = TwitterStreamEvent<TwitterStatus>.ParseJson(content);
3467 evt.Target = "@" + tweetEvent.TargetObject.User.ScreenName + ":" + WebUtility.HtmlDecode(tweetEvent.TargetObject.Text);
3468 evt.Id = tweetEvent.TargetObject.Id;
3469 if (AppendSettingDialog.Instance.IsRemoveSameEvent)
3471 if (StoredEvent.Any(ev =>
3473 return ev.Username == evt.Username && ev.Eventtype == evt.Eventtype && ev.Target == evt.Target;
3476 if (TabInformations.GetInstance().ContainsKey(tweetEvent.TargetObject.Id))
3478 var post = TabInformations.GetInstance()[tweetEvent.TargetObject.Id];
3479 if (eventData.Event == "favorite")
3481 if (evt.Username.ToLower().Equals(_uname))
3484 TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites).Add(post.StatusId, post.IsRead, false);
3488 post.FavoritedCount++;
3489 if (!TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites).Contains(post.StatusId))
3491 if (AppendSettingDialog.Instance.FavEventUnread && post.IsRead)
3493 post.IsRead = false;
3495 TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites).Add(post.StatusId, post.IsRead, false);
3499 if (AppendSettingDialog.Instance.FavEventUnread)
3501 TabInformations.GetInstance().SetRead(false, TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites).TabName, TabInformations.GetInstance().GetTabByType(MyCommon.TabUsageType.Favorites).IndexOf(post.StatusId));
3508 if (evt.Username.ToLower().Equals(_uname))
3514 post.FavoritedCount--;
3515 if (post.FavoritedCount < 0) post.FavoritedCount = 0;
3520 case "list_member_added":
3521 case "list_member_removed":
3522 case "list_destroyed":
3523 case "list_updated":
3524 case "list_user_subscribed":
3525 case "list_user_unsubscribed":
3526 var listEvent = TwitterStreamEvent<TwitterList>.ParseJson(content);
3527 evt.Target = listEvent.TargetObject.FullName;
3530 if (!TabInformations.GetInstance().BlockIds.Contains(eventData.Target.Id)) TabInformations.GetInstance().BlockIds.Add(eventData.Target.Id);
3534 if (TabInformations.GetInstance().BlockIds.Contains(eventData.Target.Id)) TabInformations.GetInstance().BlockIds.Remove(eventData.Target.Id);
3540 case "list_created":
3544 MyCommon.TraceOut("Unknown Event:" + evt.Event + Environment.NewLine + content);
3547 this.StoredEvent.Insert(0, evt);
3548 if (UserStreamEventReceived != null)
3550 UserStreamEventReceived(evt);
3554 private void userStream_Started()
3556 if (UserStreamStarted != null)
3558 UserStreamStarted();
3562 private void userStream_Stopped()
3564 if (UserStreamStopped != null)
3566 UserStreamStopped();
3570 public bool UserStreamEnabled
3574 return userStream == null ? false : userStream.Enabled;
3578 public void StartUserStream()
3580 if (userStream != null)
3584 userStream = new TwitterUserstream(twCon);
3585 userStream.StatusArrived += userStream_StatusArrived;
3586 userStream.Started += userStream_Started;
3587 userStream.Stopped += userStream_Stopped;
3588 userStream.Start(this.AllAtReply, this.TrackWord);
3591 public void StopUserStream()
3593 if (userStream != null) userStream.Dispose();
3595 if (!MyCommon._endingFlag)
3597 if (UserStreamStopped != null)
3599 UserStreamStopped();
3604 public void ReconnectUserStream()
3606 if (userStream != null)
3608 this.StartUserStream();
3612 private class TwitterUserstream : IDisposable
3614 public event Action<string> StatusArrived;
3615 public event Action Stopped;
3616 public event Action Started;
3617 private HttpTwitter twCon;
3619 private Thread _streamThread;
3620 private bool _streamActive;
3622 private bool _allAtreplies = false;
3623 private string _trackwords = "";
3625 public TwitterUserstream(HttpTwitter twitterConnection)
3627 twCon = (HttpTwitter)twitterConnection.Clone();
3630 public void Start(bool allAtReplies, string trackwords)
3632 this.AllAtReplies = allAtReplies;
3633 this.TrackWords = trackwords;
3634 _streamActive = true;
3635 if (_streamThread != null && _streamThread.IsAlive) return;
3636 _streamThread = new Thread(UserStreamLoop);
3637 _streamThread.Name = "UserStreamReceiver";
3638 _streamThread.IsBackground = true;
3639 _streamThread.Start();
3646 return _streamActive;
3650 public bool AllAtReplies
3654 return _allAtreplies;
3658 _allAtreplies = value;
3662 public string TrackWords
3670 _trackwords = value;
3674 private void UserStreamLoop()
3680 StreamReader sr = null;
3683 if (!MyCommon.IsNetworkAvailable())
3689 if (Started != null)
3693 var res = twCon.UserStream(ref st, _allAtreplies, _trackwords, MyCommon.GetUserAgentString());
3697 case HttpStatusCode.OK:
3698 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Valid;
3700 case HttpStatusCode.Unauthorized:
3701 Twitter.AccountState = MyCommon.ACCOUNT_STATE.Invalid;
3709 //MyCommon.TraceOut("Stop:stream is null")
3713 sr = new StreamReader(st);
3715 while (_streamActive && !sr.EndOfStream && Twitter.AccountState == MyCommon.ACCOUNT_STATE.Valid)
3717 if (StatusArrived != null)
3719 StatusArrived(sr.ReadLine());
3721 //this.LastTime = Now;
3724 if (sr.EndOfStream || Twitter.AccountState == MyCommon.ACCOUNT_STATE.Invalid)
3727 //MyCommon.TraceOut("Stop:EndOfStream")
3732 catch(WebException ex)
3734 if (ex.Status == WebExceptionStatus.Timeout)
3736 sleepSec = 30; //MyCommon.TraceOut("Stop:Timeout")
3738 else if (ex.Response != null && (int)((HttpWebResponse)ex.Response).StatusCode == 420)
3740 //MyCommon.TraceOut("Stop:Connection Limit")
3746 //MyCommon.TraceOut("Stop:WebException " + ex.Status.ToString())
3749 catch(ThreadAbortException)
3756 //MyCommon.TraceOut("Stop:IOException with Active." + Environment.NewLine + ex.Message)
3758 catch(ArgumentException ex)
3760 //System.ArgumentException: ストリームを読み取れませんでした。
3761 //サーバー側もしくは通信経路上で切断された場合?タイムアウト頻発後発生
3763 MyCommon.TraceOut(ex, "Stop:ArgumentException");
3767 MyCommon.TraceOut("Stop:Exception." + Environment.NewLine + ex.Message);
3768 MyCommon.ExceptionOut(ex);
3775 if (Stopped != null)
3780 twCon.RequestAbort();
3781 if (sr != null) sr.Close();
3785 while (_streamActive && ms < sleepSec * 1000)
3793 } while (this._streamActive);
3797 if (Stopped != null)
3802 MyCommon.TraceOut("Stop:EndLoop");
3805 #region "IDisposable Support"
3806 private bool disposedValue; // 重複する呼び出しを検出するには
3809 protected virtual void Dispose(bool disposing)
3811 if (!this.disposedValue)
3815 _streamActive = false;
3816 if (_streamThread != null && _streamThread.IsAlive)
3818 _streamThread.Abort();
3822 this.disposedValue = true;
3825 //protected Overrides void Finalize()
3827 // // このコードを変更しないでください。クリーンアップ コードを上の Dispose(bool disposing) に記述します。
3829 // MyBase.Finalize()
3832 // このコードは、破棄可能なパターンを正しく実装できるように Visual Basic によって追加されました。
3833 public void Dispose()
3835 // このコードを変更しないでください。クリーンアップ コードを上の Dispose(bool disposing) に記述します。
3837 GC.SuppressFinalize(this);
3844 #region "IDisposable Support"
3845 private bool disposedValue; // 重複する呼び出しを検出するには
3848 protected virtual void Dispose(bool disposing)
3850 if (!this.disposedValue)
3854 this.StopUserStream();
3857 this.disposedValue = true;
3860 //protected Overrides void Finalize()
3862 // // このコードを変更しないでください。クリーンアップ コードを上の Dispose(bool disposing) に記述します。
3864 // MyBase.Finalize()
3867 // このコードは、破棄可能なパターンを正しく実装できるように Visual Basic によって追加されました。
3868 public void Dispose()
3870 // このコードを変更しないでください。クリーンアップ コードを上の Dispose(bool disposing) に記述します。
3872 GC.SuppressFinalize(this);