#!/usr/bin/env gjs
/* PIKA - Photo and Image Kooker Application
 * a rebranding of The GNU Image Manipulation Program (created with heckimp)
 * A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
 *
 * Original copyright, applying to most contents (license remains unchanged): 
 * Copyright (C) 1995 Spencer Kimball and Peter Mattis
 *
 * hello-world.js
 * Copyright (C) Jehan
 *
 * 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 .
 */
const System = imports.system
imports.gi.versions.Pika = '3.0';
const Pika = imports.gi.Pika;
imports.gi.versions.PikaUi = '3.0';
const PikaUi = imports.gi.PikaUi;
imports.gi.versions.Gegl = '0.4';
const Gegl = imports.gi.Gegl;
imports.gi.versions.Gtk = '3.0';
const Gtk = imports.gi.Gtk;
imports.gi.versions.Gdk = '3.0';
const Gdk = imports.gi.Gdk;
const GLib = imports.gi.GLib;
const GObject = imports.gi.GObject;
const Gio = imports.gi.Gio;
/* gjs's ARGV is not C-style. We must add the program name as first
 * value.
 */
ARGV.unshift(System.programInvocationName);
let url = "https://gitlab.gnome.org/GNOME/pika/blob/master/extensions/goat-exercises/goat-exercise-gjs.js";
function _(message) { return GLib.dgettext(null, message); }
var Goat = GObject.registerClass({
    GTypeName: 'Goat',
}, class Goat extends Pika.PlugIn {
    vfunc_query_procedures() {
        return ["plug-in-goat-exercise-gjs"];
    }
    vfunc_create_procedure(name) {
        let procedure = Pika.ImageProcedure.new(this, name, Pika.PDBProcType.PLUGIN, this.run);
        procedure.set_image_types("*");
        procedure.set_sensitivity_mask(Pika.ProcedureSensitivityMask.DRAWABLE);
        procedure.set_menu_label(_("Exercise a JavaScript goat"));
        procedure.set_icon_name(PikaUi.ICON_GEGL);
        procedure.add_menu_path ('/Filters/Development/Goat exercises/');
        procedure.set_documentation(_("Exercise a goat in the JavaScript language (GJS)"),
                                    _("Takes a goat for a walk in Javascript with the GJS interpreter"),
                                    name);
        procedure.set_attribution("Jehan", "Jehan", "2019");
        return procedure;
    }
    run(procedure, run_mode, image, drawables, config, run_data) {
        /* TODO: localization. */
        if (drawables.length != 1) {
            let msg = `Procedure '${procedure.get_name()}' only works with one drawable.`;
            let error = GLib.Error.new_literal(Pika.PlugIn.error_quark(), 0, msg);
            return procedure.new_return_values(Pika.PDBStatusType.CALLING_ERROR, error)
        }
        let drawable = drawables[0];
        if (run_mode == Pika.RunMode.INTERACTIVE) {
            PikaUi.init("goat-exercise-gjs");
            /* TODO: help function and ID. */
            let dialog = new PikaUi.Dialog({
              title: _("Exercise a goat (JavaScript)"),
              role: "goat-exercise-JavaScript",
              use_header_bar: true,
            });
            dialog.add_button(_("_Cancel"), Gtk.ResponseType.CANCEL);
            dialog.add_button(_("_Source"), Gtk.ResponseType.APPLY);
            dialog.add_button(_("_OK"), Gtk.ResponseType.OK);
            let geometry = new Gdk.Geometry();
            geometry.min_aspect = 0.5;
            geometry.max_aspect = 1.0;
            dialog.set_geometry_hints (null, geometry, Gdk.WindowHints.ASPECT);
            let box = new Gtk.Box({
              orientation: Gtk.Orientation.VERTICAL,
              spacing: 2
            });
            dialog.get_content_area().add(box);
            box.show();
            let lang = "JavaScript (GJS)";
            /* XXX Can't we have nicer-looking multiline strings and
             * also in printf format like in C for sharing the same
             * string in localization?
             */
            let head_text = `This plug-in is an exercise in '${lang}' to demo plug-in creation.\n` +
                            `Check out the last version of the source code online by clicking the \"Source\" button.`;
            let label = new Gtk.Label({label:head_text});
            box.pack_start(label, false, false, 1);
            label.show();
            let contents = imports.byteArray.toString(GLib.file_get_contents(System.programInvocationName)[1]);
            if (contents) {
                let scrolled = new Gtk.ScrolledWindow();
                scrolled.set_vexpand (true);
                box.pack_start(scrolled, true, true, 1);
                scrolled.show();
                let view = new Gtk.TextView();
                view.set_wrap_mode(Gtk.WrapMode.WORD);
                view.set_editable(false);
                let buffer = view.get_buffer();
                buffer.set_text(contents, -1);
                scrolled.add(view);
                view.show();
            }
            while (true) {
                let response = dialog.run();
                if (response == Gtk.ResponseType.OK) {
                    dialog.destroy();
                    break;
                }
                else if (response == Gtk.ResponseType.APPLY) {
                    Gio.app_info_launch_default_for_uri(url, null);
                    continue;
                }
                else { /* CANCEL, CLOSE, DELETE_EVENT */
                    dialog.destroy();
                    return procedure.new_return_values(Pika.PDBStatusType.CANCEL, null)
                }
            }
        }
        let [ intersect, x, y, width, height ] = drawable.mask_intersect();
        if (intersect) {
            Gegl.init(null);
            let buffer = drawable.get_buffer();
            let shadow_buffer = drawable.get_shadow_buffer();
            let graph = new Gegl.Node();
            let input = graph.create_child("gegl:buffer-source");
            input.set_property("buffer", buffer);
            let invert = graph.create_child("gegl:invert");
            let output = graph.create_child("gegl:write-buffer");
            output.set_property("buffer", shadow_buffer);
            input.link(invert);
            invert.link(output);
            output.process();
            /* This is extremely important in bindings, since we don't
             * unref buffers. If we don't explicitly flush a buffer, we
             * may left hanging forever. This step is usually done
             * during an unref().
             */
            shadow_buffer.flush();
            drawable.merge_shadow(true);
            drawable.update(x, y, width, height);
            Pika.displays_flush();
        }
        return procedure.new_return_values(Pika.PDBStatusType.SUCCESS, null);
    }
});
Pika.main(Goat.$gtype, ARGV);