using System.Windows.Forms;
using OpenTween.Api;
using OpenTween.Api.DataModel;
+using OpenTween.Api.GraphQL;
using OpenTween.Api.TwitterV2;
using OpenTween.Connection;
using OpenTween.MediaUploadServices;
private readonly HookGlobalHotkey hookGlobalHotkey;
- private void TweenMain_Activated(object sender, EventArgs e)
- {
- // 画面がアクティブになったら、発言欄の背景色戻す
- if (this.StatusText.Focused)
- {
- this.StatusText_Enter(this.StatusText, System.EventArgs.Empty);
- }
- }
-
- private bool disposed = false;
-
- /// <summary>
- /// 使用中のリソースをすべてクリーンアップします。
- /// </summary>
- /// <param name="disposing">マネージ リソースが破棄される場合 true、破棄されない場合は false です。</param>
- protected override void Dispose(bool disposing)
- {
- base.Dispose(disposing);
-
- if (this.disposed)
- return;
-
- if (disposing)
- {
- this.components?.Dispose();
-
- // 後始末
- this.SearchDialog.Dispose();
- this.urlDialog.Dispose();
- this.themeManager.Dispose();
- this.sfTab.Dispose();
-
- this.timelineScheduler.Dispose();
- this.workerCts.Cancel();
- this.thumbnailTokenSource?.Dispose();
-
- this.hookGlobalHotkey.Dispose();
- }
-
- // 終了時にRemoveHandlerしておかないとメモリリークする
- // http://msdn.microsoft.com/ja-jp/library/microsoft.win32.systemevents.powermodechanged.aspx
- Microsoft.Win32.SystemEvents.PowerModeChanged -= this.SystemEvents_PowerModeChanged;
- Microsoft.Win32.SystemEvents.TimeChanged -= this.SystemEvents_TimeChanged;
-
- this.disposed = true;
- }
-
- private void InitColumns(ListView list, bool startup)
- {
- this.InitColumnText();
-
- ColumnHeader[]? columns = null;
- try
- {
- if (this.Use2ColumnsMode)
- {
- columns = new[]
- {
- new ColumnHeader(), // アイコン
- new ColumnHeader(), // 本文
- };
-
- columns[0].Text = this.columnText[0];
- columns[1].Text = this.columnText[2];
-
- if (startup)
- {
- var widthScaleFactor = this.CurrentAutoScaleDimensions.Width / this.settings.Local.ScaleDimension.Width;
-
- columns[0].Width = ScaleBy(widthScaleFactor, this.settings.Local.ColumnsWidth[0]);
- columns[1].Width = ScaleBy(widthScaleFactor, this.settings.Local.ColumnsWidth[2]);
- columns[0].DisplayIndex = 0;
- columns[1].DisplayIndex = 1;
- }
- else
- {
- var idx = 0;
- foreach (var curListColumn in this.CurrentListView.Columns.Cast<ColumnHeader>())
- {
- columns[idx].Width = curListColumn.Width;
- columns[idx].DisplayIndex = curListColumn.DisplayIndex;
- idx++;
- }
- }
- }
- else
- {
- columns = new[]
- {
- new ColumnHeader(), // アイコン
- new ColumnHeader(), // ニックネーム
- new ColumnHeader(), // 本文
- new ColumnHeader(), // 日付
- new ColumnHeader(), // ユーザID
- new ColumnHeader(), // 未読
- new ColumnHeader(), // マーク&プロテクト
- new ColumnHeader(), // ソース
- };
-
- foreach (var i in Enumerable.Range(0, columns.Length))
- columns[i].Text = this.columnText[i];
-
- if (startup)
- {
- var widthScaleFactor = this.CurrentAutoScaleDimensions.Width / this.settings.Local.ScaleDimension.Width;
-
- foreach (var (column, index) in columns.WithIndex())
- {
- column.Width = ScaleBy(widthScaleFactor, this.settings.Local.ColumnsWidth[index]);
- column.DisplayIndex = this.settings.Local.ColumnsOrder[index];
- }
- }
- else
- {
- var idx = 0;
- foreach (var curListColumn in this.CurrentListView.Columns.Cast<ColumnHeader>())
- {
- columns[idx].Width = curListColumn.Width;
- columns[idx].DisplayIndex = curListColumn.DisplayIndex;
- idx++;
- }
- }
- }
-
- list.Columns.AddRange(columns);
-
- columns = null;
- }
- finally
- {
- if (columns != null)
- {
- foreach (var column in columns)
- column?.Dispose();
- }
- }
- }
-
- private void InitColumnText()
- {
- this.columnText[0] = "";
- this.columnText[1] = Properties.Resources.AddNewTabText2;
- this.columnText[2] = Properties.Resources.AddNewTabText3;
- this.columnText[3] = Properties.Resources.AddNewTabText4_2;
- this.columnText[4] = Properties.Resources.AddNewTabText5;
- this.columnText[5] = "";
- this.columnText[6] = "";
- this.columnText[7] = "Source";
-
- this.columnOrgText[0] = "";
- this.columnOrgText[1] = Properties.Resources.AddNewTabText2;
- this.columnOrgText[2] = Properties.Resources.AddNewTabText3;
- this.columnOrgText[3] = Properties.Resources.AddNewTabText4_2;
- this.columnOrgText[4] = Properties.Resources.AddNewTabText5;
- this.columnOrgText[5] = "";
- this.columnOrgText[6] = "";
- this.columnOrgText[7] = "Source";
-
- var c = this.statuses.SortMode switch
- {
- ComparerMode.Nickname => 1, // ニックネーム
- ComparerMode.Data => 2, // 本文
- ComparerMode.Id => 3, // 時刻=発言Id
- ComparerMode.Name => 4, // 名前
- ComparerMode.Source => 7, // Source
- _ => 0,
- };
-
- if (this.Use2ColumnsMode)
- {
- if (this.statuses.SortOrder == SortOrder.Descending)
- {
- // U+25BE BLACK DOWN-POINTING SMALL TRIANGLE
- this.columnText[2] = this.columnOrgText[2] + "▾";
- }
- else
- {
- // U+25B4 BLACK UP-POINTING SMALL TRIANGLE
- this.columnText[2] = this.columnOrgText[2] + "▴";
- }
- }
- else
- {
- if (this.statuses.SortOrder == SortOrder.Descending)
- {
- // U+25BE BLACK DOWN-POINTING SMALL TRIANGLE
- this.columnText[c] = this.columnOrgText[c] + "▾";
- }
- else
- {
- // U+25B4 BLACK UP-POINTING SMALL TRIANGLE
- this.columnText[c] = this.columnOrgText[c] + "▴";
- }
- }
- }
-
public TweenMain(
SettingManager settingManager,
TabInformations tabInfo,
this.InitializeShortcuts();
this.ignoreConfigSave = true;
- this.Visible = false;
this.TraceOutToolStripMenuItem.Checked = MyCommon.TraceFlag;
// 現在の DPI と設定保存時の DPI との比を取得する
var configScaleFactor = this.settings.Local.GetConfigScaleFactor(this.CurrentAutoScaleDimensions);
- // 認証関連
- var account = this.settings.Common.SelectedAccount;
- if (account != null)
- this.tw.Initialize(account.GetTwitterAppToken(), account.Token, account.TokenSecret, account.Username, account.UserId);
- else
- this.tw.Initialize(TwitterAppToken.GetDefault(), "", "", "", 0L);
-
this.initial = true;
- this.tw.RestrictFavCheck = this.settings.Common.RestrictFavCheck;
- this.tw.ReadOwnPost = this.settings.Common.ReadOwnPost;
-
- // アクセストークンが有効であるか確認する
- // ここが Twitter API への最初のアクセスになるようにすること
- try
- {
- this.tw.VerifyCredentials();
- }
- catch (WebApiException ex)
- {
- MessageBox.Show(
- this,
- string.Format(Properties.Resources.StartupAuthError_Text, ex.Message),
- ApplicationSettings.ApplicationName,
- MessageBoxButtons.OK,
- MessageBoxIcon.Warning);
- }
-
// サムネイル関連の初期化
// プロキシ設定等の通信まわりの初期化が済んでから処理する
var imgazyobizinet = this.thumbGenerator.ImgAzyobuziNet;
imgazyobizinet.Enabled = this.settings.Common.EnableImgAzyobuziNet;
imgazyobizinet.DisabledInDM = this.settings.Common.ImgAzyobuziNetDisabledInDM;
- imgazyobizinet.AutoUpdate = true;
Thumbnail.Services.TonTwitterCom.GetApiConnection = () => this.tw.Api.Connection;
this.SetMainWindowTitle();
this.SetNotifyIconText();
- if (!this.settings.Common.MinimizeToTray || this.WindowState != FormWindowState.Minimized)
+ if (this.settings.Common.MinimizeToTray && this.WindowState == FormWindowState.Minimized)
{
- this.Visible = true;
+ this.Visible = false;
}
// タイマー設定
this.TimerRefreshIcon.Enabled = false;
this.ignoreConfigSave = false;
- this.TweenMain_Resize(this, EventArgs.Empty);
+ this.ApplyLayoutFromSettings();
+ }
- if (this.settings.IsFirstRun)
+ private void TweenMain_Activated(object sender, EventArgs e)
+ {
+ // 画面がアクティブになったら、発言欄の背景色戻す
+ if (this.StatusText.Focused)
{
- // 初回起動時だけ右下のメニューを目立たせる
- this.HashStripSplitButton.ShowDropDown();
+ this.StatusText_Enter(this.StatusText, System.EventArgs.Empty);
+ }
+ }
+
+ private bool disposed = false;
+
+ /// <summary>
+ /// 使用中のリソースをすべてクリーンアップします。
+ /// </summary>
+ /// <param name="disposing">マネージ リソースが破棄される場合 true、破棄されない場合は false です。</param>
+ protected override void Dispose(bool disposing)
+ {
+ base.Dispose(disposing);
+
+ if (this.disposed)
+ return;
+
+ if (disposing)
+ {
+ this.components?.Dispose();
+
+ // 後始末
+ this.SearchDialog.Dispose();
+ this.urlDialog.Dispose();
+ this.themeManager.Dispose();
+ this.sfTab.Dispose();
+
+ this.timelineScheduler.Dispose();
+ this.workerCts.Cancel();
+ this.thumbnailTokenSource?.Dispose();
+
+ this.hookGlobalHotkey.Dispose();
+ }
+
+ // 終了時にRemoveHandlerしておかないとメモリリークする
+ // http://msdn.microsoft.com/ja-jp/library/microsoft.win32.systemevents.powermodechanged.aspx
+ Microsoft.Win32.SystemEvents.PowerModeChanged -= this.SystemEvents_PowerModeChanged;
+ Microsoft.Win32.SystemEvents.TimeChanged -= this.SystemEvents_TimeChanged;
+ MyCommon.TwitterApiInfo.AccessLimitUpdated -= this.TwitterApiStatus_AccessLimitUpdated;
+
+ this.disposed = true;
+ }
+
+ private void InitColumns(ListView list, bool startup)
+ {
+ this.InitColumnText();
+
+ ColumnHeader[]? columns = null;
+ try
+ {
+ if (this.Use2ColumnsMode)
+ {
+ columns = new[]
+ {
+ new ColumnHeader(), // アイコン
+ new ColumnHeader(), // 本文
+ };
+
+ columns[0].Text = this.columnText[0];
+ columns[1].Text = this.columnText[2];
+
+ if (startup)
+ {
+ var widthScaleFactor = this.CurrentAutoScaleDimensions.Width / this.settings.Local.ScaleDimension.Width;
+
+ columns[0].Width = ScaleBy(widthScaleFactor, this.settings.Local.ColumnsWidth[0]);
+ columns[1].Width = ScaleBy(widthScaleFactor, this.settings.Local.ColumnsWidth[2]);
+ columns[0].DisplayIndex = 0;
+ columns[1].DisplayIndex = 1;
+ }
+ else
+ {
+ var idx = 0;
+ foreach (var curListColumn in this.CurrentListView.Columns.Cast<ColumnHeader>())
+ {
+ columns[idx].Width = curListColumn.Width;
+ columns[idx].DisplayIndex = curListColumn.DisplayIndex;
+ idx++;
+ }
+ }
+ }
+ else
+ {
+ columns = new[]
+ {
+ new ColumnHeader(), // アイコン
+ new ColumnHeader(), // ニックネーム
+ new ColumnHeader(), // 本文
+ new ColumnHeader(), // 日付
+ new ColumnHeader(), // ユーザID
+ new ColumnHeader(), // 未読
+ new ColumnHeader(), // マーク&プロテクト
+ new ColumnHeader(), // ソース
+ };
+
+ foreach (var i in Enumerable.Range(0, columns.Length))
+ columns[i].Text = this.columnText[i];
+
+ if (startup)
+ {
+ var widthScaleFactor = this.CurrentAutoScaleDimensions.Width / this.settings.Local.ScaleDimension.Width;
+
+ foreach (var (column, index) in columns.WithIndex())
+ {
+ column.Width = ScaleBy(widthScaleFactor, this.settings.Local.ColumnsWidth[index]);
+ column.DisplayIndex = this.settings.Local.ColumnsOrder[index];
+ }
+ }
+ else
+ {
+ var idx = 0;
+ foreach (var curListColumn in this.CurrentListView.Columns.Cast<ColumnHeader>())
+ {
+ columns[idx].Width = curListColumn.Width;
+ columns[idx].DisplayIndex = curListColumn.DisplayIndex;
+ idx++;
+ }
+ }
+ }
+
+ list.Columns.AddRange(columns);
+
+ columns = null;
+ }
+ finally
+ {
+ if (columns != null)
+ {
+ foreach (var column in columns)
+ column?.Dispose();
+ }
+ }
+ }
+
+ private void InitColumnText()
+ {
+ this.columnText[0] = "";
+ this.columnText[1] = Properties.Resources.AddNewTabText2;
+ this.columnText[2] = Properties.Resources.AddNewTabText3;
+ this.columnText[3] = Properties.Resources.AddNewTabText4_2;
+ this.columnText[4] = Properties.Resources.AddNewTabText5;
+ this.columnText[5] = "";
+ this.columnText[6] = "";
+ this.columnText[7] = "Source";
+
+ this.columnOrgText[0] = "";
+ this.columnOrgText[1] = Properties.Resources.AddNewTabText2;
+ this.columnOrgText[2] = Properties.Resources.AddNewTabText3;
+ this.columnOrgText[3] = Properties.Resources.AddNewTabText4_2;
+ this.columnOrgText[4] = Properties.Resources.AddNewTabText5;
+ this.columnOrgText[5] = "";
+ this.columnOrgText[6] = "";
+ this.columnOrgText[7] = "Source";
+
+ var c = this.statuses.SortMode switch
+ {
+ ComparerMode.Nickname => 1, // ニックネーム
+ ComparerMode.Data => 2, // 本文
+ ComparerMode.Id => 3, // 時刻=発言Id
+ ComparerMode.Name => 4, // 名前
+ ComparerMode.Source => 7, // Source
+ _ => 0,
+ };
+
+ if (this.Use2ColumnsMode)
+ {
+ if (this.statuses.SortOrder == SortOrder.Descending)
+ {
+ // U+25BE BLACK DOWN-POINTING SMALL TRIANGLE
+ this.columnText[2] = this.columnOrgText[2] + "▾";
+ }
+ else
+ {
+ // U+25B4 BLACK UP-POINTING SMALL TRIANGLE
+ this.columnText[2] = this.columnOrgText[2] + "▴";
+ }
+ }
+ else
+ {
+ if (this.statuses.SortOrder == SortOrder.Descending)
+ {
+ // U+25BE BLACK DOWN-POINTING SMALL TRIANGLE
+ this.columnText[c] = this.columnOrgText[c] + "▾";
+ }
+ else
+ {
+ // U+25B4 BLACK UP-POINTING SMALL TRIANGLE
+ this.columnText[c] = this.columnOrgText[c] + "▴";
+ }
}
}
var status = new PostStatusParams();
var statusTextCompat = this.FormatStatusText(this.StatusText.Text);
- if (this.GetRestStatusCount(statusTextCompat) >= 0 && this.tw.Api.AppToken.AuthType == APIAuthType.OAuth1)
+ if (this.GetRestStatusCount(statusTextCompat) >= 0 && this.tw.Api.AuthType == APIAuthType.OAuth1)
{
// auto_populate_reply_metadata や attachment_url を使用しなくても 140 字以内に
// 収まる場合はこれらのオプションを使用せずに投稿する
{
// 自分が RT したツイート (自分が RT した自分のツイートも含む)
// => RT を取り消し
- await this.tw.Api.StatusesDestroy(post.StatusId.ToTwitterStatusId())
- .IgnoreResponse();
+ await this.tw.DeleteRetweet(post);
}
else
{
{
// 他人に RT された自分のツイート
// => RT 元の自分のツイートを削除
- await this.tw.Api.StatusesDestroy(post.RetweetedId.ToTwitterStatusId())
- .IgnoreResponse();
+ await this.tw.DeleteTweet(post.RetweetedId.ToTwitterStatusId());
}
else
{
// 自分のツイート
// => ツイートを削除
- await this.tw.Api.StatusesDestroy(post.StatusId.ToTwitterStatusId())
- .IgnoreResponse();
+ await this.tw.DeleteTweet(post.StatusId.ToTwitterStatusId());
}
}
}
var account = this.settings.Common.SelectedAccount;
if (account != null)
- this.tw.Initialize(account.GetTwitterAppToken(), account.Token, account.TokenSecret, account.Username, account.UserId);
+ this.tw.Initialize(account.GetTwitterCredential(), account.Username, account.UserId);
else
- this.tw.Initialize(TwitterAppToken.GetDefault(), "", "", "", 0L);
+ this.tw.Initialize(new TwitterCredentialNone(), "", 0L);
this.tw.RestrictFavCheck = this.settings.Common.RestrictFavCheck;
this.tw.ReadOwnPost = this.settings.Common.ReadOwnPost;
/// <summary>
/// ツイート投稿前のフッター付与などの前処理を行います
/// </summary>
- private string FormatStatusText(string statusText)
+ internal string FormatStatusText(string statusText)
{
statusText = statusText.Replace("\r\n", "\n");
if (!forceupdate && currentPost.Equals(oldDisplayPost))
return;
- var loadTasks = new List<Task>
- {
- this.tweetDetailsView.ShowPostDetails(currentPost),
- };
+ var loadTasks = new TaskCollection();
+ loadTasks.Add(() => this.tweetDetailsView.ShowPostDetails(currentPost));
if (this.settings.Common.PreviewEnable)
{
oldTokenSource?.Cancel();
var token = this.thumbnailTokenSource!.Token;
- loadTasks.Add(this.PrepareThumbnailControl(currentPost, token));
+ loadTasks.Add(() => this.PrepareThumbnailControl(currentPost, token));
}
else
{
this.SplitContainer3.Panel2Collapsed = true;
}
- async Task DelayedTasks()
- {
- try
- {
- await Task.WhenAll(loadTasks);
- }
- catch (OperationCanceledException)
- {
- }
- }
-
// サムネイルの読み込みを待たずに次に選択されたツイートを表示するため await しない
- _ = DelayedTasks();
+ _ = loadTasks
+ .IgnoreException(x => x is OperationCanceledException)
+ .RunAll();
}
private async Task PrepareThumbnailControl(PostClass post, CancellationToken token)
this.ModifySettingCommon = false;
lock (this.syncObject)
{
- this.settings.Common.UserName = this.tw.Username;
- this.settings.Common.UserId = this.tw.UserId;
- this.settings.Common.Token = this.tw.AccessToken;
- this.settings.Common.TokenSecret = this.tw.AccessTokenSecret;
this.settings.Common.SortOrder = (int)this.statuses.SortOrder;
this.settings.Common.SortColumn = this.statuses.SortMode switch
{
break;
case UserTimelineTabModel userTab:
tabSetting.User = userTab.ScreenName;
+ tabSetting.UserId = userTab.UserId;
break;
case PublicSearchTabModel searchTab:
tabSetting.SearchWords = searchTab.SearchWords;
if (endpointName == null)
{
+ var authByCookie = this.tw.Api.AuthType == APIAuthType.TwitterComCookie;
+
// 表示中のタブに応じて更新
endpointName = tabType switch
{
- MyCommon.TabUsageType.Home => GetTimelineRequest.EndpointName,
+ MyCommon.TabUsageType.Home => "/statuses/home_timeline",
MyCommon.TabUsageType.UserDefined => "/statuses/home_timeline",
MyCommon.TabUsageType.Mentions => "/statuses/mentions_timeline",
MyCommon.TabUsageType.Favorites => "/favorites/list",
MyCommon.TabUsageType.DirectMessage => "/direct_messages/events/list",
- MyCommon.TabUsageType.UserTimeline => "/statuses/user_timeline",
- MyCommon.TabUsageType.Lists => "/lists/statuses",
- MyCommon.TabUsageType.PublicSearch => "/search/tweets",
+ MyCommon.TabUsageType.UserTimeline =>
+ authByCookie ? UserTweetsAndRepliesRequest.EndpointName : "/statuses/user_timeline",
+ MyCommon.TabUsageType.Lists =>
+ authByCookie ? ListLatestTweetsTimelineRequest.EndpointName : "/lists/statuses",
+ MyCommon.TabUsageType.PublicSearch =>
+ authByCookie ? SearchTimelineRequest.EndpointName : "/search/tweets",
MyCommon.TabUsageType.Related => "/statuses/show/:id",
_ => null,
};
}
else
{
- // 表示中のタブに関連する endpoint であれば更新
- bool update;
- if (endpointName == GetTimelineRequest.EndpointName)
- {
- update = tabType == MyCommon.TabUsageType.Home || tabType == MyCommon.TabUsageType.UserDefined;
- }
- else
- {
- update = endpointName switch
- {
- "/statuses/mentions_timeline" => tabType == MyCommon.TabUsageType.Mentions,
- "/favorites/list" => tabType == MyCommon.TabUsageType.Favorites,
- "/direct_messages/events/list" => tabType == MyCommon.TabUsageType.DirectMessage,
- "/statuses/user_timeline" => tabType == MyCommon.TabUsageType.UserTimeline,
- "/lists/statuses" => tabType == MyCommon.TabUsageType.Lists,
- "/search/tweets" => tabType == MyCommon.TabUsageType.PublicSearch,
- "/statuses/show/:id" => tabType == MyCommon.TabUsageType.Related,
- _ => false,
- };
- }
-
- if (update)
- {
- this.toolStripApiGauge.ApiEndpoint = endpointName;
- }
+ var currentEndpointName = this.toolStripApiGauge.ApiEndpoint;
+ this.toolStripApiGauge.ApiEndpoint = currentEndpointName;
}
}
{
this.Visible = false;
}
- if (this.initialLayout && this.settings.Local != null && this.WindowState == FormWindowState.Normal && this.Visible)
+ if (this.WindowState != FormWindowState.Minimized)
{
- // 現在の DPI と設定保存時の DPI との比を取得する
- var configScaleFactor = this.settings.Local.GetConfigScaleFactor(this.CurrentAutoScaleDimensions);
+ this.formWindowState = this.WindowState;
+ }
+ }
- this.ClientSize = ScaleBy(configScaleFactor, this.settings.Local.FormSize);
+ private void ApplyLayoutFromSettings()
+ {
+ // 現在の DPI と設定保存時の DPI との比を取得する
+ var configScaleFactor = this.settings.Local.GetConfigScaleFactor(this.CurrentAutoScaleDimensions);
- // Splitterの位置設定
- var splitterDistance = ScaleBy(configScaleFactor.Height, this.settings.Local.SplitterDistance);
- if (splitterDistance > this.SplitContainer1.Panel1MinSize &&
- splitterDistance < this.SplitContainer1.Height - this.SplitContainer1.Panel2MinSize - this.SplitContainer1.SplitterWidth)
- {
- this.SplitContainer1.SplitterDistance = splitterDistance;
- }
+ this.ClientSize = ScaleBy(configScaleFactor, this.settings.Local.FormSize);
- // 発言欄複数行
- this.StatusText.Multiline = this.settings.Local.StatusMultiline;
- if (this.StatusText.Multiline)
- {
- var statusTextHeight = ScaleBy(configScaleFactor.Height, this.settings.Local.StatusTextHeight);
- var dis = this.SplitContainer2.Height - statusTextHeight - this.SplitContainer2.SplitterWidth;
- if (dis > this.SplitContainer2.Panel1MinSize && dis < this.SplitContainer2.Height - this.SplitContainer2.Panel2MinSize - this.SplitContainer2.SplitterWidth)
- {
- this.SplitContainer2.SplitterDistance = this.SplitContainer2.Height - statusTextHeight - this.SplitContainer2.SplitterWidth;
- }
- this.StatusText.Height = statusTextHeight;
- }
- else
+ // Splitterの位置設定
+ var splitterDistance = ScaleBy(configScaleFactor.Height, this.settings.Local.SplitterDistance);
+ if (splitterDistance > this.SplitContainer1.Panel1MinSize &&
+ splitterDistance < this.SplitContainer1.Height - this.SplitContainer1.Panel2MinSize - this.SplitContainer1.SplitterWidth)
+ {
+ this.SplitContainer1.SplitterDistance = splitterDistance;
+ }
+
+ // 発言欄複数行
+ this.StatusText.Multiline = this.settings.Local.StatusMultiline;
+ if (this.StatusText.Multiline)
+ {
+ var statusTextHeight = ScaleBy(configScaleFactor.Height, this.settings.Local.StatusTextHeight);
+ var dis = this.SplitContainer2.Height - statusTextHeight - this.SplitContainer2.SplitterWidth;
+ if (dis > this.SplitContainer2.Panel1MinSize && dis < this.SplitContainer2.Height - this.SplitContainer2.Panel2MinSize - this.SplitContainer2.SplitterWidth)
{
- if (this.SplitContainer2.Height - this.SplitContainer2.Panel2MinSize - this.SplitContainer2.SplitterWidth > 0)
- {
- this.SplitContainer2.SplitterDistance = this.SplitContainer2.Height - this.SplitContainer2.Panel2MinSize - this.SplitContainer2.SplitterWidth;
- }
+ this.SplitContainer2.SplitterDistance = this.SplitContainer2.Height - statusTextHeight - this.SplitContainer2.SplitterWidth;
}
-
- var previewDistance = ScaleBy(configScaleFactor.Width, this.settings.Local.PreviewDistance);
- if (previewDistance > this.SplitContainer3.Panel1MinSize && previewDistance < this.SplitContainer3.Width - this.SplitContainer3.Panel2MinSize - this.SplitContainer3.SplitterWidth)
+ this.StatusText.Height = statusTextHeight;
+ }
+ else
+ {
+ if (this.SplitContainer2.Height - this.SplitContainer2.Panel2MinSize - this.SplitContainer2.SplitterWidth > 0)
{
- this.SplitContainer3.SplitterDistance = previewDistance;
+ this.SplitContainer2.SplitterDistance = this.SplitContainer2.Height - this.SplitContainer2.Panel2MinSize - this.SplitContainer2.SplitterWidth;
}
-
- // Panel2Collapsed は SplitterDistance の設定を終えるまで true にしない
- this.SplitContainer3.Panel2Collapsed = true;
-
- this.initialLayout = false;
}
- if (this.WindowState != FormWindowState.Minimized)
+
+ var previewDistance = ScaleBy(configScaleFactor.Width, this.settings.Local.PreviewDistance);
+ if (previewDistance > this.SplitContainer3.Panel1MinSize && previewDistance < this.SplitContainer3.Width - this.SplitContainer3.Panel2MinSize - this.SplitContainer3.SplitterWidth)
{
- this.formWindowState = this.WindowState;
+ this.SplitContainer3.SplitterDistance = previewDistance;
}
+
+ // Panel2Collapsed は SplitterDistance の設定を終えるまで true にしない
+ this.SplitContainer3.Panel2Collapsed = true;
+ this.initialLayout = false;
}
private void PlaySoundMenuItem_CheckedChanged(object sender, EventArgs e)
private async void TweenMain_Shown(object sender, EventArgs e)
{
this.NotifyIcon1.Visible = true;
+ this.StartTimers();
+
+ if (this.settings.IsFirstRun)
+ {
+ // 初回起動時だけ右下のメニューを目立たせる
+ this.HashStripSplitButton.ShowDropDown();
+ }
if (this.IsNetworkAvailable())
{
- var loadTasks = new List<Task>
- {
- this.RefreshMuteUserIdsAsync(),
- this.RefreshBlockIdsAsync(),
- this.RefreshNoRetweetIdsAsync(),
- this.RefreshTwitterConfigurationAsync(),
- this.RefreshTabAsync<HomeTabModel>(),
- this.RefreshTabAsync<MentionsTabModel>(),
- this.RefreshTabAsync<DirectMessagesTabModel>(),
- this.RefreshTabAsync<PublicSearchTabModel>(),
- this.RefreshTabAsync<UserTimelineTabModel>(),
- this.RefreshTabAsync<ListTimelineTabModel>(),
- };
+ var loadTasks = new TaskCollection();
+
+ loadTasks.Add(new[]
+ {
+ this.RefreshMuteUserIdsAsync,
+ this.RefreshBlockIdsAsync,
+ this.RefreshNoRetweetIdsAsync,
+ this.RefreshTwitterConfigurationAsync,
+ this.RefreshTabAsync<HomeTabModel>,
+ this.RefreshTabAsync<MentionsTabModel>,
+ this.RefreshTabAsync<DirectMessagesTabModel>,
+ this.RefreshTabAsync<PublicSearchTabModel>,
+ this.RefreshTabAsync<UserTimelineTabModel>,
+ this.RefreshTabAsync<ListTimelineTabModel>,
+ });
if (this.settings.Common.StartupFollowers)
- loadTasks.Add(this.RefreshFollowerIdsAsync());
+ loadTasks.Add(this.RefreshFollowerIdsAsync);
if (this.settings.Common.GetFav)
- loadTasks.Add(this.RefreshTabAsync<FavoritesTabModel>());
+ loadTasks.Add(this.RefreshTabAsync<FavoritesTabModel>);
- var allTasks = Task.WhenAll(loadTasks);
+ var allTasks = loadTasks.RunAll();
var i = 0;
while (true)
}
// 取得失敗の場合は再試行する
- var reloadTasks = new List<Task>();
+ var reloadTasks = new TaskCollection();
if (!this.tw.GetFollowersSuccess && this.settings.Common.StartupFollowers)
- reloadTasks.Add(this.RefreshFollowerIdsAsync());
+ reloadTasks.Add(() => this.RefreshFollowerIdsAsync());
if (!this.tw.GetNoRetweetSuccess)
- reloadTasks.Add(this.RefreshNoRetweetIdsAsync());
+ reloadTasks.Add(() => this.RefreshNoRetweetIdsAsync());
if (this.tw.Configuration.PhotoSizeLimit == 0)
- reloadTasks.Add(this.RefreshTwitterConfigurationAsync());
+ reloadTasks.Add(() => this.RefreshTwitterConfigurationAsync());
- await Task.WhenAll(reloadTasks);
+ await reloadTasks.RunAll();
}
this.initial = false;
+ }
+
+ private void StartTimers()
+ {
+ if (!this.StopRefreshAllMenuItem.Checked)
+ this.timelineScheduler.Enabled = true;
- this.timelineScheduler.Enabled = true;
+ this.selectionDebouncer.Enabled = true;
+ this.saveConfigDebouncer.Enabled = true;
+ this.thumbGenerator.ImgAzyobuziNet.AutoUpdate = true;
}
private async Task DoGetFollowersMenu()
{
if (!this.ExistCurrentPost) return;
this.doFavRetweetFlags = true;
- var retweetTask = this.DoReTweetOfficial(true);
+
+ var tasks = new TaskCollection();
+ tasks.Add(() => this.DoReTweetOfficial(true));
+
if (this.doFavRetweetFlags)
{
this.doFavRetweetFlags = false;
- var favoriteTask = this.FavoriteChange(true, false);
-
- await Task.WhenAll(retweetTask, favoriteTask);
- }
- else
- {
- await retweetTask;
+ tasks.Add(() => this.FavoriteChange(true, false));
}
+
+ await tasks.RunAll();
}
private async Task FavoritesRetweetUnofficial()
try
{
- var task = this.tw.Api.UsersShow(id);
+ var task = this.tw.GetUserInfo(id);
user = await dialog.WaitForAsync(this, task);
}
catch (WebApiException ex)