/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*-

   caja-emblem-utils.c: Utilities for handling emblems

   Copyright (C) 2002 Red Hat, Inc.

   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 2 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, write to the
   Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
   Boston, MA 02110-1301, USA.

   Author: Alexander Larsson <alexl@redhat.com>
*/

#include <config.h>

#include <sys/types.h>
#include <utime.h>
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
#include <sys/stat.h>

#include <glib.h>
#include <glib/gstdio.h>
#include <glib/gi18n.h>
#include <gtk/gtk.h>

#include <eel/eel-gdk-pixbuf-extensions.h>
#include <eel/eel-stock-dialogs.h>

#include "caja-file.h"
#include "caja-emblem-utils.h"

#include <src/glibcompat.h> /* for g_list_free_full */

#define EMBLEM_NAME_TRASH   "emblem-trash"
#define EMBLEM_NAME_SYMLINK "emblem-symbolic-link"
#define EMBLEM_NAME_NOREAD  "emblem-noread"
#define EMBLEM_NAME_NOWRITE "emblem-nowrite"
#define EMBLEM_NAME_NOTE    "emblem-note"
#define EMBLEM_NAME_SHARED  "emblem-shared"

GList *
caja_emblem_list_available (void)
{
    GtkIconTheme *icon_theme;
    GList *list;

    icon_theme = gtk_icon_theme_get_default ();
    list = gtk_icon_theme_list_icons (icon_theme, "Emblems");
    return list;
}

void
caja_emblem_refresh_list (void)
{
    GtkIconTheme *icon_theme;

    icon_theme = gtk_icon_theme_get_default ();
    gtk_icon_theme_rescan_if_needed (icon_theme);
}

char *
caja_emblem_get_icon_name_from_keyword (const char *keyword)
{
    return g_strconcat ("emblem-", keyword, NULL);
}


/* check for reserved keywords */
static gboolean
is_reserved_keyword (const char *keyword)
{
    GList *available;
    char *icon_name;
    gboolean result;

    g_assert (keyword != NULL);

    /* check intrinsic emblems */
    if (g_ascii_strcasecmp (keyword, CAJA_FILE_EMBLEM_NAME_TRASH) == 0)
    {
        return TRUE;
    }
    if (g_ascii_strcasecmp (keyword, CAJA_FILE_EMBLEM_NAME_CANT_READ) == 0)
    {
        return TRUE;
    }
    if (g_ascii_strcasecmp (keyword, CAJA_FILE_EMBLEM_NAME_CANT_WRITE) == 0)
    {
        return TRUE;
    }
    if (g_ascii_strcasecmp (keyword, CAJA_FILE_EMBLEM_NAME_SYMBOLIC_LINK) == 0)
    {
        return TRUE;
    }
    if (g_ascii_strcasecmp (keyword, CAJA_FILE_EMBLEM_NAME_NOTE) == 0)
    {
        return TRUE;
    }
    if (g_ascii_strcasecmp (keyword, CAJA_FILE_EMBLEM_NAME_SHARED) == 0)
    {
        return TRUE;
    }

    available = caja_emblem_list_available ();
    icon_name = caja_emblem_get_icon_name_from_keyword (keyword);
    /* see if the keyword already exists */
    result = g_list_find_custom (available,
                                 (char *) icon_name,
                                 (GCompareFunc) g_ascii_strcasecmp) != NULL;
    g_list_free_full (available, g_free);
    g_free (icon_name);
    return result;
}

gboolean
caja_emblem_should_show_in_list (const char *emblem)
{
    if (strcmp (emblem, EMBLEM_NAME_TRASH) == 0)
    {
        return FALSE;
    }
    if (strcmp (emblem, EMBLEM_NAME_SYMLINK) == 0)
    {
        return FALSE;
    }
    if (strcmp (emblem, EMBLEM_NAME_NOREAD) == 0)
    {
        return FALSE;
    }
    if (strcmp (emblem, EMBLEM_NAME_NOWRITE) == 0)
    {
        return FALSE;
    }
    if (strcmp (emblem, EMBLEM_NAME_NOTE) == 0)
    {
        return FALSE;
    }
    if (strcmp (emblem, EMBLEM_NAME_SHARED) == 0)
    {
        return FALSE;
    }

    return TRUE;
}

char *
caja_emblem_get_keyword_from_icon_name (const char *emblem)
{
    g_return_val_if_fail (emblem != NULL, NULL);

    if (g_str_has_prefix (emblem, "emblem-"))
    {
        return g_strdup (&emblem[7]);
    }
    else
    {
        return g_strdup (emblem);
    }
}

