aboutsummaryrefslogtreecommitdiff
path: root/main/src/ui/conversation_content_view
diff options
context:
space:
mode:
authorfiaxh <git@lightrise.org>2020-04-22 15:44:12 +0200
committerfiaxh <git@lightrise.org>2020-04-22 15:44:12 +0200
commit51a23728694a3f1312cc9396fc093ca178457c3c (patch)
tree321771ae3d807d19387a8656805a648d75347994 /main/src/ui/conversation_content_view
parent7c4260eed718961874fc0ea665263ea2ce59338b (diff)
downloaddino-51a23728694a3f1312cc9396fc093ca178457c3c.tar.gz
dino-51a23728694a3f1312cc9396fc093ca178457c3c.zip
Add file upload preview
fixes #756
Diffstat (limited to 'main/src/ui/conversation_content_view')
-rw-r--r--main/src/ui/conversation_content_view/conversation_item_skeleton.vala4
-rw-r--r--main/src/ui/conversation_content_view/file_default_widget.vala4
-rw-r--r--main/src/ui/conversation_content_view/file_image_widget.vala86
-rw-r--r--main/src/ui/conversation_content_view/file_widget.vala129
4 files changed, 131 insertions, 92 deletions
diff --git a/main/src/ui/conversation_content_view/conversation_item_skeleton.vala b/main/src/ui/conversation_content_view/conversation_item_skeleton.vala
index ef880063..b589fe9e 100644
--- a/main/src/ui/conversation_content_view/conversation_item_skeleton.vala
+++ b/main/src/ui/conversation_content_view/conversation_item_skeleton.vala
@@ -49,10 +49,10 @@ public class ConversationItemSkeleton : EventBox {
if (initial_item) {
this.add(image_content_box);
} else {
- Revealer revealer = new Revealer() { transition_duration=200, transition_type = RevealerTransitionType.SLIDE_UP, visible = true };
+ Revealer revealer = new Revealer() { transition_duration=200, transition_type=RevealerTransitionType.SLIDE_UP, reveal_child=false, visible=true };
revealer.add_with_properties(image_content_box);
- revealer.reveal_child = true;
this.add(revealer);
+ revealer.reveal_child = true;
}
diff --git a/main/src/ui/conversation_content_view/file_default_widget.vala b/main/src/ui/conversation_content_view/file_default_widget.vala
index 045223fe..c2af92e3 100644
--- a/main/src/ui/conversation_content_view/file_default_widget.vala
+++ b/main/src/ui/conversation_content_view/file_default_widget.vala
@@ -29,7 +29,7 @@ public class FileDefaultWidget : EventBox {
mime_label.leave_notify_event.connect((event) => { pointer_inside = true; return false; });
}
- public void update_file_info(string? mime_type, FileTransfer.State state, int size) {
+ public void update_file_info(string? mime_type, FileTransfer.State state, long size) {
this.state = state;
spinner.active = false; // A hidden spinning spinner still uses CPU. Deactivate asap
@@ -111,7 +111,7 @@ public class FileDefaultWidget : EventBox {
}
}
- private static string get_size_string(int size) {
+ private static string get_size_string(long size) {
if (size < 1024) {
return @"$(size) B";
} else if (size < 1000 * 1000) {
diff --git a/main/src/ui/conversation_content_view/file_image_widget.vala b/main/src/ui/conversation_content_view/file_image_widget.vala
new file mode 100644
index 00000000..8005f996
--- /dev/null
+++ b/main/src/ui/conversation_content_view/file_image_widget.vala
@@ -0,0 +1,86 @@
+using Gee;
+using Gdk;
+using Gtk;
+
+using Dino.Entities;
+
+namespace Dino.Ui {
+
+public class FileImageWidget : EventBox {
+
+ private ScalingImage image;
+ FileDefaultWidget file_default_widget;
+ FileDefaultWidgetController file_default_widget_controller;
+ private bool pointer_inside = false;
+
+ public FileImageWidget() {
+ this.halign = Align.START;
+ this.events = EventMask.POINTER_MOTION_MASK;
+
+ Util.force_css(this, "* { border: 1px solid alpha(@theme_fg_color, 0.1); border-radius: 3px; }");
+ }
+
+ public async void load_from_file(File file, string file_name, int MAX_WIDTH=600, int MAX_HEIGHT=300) throws GLib.Error {
+ // Load and prepare image in tread
+ Thread<ScalingImage?> thread = new Thread<ScalingImage?> (null, () => {
+ ScalingImage image = new ScalingImage() { halign=Align.START, visible = true, max_width = MAX_WIDTH, max_height = MAX_HEIGHT };
+
+ Gdk.Pixbuf pixbuf;
+ try {
+ pixbuf = new Gdk.Pixbuf.from_file(file.get_path());
+ } catch (Error error) {
+ warning("Can't load picture %s - %s", file.get_path(), error.message);
+ Idle.add(load_from_file.callback);
+ return null;
+ }
+
+ pixbuf = pixbuf.apply_embedded_orientation();
+
+ image.load(pixbuf);
+
+ Idle.add(load_from_file.callback);
+ return image;
+ });
+ yield;
+ image = thread.join();
+ if (image == null) throw new Error(-1, 0, "Error loading image");
+
+ FileInfo file_info = file.query_info("*", FileQueryInfoFlags.NONE);
+ string? mime_type = file_info.get_content_type();
+
+ file_default_widget = new FileDefaultWidget() { valign=Align.END, vexpand=false };
+ file_default_widget.stack_event_box.visible = false;
+ file_default_widget_controller = new FileDefaultWidgetController(file_default_widget);
+ file_default_widget_controller.set_file(file, file_name, mime_type);
+
+ Util.force_css(file_default_widget, "* { color: #eee; }");
+ Util.force_css(file_default_widget, "* { background-color: rgba(0, 0, 0, 0.5); }");
+
+ Overlay overlay = new Overlay() { visible=true };
+ overlay.add(image);
+ overlay.add_overlay(file_default_widget);
+
+ this.enter_notify_event.connect(() => {
+ file_default_widget.visible = true;
+ return false;
+ });
+ this.leave_notify_event.connect(() => {
+ pointer_inside = false;
+ Timeout.add(20, () => {
+ if (!pointer_inside) {
+ file_default_widget.visible = false;
+ }
+ return false;
+ });
+
+ return false;
+ });
+
+ file_default_widget.enter_notify_event.connect(() => { pointer_inside = true; return false; });
+ file_default_widget.leave_notify_event.connect(() => { pointer_inside = true; return false; });
+
+ this.add(overlay);
+ }
+}
+
+}
diff --git a/main/src/ui/conversation_content_view/file_widget.vala b/main/src/ui/conversation_content_view/file_widget.vala
index d92eedb5..3819e5bc 100644
--- a/main/src/ui/conversation_content_view/file_widget.vala
+++ b/main/src/ui/conversation_content_view/file_widget.vala
@@ -5,9 +5,9 @@ using Pango;
using Dino.Entities;
-namespace Dino.Ui.ConversationSummary {
+namespace Dino.Ui {
-public class FileMetaItem : ContentMetaItem {
+public class FileMetaItem : ConversationSummary.ContentMetaItem {
private StreamInteractor stream_interactor;
@@ -25,7 +25,7 @@ public class FileMetaItem : ContentMetaItem {
public override Gee.List<Plugins.MessageAction>? get_item_actions(Plugins.WidgetType type) { return null; }
}
-public class FileWidget : Box {
+public class FileWidget : SizeRequestBox {
enum State {
IMAGE,
@@ -42,6 +42,11 @@ public class FileWidget : Box {
private FileDefaultWidgetController default_widget_controller;
private Widget? content = null;
+ construct {
+ margin_top = 4;
+ size_request_mode = SizeRequestMode.HEIGHT_FOR_WIDTH;
+ }
+
public FileWidget(StreamInteractor stream_interactor, FileTransfer file_transfer) {
this.stream_interactor = stream_interactor;
this.file_transfer = file_transfer;
@@ -60,100 +65,34 @@ public class FileWidget : Box {
private async void update_widget() {
if (show_image() && state != State.IMAGE) {
var content_bak = content;
- Widget? image_widget = yield get_image_widget(file_transfer.get_file(), file_transfer.file_name);
- // If the widget changed in the meanwhile, stop
- if (content != content_bak) return;
+ FileImageWidget file_image_widget = null;
+ try {
+ file_image_widget = new FileImageWidget() { visible=true };
+ yield file_image_widget.load_from_file(file_transfer.get_file(), file_transfer.file_name);
+
+ // If the widget changed in the meanwhile, stop
+ if (content != content_bak) return;
- if (image_widget != null) {
if (content != null) this.remove(content);
- content = image_widget;
+ content = file_image_widget;
state = State.IMAGE;
this.add(content);
return;
- }
+ } catch (Error e) { }
}
if (state != State.DEFAULT) {
if (content != null) this.remove(content);
FileDefaultWidget default_file_widget = new FileDefaultWidget() { visible=true };
- default_widget_controller = new FileDefaultWidgetController(default_file_widget, file_transfer, stream_interactor);
+ default_widget_controller = new FileDefaultWidgetController(default_file_widget);
+ default_widget_controller.set_file_transfer(file_transfer, stream_interactor);
content = default_file_widget;
this.state = State.DEFAULT;
this.add(content);
}
}
- public static async Widget? get_image_widget(File file, string file_name, int MAX_WIDTH=600, int MAX_HEIGHT=300) {
- // Load and prepare image in tread
- Thread<Image?> thread = new Thread<Image?> (null, () => {
- ScalingImage image = new ScalingImage() { halign=Align.START, visible = true, max_width = MAX_WIDTH, max_height = MAX_HEIGHT };
-
- Gdk.Pixbuf pixbuf;
- try {
- pixbuf = new Gdk.Pixbuf.from_file(file.get_path());
- } catch (Error error) {
- warning("Can't load picture %s - %s", file.get_path(), error.message);
- Idle.add(get_image_widget.callback);
- return null;
- }
-
- pixbuf = pixbuf.apply_embedded_orientation();
-
- image.load(pixbuf);
-
- Idle.add(get_image_widget.callback);
- return image;
- });
- yield;
- Image image = thread.join();
- if (image == null) return null;
-
- Util.force_css(image, "* { box-shadow: 0px 0px 2px 0px rgba(0,0,0,0.1); margin: 2px; border-radius: 3px; }");
-
- Builder builder = new Builder.from_resource("/im/dino/Dino/conversation_content_view/image_toolbar.ui");
- Widget toolbar = builder.get_object("main") as Widget;
- Util.force_background(toolbar, "rgba(0, 0, 0, 0.5)");
- Util.force_css(toolbar, "* { padding: 3px; border-radius: 3px; }");
-
- Label url_label = builder.get_object("url_label") as Label;
- Util.force_color(url_label, "#eee");
-
- if (file_name != null && file_name != "") {
- string caption = file_name;
- url_label.label = caption;
- } else {
- url_label.visible = false;
- }
-
- Image open_image = builder.get_object("open_image") as Image;
- Util.force_css(open_image, "*:not(:hover) { color: #eee; }");
- Button open_button = builder.get_object("open_button") as Button;
- Util.force_css(open_button, "*:hover { background-color: rgba(255,255,255,0.3); border-color: transparent; }");
- open_button.clicked.connect(() => {
- try{
- AppInfo.launch_default_for_uri(file.get_uri(), null);
- } catch (Error err) {
- info("Could not to open file://%s: %s", file.get_path(), err.message);
- }
- });
-
- Revealer toolbar_revealer = new Revealer() { transition_type=RevealerTransitionType.CROSSFADE, transition_duration=400, visible=true };
- toolbar_revealer.add(toolbar);
-
- Grid grid = new Grid() { visible=true };
- grid.attach(toolbar_revealer, 0, 0, 1, 1);
- grid.attach(image, 0, 0, 1, 1);
-
- EventBox event_box = new EventBox() { margin_top=5, halign=Align.START, visible=true };
- event_box.events = EventMask.POINTER_MOTION_MASK;
- event_box.add(grid);
- event_box.enter_notify_event.connect(() => { toolbar_revealer.reveal_child = true; return false; });
- event_box.leave_notify_event.connect(() => { toolbar_revealer.reveal_child = false; return false; });
-
- return event_box;
- }
-
private bool show_image() {
if (file_transfer.mime_type == null) return false;
if (file_transfer.state != FileTransfer.State.COMPLETE &&
@@ -175,18 +114,22 @@ public class FileWidget : Box {
public class FileDefaultWidgetController : Object {
private FileDefaultWidget widget;
- private FileTransfer file_transfer;
- private StreamInteractor stream_interactor;
+ private FileTransfer? file_transfer;
+ private StreamInteractor? stream_interactor;
+ private string file_uri;
+ private FileTransfer.State state;
- public FileDefaultWidgetController(FileDefaultWidget widget, FileTransfer file_transfer, StreamInteractor stream_interactor) {
+ public FileDefaultWidgetController(FileDefaultWidget widget) {
this.widget = widget;
+ widget.button_release_event.connect(on_clicked);
+ }
+
+ public void set_file_transfer(FileTransfer file_transfer, StreamInteractor stream_interactor) {
this.file_transfer = file_transfer;
this.stream_interactor = stream_interactor;
widget.name_label.label = file_transfer.file_name;
- widget.button_release_event.connect(on_clicked);
-
file_transfer.notify["path"].connect(update_file_info);
file_transfer.notify["state"].connect(update_file_info);
file_transfer.notify["mime-type"].connect(update_file_info);
@@ -194,22 +137,32 @@ public class FileDefaultWidgetController : Object {
update_file_info();
}
+ public void set_file(File file, string file_name, string? mime_type) {
+ file_uri = file.get_uri();
+ state = FileTransfer.State.COMPLETE;
+ widget.name_label.label = file_name;
+ widget.update_file_info(mime_type, state, -1);
+ }
+
private void update_file_info() {
+ file_uri = file_transfer.get_file().get_uri();
+ state = file_transfer.state;
widget.update_file_info(file_transfer.mime_type, file_transfer.state, file_transfer.size);
}
private bool on_clicked(EventButton event_button) {
- switch (file_transfer.state) {
+ switch (state) {
case FileTransfer.State.COMPLETE:
if (event_button.button == 1) {
try{
- AppInfo.launch_default_for_uri(file_transfer.get_file().get_uri(), null);
+ AppInfo.launch_default_for_uri(file_uri, null);
} catch (Error err) {
- print("Tried to open " + file_transfer.get_file().get_path());
+ print("Tried to open " + file_uri);
}
}
break;
case FileTransfer.State.NOT_STARTED:
+ assert(stream_interactor != null && file_transfer != null);
stream_interactor.get_module(FileManager.IDENTITY).download_file.begin(file_transfer);
break;
}