Count = 20,
};
- var tweets = await request.Send(mock.Object).ConfigureAwait(false);
- Assert.Single(tweets);
+ var response = await request.Send(mock.Object).ConfigureAwait(false);
+ Assert.Single(response.Tweets);
+ Assert.Equal("DAABCgABF0HfRMi__7QKAAIVAxUYmFWQAwgAAwAAAAIAAA", response.CursorBottom);
mock.VerifyAll();
}
+
+ [Fact]
+ public async Task Send_RequestCursor_Test()
+ {
+ using var responseStream = File.OpenRead("Resources/Responses/ListLatestTweetsTimeline_SimpleTweet.json");
+
+ var mock = new Mock<IApiConnection>();
+ mock.Setup(x =>
+ x.GetStreamAsync(It.IsAny<Uri>(), It.IsAny<IDictionary<string, string>>())
+ )
+ .Callback<Uri, IDictionary<string, string>>((url, param) =>
+ {
+ Assert.Equal(new("https://twitter.com/i/api/graphql/6ClPnsuzQJ1p7-g32GQw9Q/ListLatestTweetsTimeline"), url);
+ Assert.Equal(2, param.Count);
+ Assert.Equal("""{"listId":"1675863884757110790","count":20,"cursor":"aaa"}""", param["variables"]);
+ Assert.True(param.ContainsKey("features"));
+ })
+ .ReturnsAsync(responseStream);
+
+ var request = new ListLatestTweetsTimelineRequest(listId: "1675863884757110790")
+ {
+ Count = 20,
+ Cursor = "aaa",
+ };
+
+ await request.Send(mock.Object).ConfigureAwait(false);
+ mock.VerifyAll();
+ }
}
}
using System.Threading.Tasks;
using System.Xml;
using System.Xml.Linq;
+using System.Xml.XPath;
using OpenTween.Connection;
namespace OpenTween.Api.GraphQL
public int Count { get; set; } = 20;
+ public string? Cursor { get; set; }
+
public ListLatestTweetsTimelineRequest(string listId)
=> this.ListId = listId;
["variables"] = "{" +
$@"""listId"":""{JsonUtils.EscapeJsonString(this.ListId)}""," +
$@"""count"":{this.Count}" +
+ (this.Cursor != null ? $@",""cursor"":""{JsonUtils.EscapeJsonString(this.Cursor)}""" : "") +
"}",
["features"] = "{" +
@"""rweb_lists_timeline_redesign_enabled"":true," +
};
}
- public async Task<TimelineTweet[]> Send(IApiConnection apiConnection)
+ public async Task<TimelineResponse> Send(IApiConnection apiConnection)
{
var param = this.CreateParameters();
using var stream = await apiConnection.GetStreamAsync(EndpointUri, param);
throw new WebApiException("IO Error", ex);
}
- return TimelineTweet.ExtractTimelineTweets(rootElm);
+ var tweets = TimelineTweet.ExtractTimelineTweets(rootElm);
+ var cursorBottom = rootElm.XPathSelectElement("//content[__typename[text()='TimelineTimelineCursor']][cursorType[text()='Bottom']]/value")?.Value;
+
+ return new(tweets, cursorBottom);
}
}
}
--- /dev/null
+// OpenTween - Client of Twitter
+// Copyright (c) 2023 kim_upsilon (@kim_upsilon) <https://upsilo.net/~upsilon/>
+// All rights reserved.
+//
+// This file is part of OpenTween.
+//
+// This program is free software; you can redistribute it and/or modify it
+// under the terms of the GNU General Public License as published by the Free
+// Software Foundation; either version 3 of the License, or (at your option)
+// any later version.
+//
+// This program is distributed in the hope that it will be useful, but
+// WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
+// or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+// for more details.
+//
+// You should have received a copy of the GNU General Public License along
+// with this program. If not, see <http://www.gnu.org/licenses/>, or write to
+// the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor,
+// Boston, MA 02110-1301, USA.
+
+#nullable enable
+
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace OpenTween.Api.GraphQL
+{
+ public record TimelineResponse(
+ TimelineTweet[] Tweets,
+ string? CursorBottom
+ );
+}
public long OldestId { get; set; } = long.MaxValue;
+ public string? CursorBottom { get; set; }
+
public ListTimelineTabModel(string tabName, ListElement list)
: base(tabName)
{
更新履歴
==== Unreleased
+ * NEW: graphqlエンドポイントを使用しているタブでの「前データを取得」に対応
==== Ver 3.6.2(2023/07/07)
* FIX: リプライ制限されたツイートのRTがリストのタイムラインに含まれているとエラーになる不具合を修正
var request = new ListLatestTweetsTimelineRequest(tab.ListInfo.Id.ToString())
{
Count = count,
+ Cursor = more ? tab.CursorBottom : null,
};
- var timelineTweets = await request.Send(this.Api.Connection)
+ var response = await request.Send(this.Api.Connection)
.ConfigureAwait(false);
- statuses = timelineTweets.Select(x => x.ToTwitterStatus()).ToArray();
+
+ statuses = response.Tweets.Select(x => x.ToTwitterStatus()).ToArray();
+ tab.CursorBottom = response.CursorBottom;
}
else if (more)
{