diff --git a/data/write-as-gtk.appdata.xml b/data/write-as-gtk.appdata.xml index 6f311fe..a6f2445 100644 --- a/data/write-as-gtk.appdata.xml +++ b/data/write-as-gtk.appdata.xml @@ -1,34 +1,46 @@ write-as-gtk GPL-3.0+ GPL-3.0+ write.as Spread your ideas

A Distraction Free Writing Tool:

We eliminate notifications, streams, likes, and commentary so you can focus on your words. Enjoy a clear mind and a beautifully simple space to write your thoughts.

Publish Articles:

Write your apps in the privacy and convenience of your desktop of choice with this native app, all without signing up. Write something, press publish, and share the link to your new post — that's all there is to it!

Privacy By Default:

Focus on your ideas — not who's listening. We go farther than the rest to protect your privacy, and even make it easy to publish under multiple pen names if you want.

A Bunch Tell LLC https://write.as/ https://write.as/contact https://code.as/writeas/writeas-gtk/issues hello@write.as write-as-gtk - - + + + The Write.as editor. + https://write.as/img/screens/gtk/serif.png + + + The Write.as editor in dark mode. + https://write.as/img/screens/gtk/serif-dark.png + + + https://write.as/img/screens/gtk/sans.png + + + https://write.as/img/screens/gtk/monospace.png + + diff --git a/debian/control b/debian/control index b825cc4..ab2d071 100644 --- a/debian/control +++ b/debian/control @@ -1,17 +1,17 @@ Source: write-as-gtk Section: x11 Priority: extra Maintainer: Adrian Cochrane Build-Depends: meson, debhelper (>= 9), libgtk-3-dev, valac (>= 0.26), libjson-glib-dev (>= 1.1.2), libsoup2.4-dev (>= 2.52.2) Standards-Version: 3.9.3 Package: write-as-gtk Architecture: any Depends: ${misc:Depends}, ${shlibs:Depends} -Recommend: fonts-open-sans, fonts-hack +Recommends: fonts-open-sans, fonts-hack Description: A distraction free and private writing tool, with builtin publishing. diff --git a/src/window.vala b/src/window.vala index 6c4463e..f35a551 100644 --- a/src/window.vala +++ b/src/window.vala @@ -1,250 +1,250 @@ public class WriteAs.MainWindow : Gtk.ApplicationWindow { private Gtk.TextView canvas; private bool dark_mode = false; private string font = "Lora, 'Palatino Linotype'," + "'Book Antiqua', 'New York', 'DejaVu serif', serif"; private string fontstyle = "serif"; construct { construct_toolbar(); build_keyboard_shortcuts(); var scrolled = new Gtk.ScrolledWindow(null, null); canvas = new Gtk.TextView(); canvas.wrap_mode = Gtk.WrapMode.WORD_CHAR; scrolled.add(canvas); add(scrolled); var text_changed = false; canvas.event_after.connect((evt) => { // TODO This word count algorithm may be quite naive // and could do improvement. var word_count = canvas.buffer.text.split(" ").length; title = ngettext("%i word","%i words",word_count).printf(word_count); text_changed = true; }); Timeout.add_full(Priority.DEFAULT_IDLE, 100/*ms*/, () => { if (!text_changed) return Source.CONTINUE; try { draft_file().replace_contents(canvas.buffer.text.data, null, false, FileCreateFlags.PRIVATE | FileCreateFlags.REPLACE_DESTINATION, null); text_changed = false; } catch (Error err) {/* We'll try again anyways. */} return Source.CONTINUE; }); adjust_text_style(); } public MainWindow(Gtk.Application app) { set_application(app); try { open_file(draft_file()); } catch (Error err) {/* It's fine... */} set_default_size(800, 600); } private static File draft_file() { var home = File.new_for_path(Environment.get_home_dir()); return home.get_child(".writeas-draft.txt"); } private void construct_toolbar() { var header = new Gtk.HeaderBar(); header.show_close_button = true; set_titlebar(header); var publish_button = new Gtk.Button.from_icon_name("document-send", Gtk.IconSize.SMALL_TOOLBAR); publish_button.clicked.connect(() => { title = _("Publishing post…"); canvas.sensitive = false; publish.begin((obj, res) => { - canvas.buffer.text = publish.end(res); + canvas.buffer.text += "\n\n" + publish.end(res); canvas.sensitive = true; }); }); header.pack_end(publish_button); var darkmode_button = new Gtk.ToggleButton(); darkmode_button.tooltip_text = _("Toggle dark theme"); // NOTE the fallback icon is a bit of a meaning stretch, but it works. var icon_theme = Gtk.IconTheme.get_default(); darkmode_button.image = new Gtk.Image.from_icon_name( icon_theme.has_icon("writeas-bright-dark") ? "writeas-bright-dark" : "weather-clear-night", Gtk.IconSize.SMALL_TOOLBAR); darkmode_button.draw_indicator = false; var settings = Gtk.Settings.get_default(); darkmode_button.toggled.connect(() => { settings.gtk_application_prefer_dark_theme = darkmode_button.active; dark_mode = darkmode_button.active; adjust_text_style(); }); header.pack_end(darkmode_button); var fonts = new Gtk.MenuButton(); fonts.tooltip_text = _("Change document font"); fonts.image = new Gtk.Image.from_icon_name("font-x-generic", Gtk.IconSize.SMALL_TOOLBAR); fonts.popup = new Gtk.Menu(); header.pack_start(fonts); build_fontoption(fonts.popup, _("Serif"), "serif", font); build_fontoption(fonts.popup, _("Sans-serif"), "sans", "'Open Sans', 'Segoe UI', Tahoma, Arial, sans-serif"); - build_fontoption(fonts.popup, _("Monospace"), "mono", "Hack, consolas," + + build_fontoption(fonts.popup, _("Monospace"), "wrap", "Hack, consolas," + "Menlo-Regular, Menlo, Monaco, 'ubuntu mono', monospace"); fonts.popup.show_all(); } private void build_fontoption(Gtk.Menu menu, string label, string fontstyle, string families) { var option = new Gtk.MenuItem.with_label(label); option.activate.connect(() => { this.font = families; this.fontstyle = fontstyle; adjust_text_style(); }); var styles = option.get_style_context(); var provider = new Gtk.CssProvider(); try { provider.load_from_data("* {font: %s;}".printf(families)); styles.add_provider(provider, Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION); } catch (Error e) { warning(e.message); } menu.add(option); } private Gtk.CssProvider cur_styles = null; private void adjust_text_style() { try { var styles = canvas.get_style_context(); if (cur_styles != null) styles.remove_provider(cur_styles); var css = "* {font: %s; padding: 20px;}".printf(font); if (dark_mode) { // Try to detect whether the system provided a better dark mode. var text_color = styles.get_color(Gtk.StateFlags.ACTIVE); double h, s, v; Gtk.rgb_to_hsv(text_color.red, text_color.green, text_color.blue, out h, out s, out v); if (v < 0.5) css += "* {background: black; color: white;}"; } cur_styles = new Gtk.CssProvider(); cur_styles.load_from_data(css); styles.add_provider(cur_styles, Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION); } catch (Error e) { warning(e.message); } } private async string publish() { var session = new Soup.Session(); // Send the request var req = new Soup.Message("POST", "https://write.as/api/posts"); // TODO specify font. var req_body = "{\"body\": \"%s\", \"font\": \"%s\"}".printf( canvas.buffer.text, fontstyle); req.set_request("application/json", Soup.MemoryUse.COPY, req_body.data); try { var resp = yield session.send_async(req); // Handle the response if (req.status_code != 201) return _("Error code: HTTP %u").printf(req.status_code); var json = new Json.Parser(); json.load_from_stream(resp); var data = json.get_root().get_object().get_object_member("data"); var url = "https://write.as/" + data.get_string_member("id"); Gtk.Clipboard.get_default(get_display()).set_text(url, -1); // Open it in the browser var browser = AppInfo.get_default_for_uri_scheme("https"); var urls = new List(); urls.append(url); browser.launch_uris(urls, null); return _("The link to your published article has been copied into your clipboard for you."); } catch (Error err) { return _("Failed to upload post! Are you connected to the Internet?") + "\n\n" + err.message; } } /* --- */ private void build_keyboard_shortcuts() { /* These operations are not exposed to the UI as buttons, as most people are very familiar with them and they are not the focus of this app. */ var accels = new Gtk.AccelGroup(); accels.connect(Gdk.Key.S, Gdk.ModifierType.CONTROL_MASK, Gtk.AccelFlags.VISIBLE | Gtk.AccelFlags.LOCKED, (g,a,k,m) => save_as()); accels.connect(Gdk.Key.S, Gdk.ModifierType.CONTROL_MASK | Gdk.ModifierType.SHIFT_MASK, Gtk.AccelFlags.VISIBLE | Gtk.AccelFlags.LOCKED, (g,a,k,m) => save_as()); accels.connect(Gdk.Key.O, Gdk.ModifierType.CONTROL_MASK, Gtk.AccelFlags.VISIBLE | Gtk.AccelFlags.LOCKED, (g, a, k, m) => { try { open_file(prompt_file(Gtk.FileChooserAction.OPEN, _("_Open"))); } catch (Error e) { // It's fine... } return true; }); add_accel_group(accels); } private bool save_as() { try { var file = prompt_file(Gtk.FileChooserAction.SAVE, _("_Save as")); file.replace_contents(canvas.buffer.text.data, null, false, FileCreateFlags.PRIVATE | FileCreateFlags.REPLACE_DESTINATION, null); } catch (Error e) { // It's fine... } return true; } private File prompt_file(Gtk.FileChooserAction mode, string action) throws UserCancellable { var file_chooser = new Gtk.FileChooserDialog(action, this, mode, _("_Cancel"), Gtk.ResponseType.CANCEL, action, Gtk.ResponseType.ACCEPT); file_chooser.select_multiple = false; var filter = new Gtk.FileFilter(); filter.add_mime_type("text/plain"); file_chooser.set_filter(filter); var resp = file_chooser.run(); file_chooser.close(); if (resp == Gtk.ResponseType.ACCEPT) return file_chooser.get_file(); else throw new UserCancellable.USER_CANCELLED("FileChooserDialog"); } public void open_file(File file) throws Error { uint8[] text; file.load_contents(null, out text, null); canvas.buffer.text = (string) text; } } errordomain WriteAs.UserCancellable {USER_CANCELLED}