GdkPixbuf *
caja_emblem_load_pixbuf_for_emblem (GFile *emblem)
{
    GInputStream *stream;
    GdkPixbuf *pixbuf;
    GdkPixbuf *scaled;

    stream = (GInputStream *) g_file_read (emblem, NULL, NULL);
    if (!stream)
    {
        return NULL;
    }

    pixbuf = eel_gdk_pixbuf_load_from_stream (stream);
    g_return_val_if_fail (pixbuf != NULL, NULL);

    scaled = eel_gdk_pixbuf_scale_down_to_fit (pixbuf,
             CAJA_ICON_SIZE_STANDARD,
             CAJA_ICON_SIZE_STANDARD);

    g_object_unref (pixbuf);
    g_object_unref (stream);

    return scaled;
}

/* utility to make sure the passed-in keyword only contains alphanumeric characters */
static gboolean
emblem_keyword_valid (const char *keyword)
{
    const char *p;
    gunichar c;

    for (p = keyword; *p; p = g_utf8_next_char (p))
    {
        c = g_utf8_get_char (p);

        if (!g_unichar_isalnum (c) &&
                !g_unichar_isspace (c))
        {
            return FALSE;
        }
    }

    return TRUE;
}

gboolean
caja_emblem_verify_keyword (GtkWindow *parent_window,
                            const char *keyword,
                            const char *display_name)
{
    if (keyword == NULL || strlen (keyword) == 0)
    {
        eel_show_error_dialog (_("The emblem cannot be installed."),
                               _("Sorry, but you must specify a non-blank keyword for the new emblem."),
                               GTK_WINDOW (parent_window));
        return FALSE;
    }
    else if (!emblem_keyword_valid (keyword))
    {
        eel_show_error_dialog (_("The emblem cannot be installed."),
                               _("Sorry, but emblem keywords can only contain letters, spaces and numbers."),
                               GTK_WINDOW (parent_window));
        return FALSE;
    }
    else if (is_reserved_keyword (keyword))
    {
        char *error_string;

        /* this really should never happen, as a user has no idea
         * what a keyword is, and people should be passing a unique
         * keyword to us anyway
         */
        error_string = g_strdup_printf (_("Sorry, but there is already an emblem named \"%s\"."), display_name);
        eel_show_error_dialog (_("Please choose a different emblem name."), error_string,
                               GTK_WINDOW (parent_window));
        g_free (error_string);
        return FALSE;
    }

    return TRUE;
}

void
caja_emblem_install_custom_emblem (GdkPixbuf *pixbuf,
                                   const char *keyword,
                                   const char *display_name,
                                   GtkWindow *parent_window)
{
    char *basename, *path, *dir, *stat_dir;
    struct stat stat_buf;
    struct utimbuf ubuf;

    g_return_if_fail (pixbuf != NULL);

    if (!caja_emblem_verify_keyword (parent_window, keyword, display_name))
    {
        return;
    }

    dir = g_build_filename (g_get_home_dir (),
                            ".icons", "hicolor", "48x48", "emblems",
                            NULL);
    stat_dir = g_build_filename (g_get_home_dir (),
                                 ".icons", "hicolor",
                                 NULL);

    if (g_mkdir_with_parents (dir, 0755) != 0)
    {
        eel_show_error_dialog (_("The emblem cannot be installed."),
                               _("Sorry, unable to save custom emblem."),
                               GTK_WINDOW (parent_window));
        g_free (dir);
        g_free (stat_dir);
        return;
    }

    basename = g_strdup_printf ("emblem-%s.png", keyword);
    path = g_build_filename (dir, basename, NULL);
    g_free (basename);

    /* save the image */
    if (eel_gdk_pixbuf_save_to_file (pixbuf, path) != TRUE)
    {
        eel_show_error_dialog (_("The emblem cannot be installed."),
                               _("Sorry, unable to save custom emblem."),
                               GTK_WINDOW (parent_window));
        g_free (dir);
        g_free (stat_dir);
        g_free (path);
        return;
    }

    g_free (path);

    if (display_name != NULL)
    {
        char *contents;

        basename = g_strdup_printf ("emblem-%s.icon", keyword);
        path = g_build_filename (dir, basename, NULL);
        g_free (basename);

        contents = g_strdup_printf ("\n[Icon Data]\n\nDisplayName=%s\n",
                                    display_name);

        if (!g_file_set_contents (path, contents, strlen (contents), NULL))
        {
            eel_show_error_dialog (_("The emblem cannot be installed."),
                                   _("Sorry, unable to save custom emblem name."),
                                   GTK_WINDOW (parent_window));
            g_free (contents);
            g_free (path);
            g_free (stat_dir);
            g_free (dir);
            return;
        }

        g_free (contents);
        g_free (path);
    }

    /* Touch the toplevel dir */
    if (g_stat (stat_dir, &stat_buf) == 0)
    {
        ubuf.actime = stat_buf.st_atime;
        ubuf.modtime = time (NULL);
        utime (stat_dir, &ubuf);
    }

    g_free (dir);
    g_free (stat_dir);

    return;
}

