2019-05-10 19:15:11 +02:00
|
|
|
import 'package:comunic/enums/post_kind.dart';
|
2019-07-05 11:40:43 +02:00
|
|
|
import 'package:comunic/enums/post_target.dart';
|
2019-05-10 19:15:11 +02:00
|
|
|
import 'package:comunic/enums/post_visibility_level.dart';
|
|
|
|
import 'package:comunic/enums/user_access_levels.dart';
|
2019-05-16 14:52:22 +02:00
|
|
|
import 'package:comunic/helpers/comments_helper.dart';
|
2019-06-28 11:32:36 +02:00
|
|
|
import 'package:comunic/helpers/survey_helper.dart';
|
2020-04-18 16:07:56 +02:00
|
|
|
import 'package:comunic/helpers/websocket_helper.dart';
|
2019-05-16 14:52:22 +02:00
|
|
|
import 'package:comunic/lists/comments_list.dart';
|
2019-05-10 19:15:11 +02:00
|
|
|
import 'package:comunic/lists/posts_list.dart';
|
|
|
|
import 'package:comunic/models/api_request.dart';
|
2020-04-16 14:07:21 +02:00
|
|
|
import 'package:comunic/models/displayed_content.dart';
|
2019-07-05 11:40:43 +02:00
|
|
|
import 'package:comunic/models/new_post.dart';
|
2019-05-10 19:15:11 +02:00
|
|
|
import 'package:comunic/models/post.dart';
|
2020-04-24 13:35:05 +02:00
|
|
|
import 'package:http_parser/http_parser.dart';
|
2019-05-10 19:15:11 +02:00
|
|
|
|
|
|
|
/// Posts helper
|
|
|
|
///
|
|
|
|
/// @author Pierre HUBERT
|
|
|
|
|
|
|
|
const _APIPostsVisibilityLevelMap = {
|
|
|
|
"public": PostVisibilityLevel.PUBLIC,
|
|
|
|
"friends": PostVisibilityLevel.FRIENDS,
|
|
|
|
"private": PostVisibilityLevel.USER,
|
|
|
|
"members": PostVisibilityLevel.GROUP_MEMBERS
|
|
|
|
};
|
|
|
|
|
|
|
|
const _APIPostsKindsMap = {
|
|
|
|
"text": PostKind.TEXT,
|
|
|
|
"image": PostKind.IMAGE,
|
|
|
|
"weblink": PostKind.WEB_LINK,
|
|
|
|
"pdf": PostKind.PDF,
|
|
|
|
"countdown": PostKind.COUNTDOWN,
|
|
|
|
"survey": PostKind.SURVEY,
|
|
|
|
"youtube": PostKind.YOUTUBE
|
|
|
|
};
|
|
|
|
|
|
|
|
const _APIUserAccessMap = {
|
|
|
|
"no-access": UserAccessLevels.NONE,
|
|
|
|
"basic": UserAccessLevels.BASIC,
|
|
|
|
"intermediate": UserAccessLevels.INTERMEDIATE,
|
|
|
|
"full": UserAccessLevels.FULL
|
|
|
|
};
|
|
|
|
|
2019-07-05 11:40:43 +02:00
|
|
|
const _APIPostsTargetKindsMap = {
|
|
|
|
PostTarget.USER_PAGE: "user",
|
|
|
|
PostTarget.GROUP_PAGE: "group"
|
|
|
|
};
|
|
|
|
|
2019-05-10 19:15:11 +02:00
|
|
|
class PostsHelper {
|
2020-04-18 16:07:56 +02:00
|
|
|
/// Stores the list of posts we are registered to
|
|
|
|
///
|
|
|
|
/// First int = post ID
|
|
|
|
/// Second int = number of registered people
|
|
|
|
static final _registeredPosts = Map<int, int>();
|
|
|
|
|
2019-05-10 19:15:11 +02:00
|
|
|
/// Get the list of latest posts. Return the list of posts or null in case of
|
|
|
|
/// failure
|
2019-06-15 16:25:06 +02:00
|
|
|
Future<PostsList> getLatest({int from = 0}) async {
|
2019-06-28 11:32:36 +02:00
|
|
|
final response =
|
2020-04-24 13:35:05 +02:00
|
|
|
await APIRequest(uri: "posts/get_latest", needLogin: true, args: {
|
2019-06-28 11:32:36 +02:00
|
|
|
"include_groups": true.toString(),
|
|
|
|
"startFrom": from.toString(),
|
|
|
|
}).exec();
|
2019-05-10 19:15:11 +02:00
|
|
|
|
|
|
|
if (response.code != 200) return null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Parse & return the list of posts
|
|
|
|
return PostsList()..addAll(response.getArray().map((f) => _apiToPost(f)));
|
|
|
|
} catch (e) {
|
|
|
|
print(e.toString());
|
2019-06-10 14:47:27 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the list of posts of a user
|
|
|
|
Future<PostsList> getUserPosts(int userID, {int from = 0}) async {
|
2020-04-16 09:17:10 +02:00
|
|
|
final response = await (APIRequest(uri: "posts/get_user", needLogin: true)
|
2020-04-24 13:35:05 +02:00
|
|
|
..addInt("userID", userID)
|
|
|
|
..addInt("startFrom", from == 0 ? 0 : from - 1))
|
2019-06-28 11:32:36 +02:00
|
|
|
.exec();
|
2019-06-10 14:47:27 +02:00
|
|
|
|
|
|
|
if (response.code != 200) return null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Parse & return the list of posts
|
|
|
|
return PostsList()..addAll(response.getArray().map((f) => _apiToPost(f)));
|
|
|
|
} catch (e) {
|
|
|
|
print(e.toString());
|
2019-05-10 19:15:11 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-16 08:24:34 +02:00
|
|
|
/// Get the list of posts of a group
|
|
|
|
Future<PostsList> getGroupPosts(int groupID, {int from = 0}) async {
|
|
|
|
final response = await (APIRequest(uri: "posts/get_group", needLogin: true)
|
2020-04-24 13:35:05 +02:00
|
|
|
..addInt("groupID", groupID)
|
|
|
|
..addInt("startFrom", from == 0 ? 0 : from - 1))
|
2020-04-16 08:24:34 +02:00
|
|
|
.exec();
|
|
|
|
|
|
|
|
if (response.code != 200) return null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Parse & return the list of posts
|
|
|
|
return PostsList()..addAll(response.getArray().map((f) => _apiToPost(f)));
|
|
|
|
} catch (e) {
|
|
|
|
print(e.toString());
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-02 12:46:17 +01:00
|
|
|
/// Get a single post information
|
|
|
|
Future<Post> getSingle(int postID) async {
|
|
|
|
final response = await APIRequest(
|
|
|
|
uri: "posts/get_single",
|
|
|
|
args: {"postID": postID.toString()},
|
|
|
|
needLogin: true,
|
|
|
|
).exec();
|
|
|
|
|
|
|
|
if (!response.isOK)
|
|
|
|
throw Exception("Could not get information about the post!");
|
|
|
|
|
|
|
|
return _apiToPost(response.getObject());
|
|
|
|
}
|
|
|
|
|
2019-07-05 11:40:43 +02:00
|
|
|
/// Create a new post
|
|
|
|
///
|
|
|
|
/// This function crash in case of error
|
|
|
|
Future<void> createPost(NewPost post) async {
|
|
|
|
APIRequest request =
|
2020-04-24 13:35:05 +02:00
|
|
|
APIRequest(uri: "posts/create", needLogin: true, args: {
|
2019-07-05 11:40:43 +02:00
|
|
|
"kind-page": _APIPostsTargetKindsMap[post.target],
|
|
|
|
"kind-id": post.targetID.toString(),
|
|
|
|
"visibility": _APIPostsVisibilityLevelMap.map(
|
2020-04-24 13:35:05 +02:00
|
|
|
(s, v) => MapEntry(v, s))[post.visibility],
|
2019-07-05 11:40:43 +02:00
|
|
|
"kind": _APIPostsKindsMap.map((s, k) => MapEntry(k, s))[post.kind],
|
|
|
|
"content": post.content
|
|
|
|
});
|
|
|
|
|
|
|
|
switch (post.kind) {
|
|
|
|
case PostKind.TEXT:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PostKind.IMAGE:
|
2021-02-07 17:09:08 +01:00
|
|
|
request.addPickedFile("image", post.image);
|
2019-07-05 11:40:43 +02:00
|
|
|
break;
|
|
|
|
|
2020-04-25 14:38:15 +02:00
|
|
|
case PostKind.WEB_LINK:
|
|
|
|
request.addString("url", post.url);
|
|
|
|
break;
|
|
|
|
|
2020-04-24 13:35:05 +02:00
|
|
|
case PostKind.PDF:
|
|
|
|
request.addBytesFile(
|
|
|
|
"pdf",
|
|
|
|
BytesFile("file.pdf", post.pdf,
|
|
|
|
type: MediaType.parse("application/pdf")));
|
|
|
|
break;
|
|
|
|
|
2020-04-25 08:23:52 +02:00
|
|
|
case PostKind.COUNTDOWN:
|
|
|
|
request.addInt(
|
|
|
|
"time-end", (post.timeEnd.millisecondsSinceEpoch / 1000).floor());
|
|
|
|
break;
|
|
|
|
|
2020-04-25 11:58:45 +02:00
|
|
|
case PostKind.SURVEY:
|
|
|
|
request.addString("question", post.survey.question);
|
|
|
|
request.addString("answers", post.survey.answers.join("<>"));
|
2020-05-18 17:58:54 +02:00
|
|
|
request.addBool("allowNewAnswers", post.survey.allowNewChoicesCreation);
|
2020-04-25 11:58:45 +02:00
|
|
|
break;
|
|
|
|
|
2020-04-25 17:16:33 +02:00
|
|
|
case PostKind.YOUTUBE:
|
|
|
|
request.addString("youtube_id", post.youtubeId);
|
|
|
|
break;
|
|
|
|
|
2019-07-05 11:40:43 +02:00
|
|
|
default:
|
|
|
|
throw Exception("Unsupported post type :" + post.kind.toString());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
final response = await request.execWithFiles();
|
|
|
|
|
|
|
|
if (!response.isOK) throw Exception("Could not create the post !");
|
|
|
|
}
|
|
|
|
|
2019-05-19 17:42:09 +02:00
|
|
|
/// Update a post content
|
|
|
|
Future<bool> updateContent(int id, String newContent) async {
|
|
|
|
return (await APIRequest(
|
|
|
|
uri: "posts/update_content",
|
|
|
|
needLogin: true,
|
|
|
|
args: {
|
|
|
|
"postID": id.toString(),
|
|
|
|
"new_content": newContent,
|
|
|
|
},
|
|
|
|
).exec())
|
|
|
|
.isOK;
|
|
|
|
}
|
|
|
|
|
2019-05-23 18:27:43 +02:00
|
|
|
/// Update a post visibility
|
|
|
|
Future<bool> setVisibility(int id, PostVisibilityLevel level) async {
|
|
|
|
return (await APIRequest(
|
|
|
|
uri: "posts/set_visibility_level",
|
|
|
|
needLogin: true,
|
|
|
|
args: {
|
|
|
|
"postID": id.toString(),
|
2019-06-28 11:32:36 +02:00
|
|
|
"new_level":
|
2020-04-24 13:35:05 +02:00
|
|
|
_APIPostsVisibilityLevelMap.map((k, v) => MapEntry(v, k))[level]
|
2019-05-23 18:27:43 +02:00
|
|
|
},
|
|
|
|
).exec())
|
|
|
|
.isOK;
|
|
|
|
}
|
|
|
|
|
2019-05-19 14:54:09 +02:00
|
|
|
/// Delete a post
|
|
|
|
Future<bool> delete(int id) async {
|
|
|
|
return (await APIRequest(
|
|
|
|
uri: "posts/delete",
|
|
|
|
needLogin: true,
|
|
|
|
args: {"postID": id.toString()},
|
|
|
|
).exec())
|
|
|
|
.isOK;
|
|
|
|
}
|
|
|
|
|
2020-04-18 16:07:56 +02:00
|
|
|
/// Register to a post events
|
|
|
|
Future<void> registerPostEvents(int id) async {
|
|
|
|
if (_registeredPosts.containsKey(id))
|
|
|
|
_registeredPosts[id]++;
|
|
|
|
else {
|
|
|
|
_registeredPosts[id] = 1;
|
|
|
|
await ws("\$main/register_post", {"postID": id});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Un-register to post events
|
|
|
|
Future<void> unregisterPostEvents(int id) async {
|
|
|
|
if (!_registeredPosts.containsKey(id)) return;
|
|
|
|
|
|
|
|
_registeredPosts[id]--;
|
|
|
|
|
|
|
|
if (_registeredPosts[id] <= 0) {
|
|
|
|
_registeredPosts.remove(id);
|
|
|
|
await ws("\$main/unregister_post", {"postID": id});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-10 19:15:11 +02:00
|
|
|
/// Turn an API entry into a [Post] object
|
|
|
|
Post _apiToPost(Map<String, dynamic> map) {
|
2019-06-28 11:32:36 +02:00
|
|
|
final postKind = _APIPostsKindsMap[map["kind"]];
|
|
|
|
|
2019-05-16 14:52:22 +02:00
|
|
|
// Parse comments
|
|
|
|
CommentsList comments;
|
|
|
|
if (map["comments"] != null) {
|
|
|
|
comments = CommentsList();
|
|
|
|
map["comments"]
|
|
|
|
.forEach((v) => comments.add(CommentsHelper.apiToComment(v)));
|
|
|
|
}
|
|
|
|
|
2019-06-28 11:32:36 +02:00
|
|
|
final survey = postKind == PostKind.SURVEY
|
|
|
|
? SurveyHelper.apiToSurvey(map["data_survey"])
|
|
|
|
: null;
|
|
|
|
|
2019-05-10 19:15:11 +02:00
|
|
|
return Post(
|
2019-06-28 11:32:36 +02:00
|
|
|
id: map["ID"],
|
|
|
|
userID: map["userID"],
|
|
|
|
userPageID: map["user_page_id"],
|
|
|
|
groupID: map["group_id"],
|
|
|
|
timeSent: map["post_time"],
|
2020-04-16 14:07:21 +02:00
|
|
|
content: DisplayedString(map["content"]),
|
2019-06-28 11:32:36 +02:00
|
|
|
visibilityLevel: _APIPostsVisibilityLevelMap[map["visibility_level"]],
|
|
|
|
kind: postKind,
|
|
|
|
fileSize: map["file_size"],
|
|
|
|
fileType: map["file_type"],
|
|
|
|
filePath: map["file_path"],
|
|
|
|
fileURL: map["file_path_url"],
|
|
|
|
timeEnd: map["time_end"],
|
|
|
|
linkURL: map["link_url"],
|
|
|
|
linkTitle: map["link_title"],
|
|
|
|
linkDescription: map["link_description"],
|
|
|
|
linkImage: map["link_image"],
|
|
|
|
likes: map["likes"],
|
|
|
|
userLike: map["userlike"],
|
|
|
|
access: _APIUserAccessMap[map["user_access"]],
|
|
|
|
comments: comments,
|
|
|
|
survey: survey);
|
2019-05-10 19:15:11 +02:00
|
|
|
}
|
|
|
|
}
|