2017-08-27 21:55:49 +00:00
|
|
|
|
using Gee;
|
|
|
|
|
using Gdk;
|
|
|
|
|
using Gtk;
|
|
|
|
|
using Markup;
|
|
|
|
|
|
|
|
|
|
using Dino.Entities;
|
|
|
|
|
|
|
|
|
|
namespace Dino.Ui.ConversationSummary {
|
|
|
|
|
|
2019-06-01 14:00:21 +00:00
|
|
|
|
public class ConversationItemSkeleton : EventBox {
|
2017-08-27 21:55:49 +00:00
|
|
|
|
|
2020-03-24 12:58:25 +00:00
|
|
|
|
public bool show_skeleton { get; set; default=false; }
|
|
|
|
|
public bool last_group_item { get; set; default=true; }
|
2019-06-01 14:00:21 +00:00
|
|
|
|
|
2017-08-27 21:55:49 +00:00
|
|
|
|
public StreamInteractor stream_interactor;
|
|
|
|
|
public Conversation conversation { get; set; }
|
2019-06-01 14:00:21 +00:00
|
|
|
|
public Plugins.MetaConversationItem item;
|
2020-05-28 15:31:31 +00:00
|
|
|
|
public bool item_in_edit_mode { get; set; }
|
2021-08-14 18:22:52 +00:00
|
|
|
|
public Entities.Message.Marked item_mark { get; set; }
|
2020-04-03 20:49:59 +00:00
|
|
|
|
public ContentMetaItem? content_meta_item = null;
|
|
|
|
|
public Widget? widget = null;
|
2017-08-27 21:55:49 +00:00
|
|
|
|
|
2019-06-01 14:00:21 +00:00
|
|
|
|
private Box image_content_box = new Box(Orientation.HORIZONTAL, 8) { visible=true };
|
|
|
|
|
private Box header_content_box = new Box(Orientation.VERTICAL, 0) { visible=true };
|
2020-03-24 12:58:25 +00:00
|
|
|
|
private ItemMetaDataHeader? metadata_header = null;
|
|
|
|
|
private AvatarImage? image = null;
|
2017-08-27 21:55:49 +00:00
|
|
|
|
|
2020-03-21 19:53:10 +00:00
|
|
|
|
public ConversationItemSkeleton(StreamInteractor stream_interactor, Conversation conversation, Plugins.MetaConversationItem item, bool initial_item) {
|
2017-08-27 21:55:49 +00:00
|
|
|
|
this.stream_interactor = stream_interactor;
|
2019-09-10 23:19:24 +00:00
|
|
|
|
this.conversation = conversation;
|
|
|
|
|
this.item = item;
|
2020-04-03 20:49:59 +00:00
|
|
|
|
this.content_meta_item = item as ContentMetaItem;
|
2019-06-01 14:00:21 +00:00
|
|
|
|
this.get_style_context().add_class("message-box");
|
2017-08-27 21:55:49 +00:00
|
|
|
|
|
2020-05-28 15:31:31 +00:00
|
|
|
|
item.bind_property("in-edit-mode", this, "item-in-edit-mode");
|
2021-02-17 20:50:23 +00:00
|
|
|
|
this.notify["item-in-edit-mode"].connect(update_edit_mode);
|
2020-04-03 20:49:59 +00:00
|
|
|
|
|
2021-08-14 18:22:52 +00:00
|
|
|
|
item.bind_property("mark", this, "item-mark", BindingFlags.SYNC_CREATE);
|
|
|
|
|
this.notify["item-mark"].connect(update_error_mode);
|
|
|
|
|
update_error_mode();
|
|
|
|
|
|
2020-04-03 20:49:59 +00:00
|
|
|
|
widget = item.get_widget(Plugins.WidgetType.GTK) as Widget;
|
2017-11-26 18:28:44 +00:00
|
|
|
|
if (widget != null) {
|
2019-06-16 13:17:08 +00:00
|
|
|
|
widget.valign = Align.END;
|
2019-06-01 14:00:21 +00:00
|
|
|
|
header_content_box.add(widget);
|
2017-08-27 21:55:49 +00:00
|
|
|
|
}
|
2017-11-21 21:17:04 +00:00
|
|
|
|
|
2019-06-01 14:00:21 +00:00
|
|
|
|
image_content_box.add(header_content_box);
|
2020-03-21 19:53:10 +00:00
|
|
|
|
|
|
|
|
|
if (initial_item) {
|
|
|
|
|
this.add(image_content_box);
|
|
|
|
|
} else {
|
2020-04-22 13:44:12 +00:00
|
|
|
|
Revealer revealer = new Revealer() { transition_duration=200, transition_type=RevealerTransitionType.SLIDE_UP, reveal_child=false, visible=true };
|
2020-03-21 19:53:10 +00:00
|
|
|
|
revealer.add_with_properties(image_content_box);
|
|
|
|
|
this.add(revealer);
|
2020-04-22 13:44:12 +00:00
|
|
|
|
revealer.reveal_child = true;
|
2019-06-01 14:00:21 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-21 19:53:10 +00:00
|
|
|
|
|
2019-06-01 14:00:21 +00:00
|
|
|
|
this.notify["show-skeleton"].connect(update_margin);
|
|
|
|
|
this.notify["last-group-item"].connect(update_margin);
|
|
|
|
|
|
|
|
|
|
update_margin();
|
2017-08-27 21:55:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-04-03 20:49:59 +00:00
|
|
|
|
private void update_margin() {
|
2020-03-24 12:58:25 +00:00
|
|
|
|
if (item.requires_header && show_skeleton && metadata_header == null) {
|
|
|
|
|
metadata_header = new ItemMetaDataHeader(stream_interactor, conversation, item) { visible=true };
|
|
|
|
|
header_content_box.add(metadata_header);
|
|
|
|
|
header_content_box.reorder_child(metadata_header, 0);
|
|
|
|
|
}
|
|
|
|
|
if (item.requires_avatar && show_skeleton && image == null) {
|
|
|
|
|
image = new AvatarImage() { margin_top=2, valign=Align.START, visible=true, allow_gray = false };
|
|
|
|
|
image.set_conversation_participant(stream_interactor, conversation, item.jid);
|
|
|
|
|
image_content_box.add(image);
|
|
|
|
|
image_content_box.reorder_child(image, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (image != null) {
|
|
|
|
|
image.visible = this.show_skeleton;
|
|
|
|
|
}
|
2019-09-10 23:19:24 +00:00
|
|
|
|
if (metadata_header != null) {
|
|
|
|
|
metadata_header.visible = this.show_skeleton;
|
2019-06-01 14:00:21 +00:00
|
|
|
|
}
|
|
|
|
|
image_content_box.margin_start = this.show_skeleton ? 15 : 58;
|
2019-09-09 17:47:11 +00:00
|
|
|
|
image_content_box.margin_end = 15;
|
2019-06-01 14:00:21 +00:00
|
|
|
|
|
|
|
|
|
if (this.show_skeleton && this.last_group_item) {
|
|
|
|
|
image_content_box.margin_top = 8;
|
|
|
|
|
image_content_box.margin_bottom = 8;
|
|
|
|
|
} else {
|
|
|
|
|
image_content_box.margin_top = 4;
|
|
|
|
|
image_content_box.margin_bottom = 4;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-02-17 20:50:23 +00:00
|
|
|
|
|
|
|
|
|
private void update_edit_mode() {
|
|
|
|
|
if (item.in_edit_mode) {
|
|
|
|
|
this.get_style_context().add_class("edit-mode");
|
|
|
|
|
} else {
|
|
|
|
|
this.get_style_context().remove_class("edit-mode");
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-08-14 18:22:52 +00:00
|
|
|
|
|
|
|
|
|
private void update_error_mode() {
|
|
|
|
|
if (item_mark == Message.Marked.ERROR) {
|
|
|
|
|
this.get_style_context().add_class("error");
|
|
|
|
|
} else {
|
|
|
|
|
this.get_style_context().remove_class("error");
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-26 18:28:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-21 01:49:53 +00:00
|
|
|
|
[GtkTemplate (ui = "/im/dino/Dino/conversation_content_view/item_metadata_header.ui")]
|
2019-09-10 23:19:24 +00:00
|
|
|
|
public class ItemMetaDataHeader : Box {
|
|
|
|
|
[GtkChild] public Label name_label;
|
|
|
|
|
[GtkChild] public Label dot_label;
|
|
|
|
|
[GtkChild] public Label time_label;
|
2020-03-28 13:46:51 +00:00
|
|
|
|
public Image received_image = new Image() { opacity=0.4 };
|
2021-04-08 10:07:04 +00:00
|
|
|
|
public Widget? encryption_image = null;
|
2019-09-10 23:19:24 +00:00
|
|
|
|
|
|
|
|
|
public static IconSize ICON_SIZE_HEADER = Gtk.icon_size_register("im.dino.Dino.HEADER_ICON", 17, 12);
|
2017-11-26 18:28:44 +00:00
|
|
|
|
|
|
|
|
|
private StreamInteractor stream_interactor;
|
|
|
|
|
private Conversation conversation;
|
|
|
|
|
private Plugins.MetaConversationItem item;
|
2020-05-28 15:31:31 +00:00
|
|
|
|
public Entities.Message.Marked item_mark { get; set; }
|
2017-11-26 18:28:44 +00:00
|
|
|
|
private ArrayList<Plugins.MetaConversationItem> items = new ArrayList<Plugins.MetaConversationItem>();
|
2020-05-28 15:31:31 +00:00
|
|
|
|
private uint time_update_timeout = 0;
|
2021-08-24 17:35:00 +00:00
|
|
|
|
private ulong updated_roster_handler_id = 0;
|
2017-11-26 18:28:44 +00:00
|
|
|
|
|
2019-09-10 23:19:24 +00:00
|
|
|
|
public ItemMetaDataHeader(StreamInteractor stream_interactor, Conversation conversation, Plugins.MetaConversationItem item) {
|
2017-11-26 18:28:44 +00:00
|
|
|
|
this.stream_interactor = stream_interactor;
|
|
|
|
|
this.conversation = conversation;
|
|
|
|
|
this.item = item;
|
2019-09-10 23:19:24 +00:00
|
|
|
|
items.add(item);
|
2017-11-26 18:28:44 +00:00
|
|
|
|
|
|
|
|
|
update_name_label();
|
|
|
|
|
name_label.style_updated.connect(update_name_label);
|
2021-08-24 17:35:00 +00:00
|
|
|
|
updated_roster_handler_id = stream_interactor.get_module(RosterManager.IDENTITY).updated_roster_item.connect((account, jid, roster_item) => {
|
|
|
|
|
if (this.conversation.account.equals(account) && this.conversation.counterpart.equals(jid)) {
|
|
|
|
|
update_name_label();
|
|
|
|
|
}
|
|
|
|
|
});
|
2020-03-28 13:46:51 +00:00
|
|
|
|
|
2021-04-08 10:07:04 +00:00
|
|
|
|
conversation.notify["encryption"].connect(update_unencrypted_icon);
|
|
|
|
|
item.notify["encryption"].connect(update_encryption_icon);
|
|
|
|
|
update_encryption_icon();
|
|
|
|
|
|
|
|
|
|
this.add(received_image);
|
|
|
|
|
|
|
|
|
|
if (item.time != null) {
|
|
|
|
|
update_time();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
item.bind_property("mark", this, "item-mark");
|
|
|
|
|
this.notify["item-mark"].connect_after(update_received_mark);
|
|
|
|
|
update_received_mark();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void update_encryption_icon() {
|
2020-03-28 13:46:51 +00:00
|
|
|
|
Application app = GLib.Application.get_default() as Application;
|
|
|
|
|
|
|
|
|
|
ContentMetaItem ci = item as ContentMetaItem;
|
2021-04-08 10:07:04 +00:00
|
|
|
|
if (item.encryption != Encryption.NONE && ci != null) {
|
|
|
|
|
Widget? widget = null;
|
2020-03-28 13:46:51 +00:00
|
|
|
|
foreach(var e in app.plugin_registry.encryption_list_entries) {
|
|
|
|
|
if (e.encryption == item.encryption) {
|
2021-04-08 10:07:04 +00:00
|
|
|
|
widget = e.get_encryption_icon(conversation, ci.content_item) as Widget;
|
2020-03-28 13:46:51 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-04-08 10:07:04 +00:00
|
|
|
|
if (widget == null) {
|
|
|
|
|
widget = new Image.from_icon_name("dino-changes-prevent-symbolic", ICON_SIZE_HEADER) { opacity=0.4, visible = true };
|
|
|
|
|
}
|
|
|
|
|
update_encryption_image(widget);
|
2017-08-27 21:55:49 +00:00
|
|
|
|
}
|
2020-03-28 18:55:22 +00:00
|
|
|
|
if (item.encryption == Encryption.NONE) {
|
|
|
|
|
update_unencrypted_icon();
|
|
|
|
|
}
|
2021-04-08 10:07:04 +00:00
|
|
|
|
}
|
2020-03-28 18:55:22 +00:00
|
|
|
|
|
2021-04-08 10:07:04 +00:00
|
|
|
|
private void update_unencrypted_icon() {
|
|
|
|
|
if (item.encryption != Encryption.NONE) return;
|
|
|
|
|
|
|
|
|
|
if (conversation.encryption != Encryption.NONE && encryption_image == null) {
|
|
|
|
|
Image image = new Image() { opacity=0.4, visible = true };
|
|
|
|
|
image.set_from_icon_name("dino-changes-allowed-symbolic", ICON_SIZE_HEADER);
|
|
|
|
|
image.tooltip_text = _("Unencrypted");
|
|
|
|
|
update_encryption_image(image);
|
|
|
|
|
Util.force_error_color(image);
|
|
|
|
|
} else if (conversation.encryption == Encryption.NONE && encryption_image != null) {
|
|
|
|
|
update_encryption_image(null);
|
2020-03-26 14:27:48 +00:00
|
|
|
|
}
|
2017-11-26 18:28:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-08 10:07:04 +00:00
|
|
|
|
private void update_encryption_image(Widget? widget) {
|
|
|
|
|
if (encryption_image != null) {
|
|
|
|
|
this.remove(encryption_image);
|
|
|
|
|
encryption_image = null;
|
|
|
|
|
}
|
|
|
|
|
if (widget != null) {
|
|
|
|
|
this.add(widget);
|
|
|
|
|
this.reorder_child(widget, 3);
|
|
|
|
|
encryption_image = widget;
|
2020-03-28 18:55:22 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-26 14:27:48 +00:00
|
|
|
|
private void update_time() {
|
2020-12-04 18:11:27 +00:00
|
|
|
|
time_label.label = get_relative_time(item.time.to_local()).to_string();
|
2020-03-26 14:27:48 +00:00
|
|
|
|
|
2020-05-28 15:31:31 +00:00
|
|
|
|
time_update_timeout = Timeout.add_seconds((int) get_next_time_change(), () => {
|
2020-03-26 14:27:48 +00:00
|
|
|
|
if (this.parent == null) return false;
|
|
|
|
|
update_time();
|
|
|
|
|
return false;
|
|
|
|
|
});
|
2017-11-26 18:28:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void update_name_label() {
|
2019-10-18 14:52:29 +00:00
|
|
|
|
string display_name = Markup.escape_text(Util.get_participant_display_name(stream_interactor, conversation, item.jid));
|
2017-11-26 18:28:44 +00:00
|
|
|
|
string color = Util.get_name_hex_color(stream_interactor, conversation.account, item.jid, Util.is_dark_theme(name_label));
|
2019-09-10 23:19:24 +00:00
|
|
|
|
name_label.label = @"<span foreground=\"#$color\">$display_name</span>";
|
2017-08-27 21:55:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-11-21 21:17:04 +00:00
|
|
|
|
private void update_received_mark() {
|
2017-08-27 21:55:49 +00:00
|
|
|
|
bool all_received = true;
|
|
|
|
|
bool all_read = true;
|
2017-08-29 22:03:37 +00:00
|
|
|
|
bool all_sent = true;
|
2017-08-27 21:55:49 +00:00
|
|
|
|
foreach (Plugins.MetaConversationItem item in items) {
|
|
|
|
|
if (item.mark == Message.Marked.WONTSEND) {
|
|
|
|
|
received_image.visible = true;
|
2017-11-26 18:28:44 +00:00
|
|
|
|
received_image.set_from_icon_name("dialog-warning-symbolic", ICON_SIZE_HEADER);
|
2017-08-27 21:55:49 +00:00
|
|
|
|
Util.force_error_color(received_image);
|
|
|
|
|
Util.force_error_color(time_label);
|
2020-02-21 17:22:27 +00:00
|
|
|
|
string error_text = _("Unable to send message");
|
|
|
|
|
received_image.tooltip_text = error_text;
|
|
|
|
|
time_label.tooltip_text = error_text;
|
2017-08-27 21:55:49 +00:00
|
|
|
|
return;
|
|
|
|
|
} else if (item.mark != Message.Marked.READ) {
|
|
|
|
|
all_read = false;
|
|
|
|
|
if (item.mark != Message.Marked.RECEIVED) {
|
|
|
|
|
all_received = false;
|
2017-08-29 22:03:37 +00:00
|
|
|
|
if (item.mark == Message.Marked.UNSENT) {
|
|
|
|
|
all_sent = false;
|
|
|
|
|
}
|
2017-08-27 21:55:49 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (all_read) {
|
|
|
|
|
received_image.visible = true;
|
2017-11-26 18:28:44 +00:00
|
|
|
|
received_image.set_from_icon_name("dino-double-tick-symbolic", ICON_SIZE_HEADER);
|
2017-08-27 21:55:49 +00:00
|
|
|
|
} else if (all_received) {
|
|
|
|
|
received_image.visible = true;
|
2017-11-26 18:28:44 +00:00
|
|
|
|
received_image.set_from_icon_name("dino-tick-symbolic", ICON_SIZE_HEADER);
|
2017-08-29 22:03:37 +00:00
|
|
|
|
} else if (!all_sent) {
|
|
|
|
|
received_image.visible = true;
|
2017-11-26 18:28:44 +00:00
|
|
|
|
received_image.set_from_icon_name("image-loading-symbolic", ICON_SIZE_HEADER);
|
2017-08-27 21:55:49 +00:00
|
|
|
|
} else if (received_image.visible) {
|
2017-11-26 18:28:44 +00:00
|
|
|
|
received_image.set_from_icon_name("image-loading-symbolic", ICON_SIZE_HEADER);
|
|
|
|
|
|
2017-08-27 21:55:49 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-26 14:27:48 +00:00
|
|
|
|
private int get_next_time_change() {
|
|
|
|
|
DateTime now = new DateTime.now_local();
|
2020-12-04 18:11:27 +00:00
|
|
|
|
DateTime item_time = item.time;
|
2020-03-26 14:27:48 +00:00
|
|
|
|
TimeSpan timespan = now.difference(item_time);
|
|
|
|
|
|
|
|
|
|
if (timespan < 10 * TimeSpan.MINUTE) {
|
|
|
|
|
if (now.get_second() < item_time.get_second()) {
|
|
|
|
|
return item_time.get_second() - now.get_second();
|
|
|
|
|
} else {
|
|
|
|
|
return 60 - (now.get_second() - item_time.get_second());
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return (23 - now.get_hour()) * 3600 + (59 - now.get_minute()) * 60 + (59 - now.get_second());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-26 18:28:44 +00:00
|
|
|
|
public static string format_time(DateTime datetime, string format_24h, string format_12h) {
|
2017-08-29 19:51:08 +00:00
|
|
|
|
string format = Util.is_24h_format() ? format_24h : format_12h;
|
|
|
|
|
if (!get_charset(null)) {
|
|
|
|
|
// No UTF-8 support, use simple colon for time instead
|
|
|
|
|
format = format.replace("∶", ":");
|
|
|
|
|
}
|
|
|
|
|
return datetime.format(format);
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-09 22:31:39 +00:00
|
|
|
|
public static string get_relative_time(DateTime datetime) {
|
2017-08-27 21:55:49 +00:00
|
|
|
|
DateTime now = new DateTime.now_local();
|
|
|
|
|
TimeSpan timespan = now.difference(datetime);
|
|
|
|
|
if (timespan > 365 * TimeSpan.DAY) {
|
2017-08-29 19:51:08 +00:00
|
|
|
|
return format_time(datetime,
|
|
|
|
|
/* xgettext:no-c-format */ /* Date + time in 24h format (w/o seconds) */ _("%x, %H∶%M"),
|
|
|
|
|
/* xgettext:no-c-format */ /* Date + time in 12h format (w/o seconds)*/ _("%x, %l∶%M %p"));
|
2017-08-27 21:55:49 +00:00
|
|
|
|
} else if (timespan > 7 * TimeSpan.DAY) {
|
2017-08-29 19:51:08 +00:00
|
|
|
|
return format_time(datetime,
|
|
|
|
|
/* xgettext:no-c-format */ /* Month, day and time in 24h format (w/o seconds) */ _("%b %d, %H∶%M"),
|
|
|
|
|
/* xgettext:no-c-format */ /* Month, day and time in 12h format (w/o seconds) */ _("%b %d, %l∶%M %p"));
|
2017-08-29 22:03:37 +00:00
|
|
|
|
} else if (datetime.get_day_of_month() != now.get_day_of_month()) {
|
2017-08-29 19:51:08 +00:00
|
|
|
|
return format_time(datetime,
|
|
|
|
|
/* xgettext:no-c-format */ /* Day of week and time in 24h format (w/o seconds) */ _("%a, %H∶%M"),
|
|
|
|
|
/* xgettext:no-c-format */ /* Day of week and time in 12h format (w/o seconds) */_("%a, %l∶%M %p"));
|
2017-08-27 21:55:49 +00:00
|
|
|
|
} else if (timespan > 9 * TimeSpan.MINUTE) {
|
2017-08-29 19:51:08 +00:00
|
|
|
|
return format_time(datetime,
|
|
|
|
|
/* xgettext:no-c-format */ /* Time in 24h format (w/o seconds) */ _("%H∶%M"),
|
|
|
|
|
/* xgettext:no-c-format */ /* Time in 12h format (w/o seconds) */ _("%l∶%M %p"));
|
2017-08-27 21:55:49 +00:00
|
|
|
|
} else if (timespan > TimeSpan.MINUTE) {
|
|
|
|
|
ulong mins = (ulong) (timespan.abs() / TimeSpan.MINUTE);
|
|
|
|
|
/* xgettext:this is the beginning of a sentence. */
|
|
|
|
|
return n("%i min ago", "%i mins ago", mins).printf(mins);
|
|
|
|
|
} else {
|
|
|
|
|
return _("Just now");
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-05-28 15:31:31 +00:00
|
|
|
|
|
|
|
|
|
public override void dispose() {
|
|
|
|
|
if (time_update_timeout != 0) {
|
|
|
|
|
Source.remove(time_update_timeout);
|
|
|
|
|
time_update_timeout = 0;
|
|
|
|
|
}
|
2021-08-24 17:35:00 +00:00
|
|
|
|
if (updated_roster_handler_id != 0){
|
|
|
|
|
stream_interactor.get_module(RosterManager.IDENTITY).disconnect(updated_roster_handler_id);
|
|
|
|
|
updated_roster_handler_id = 0;
|
|
|
|
|
}
|
|
|
|
|
base.dispose();
|
2020-05-28 15:31:31 +00:00
|
|
|
|
}
|
2017-08-27 21:55:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|