gboolean
caja_emblem_can_remove_emblem (const char *keyword)
{
    char *path;
    gboolean ret = TRUE;

    path = g_strdup_printf ("%s/.icons/hicolor/48x48/emblems/emblem-%s.png",
                            g_get_home_dir (), keyword);

    if (g_access (path, F_OK|W_OK) != 0)
    {
        ret = FALSE;
    }

    g_free (path);

    return ret;
}

gboolean
caja_emblem_can_rename_emblem (const char *keyword)
{
    char *path;
    gboolean ret = TRUE;

    path = g_strdup_printf ("%s/.icons/hicolor/48x48/emblems/emblem-%s.png",
                            g_get_home_dir (), keyword);

    if (access (path, F_OK|R_OK) != 0)
    {
        ret = FALSE;
    }

    g_free (path);

    return ret;
}

/* of course, this only works for custom installed emblems */
gboolean
caja_emblem_remove_emblem (const char *keyword)
{
    char *path, *dir, *stat_dir;
    struct stat stat_buf;
    struct utimbuf ubuf;


    dir = g_strdup_printf ("%s/.icons/hicolor/48x48/emblems",
                           g_get_home_dir ());
    stat_dir = g_strdup_printf ("%s/.icons/hicolor",
                                g_get_home_dir ());

    path = g_strdup_printf ("%s/emblem-%s.png", dir, keyword);

    /* delete the image */
    if (unlink (path) != 0)
    {
        /* couldn't delete it */
        g_free (dir);
        g_free (stat_dir);
        g_free (path);
        return FALSE;
    }

    g_free (path);

    path = g_strdup_printf ("%s/emblem-%s.icon", dir, keyword);

    if (unlink (path) != 0)
    {
        g_free (dir);
        g_free (stat_dir);
        g_free (path);
        return FALSE;
    }

    /* Touch the toplevel dir */
    if (stat (stat_dir, &stat_buf) == 0)
    {
        ubuf.actime = stat_buf.st_atime;
        ubuf.modtime = time (NULL);
        utime (stat_dir, &ubuf);
    }

    g_free (dir);
    g_free (stat_dir);

    return TRUE;
}

/* this only works for custom emblems as well */
gboolean
caja_emblem_rename_emblem (const char *keyword, const char *name)
{
    char *path, *dir, *stat_dir, *icon_name;
    struct stat stat_buf;
    struct utimbuf ubuf;
    FILE *file;

    dir = g_strdup_printf ("%s/.icons/hicolor/48x48/emblems",
                           g_get_home_dir ());
    stat_dir = g_strdup_printf ("%s/.icons/hicolor",
                                g_get_home_dir ());

    path = g_strdup_printf ("%s/emblem-%s.icon", dir, keyword);

    file = fopen (path, "w+");
    g_free (path);

    if (file == NULL)
    {
        g_free (dir);
        g_free (stat_dir);
        return FALSE;
    }


    /* write the new icon description */
    fprintf (file, "\n[Icon Data]\n\nDisplayName=%s\n", name);
    fflush (file);
    fclose (file);

    icon_name = caja_emblem_get_icon_name_from_keyword (keyword);
    caja_icon_info_clear_caches (); /* A bit overkill, but this happens rarely */

    g_free (icon_name);

    /* Touch the toplevel dir */
    if (stat (stat_dir, &stat_buf) == 0)
    {
        ubuf.actime = stat_buf.st_atime;
        ubuf.modtime = time (NULL);
        utime (stat_dir, &ubuf);
    }

    g_free (dir);
    g_free (stat_dir);

    return TRUE;
}

char *
caja_emblem_create_unique_keyword (const char *base)
{
    char *keyword;
    time_t t;
    int i;

    time (&t);
    i=0;

    keyword = NULL;
    do
    {
        g_free (keyword);
        keyword = g_strdup_printf ("user%s%d%d", base, (int)t, i++);
    }
    while (is_reserved_keyword (keyword));

    return keyword;
}