medit/moo/mooedit/mooedit-fileops.c

1290 lines
34 KiB
C
Raw Normal View History

2006-05-21 18:11:05 -05:00
/*
2008-09-07 00:15:07 -05:00
* mooedit-fileops.c
2005-06-22 18:20:32 +00:00
*
* Copyright (C) 2004-2008 by Yevgen Muntyan <muntyan@tamu.edu>
2005-06-22 18:20:32 +00:00
*
* This file is part of medit. medit is free software; you can
* redistribute it and/or modify it under the terms of the
* GNU Lesser General Public License as published by the
* Free Software Foundation; either version 2.1 of the License,
* or (at your option) any later version.
2005-06-22 18:20:32 +00:00
*
* You should have received a copy of the GNU Lesser General Public
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
2005-06-22 18:20:32 +00:00
*/
2006-06-13 00:58:44 -05:00
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
2005-06-22 18:20:32 +00:00
#define MOOEDIT_COMPILATION
#include "mooedit/mooedit-private.h"
#include "mooedit/mooeditor-private.h"
2008-09-07 00:15:07 -05:00
#include "mooedit/mooedit-fileops.h"
2005-06-22 18:20:32 +00:00
#include "mooedit/mooeditdialogs.h"
#include "mooedit/mootextbuffer.h"
#include "mooedit/mooeditprefs.h"
2008-08-31 01:40:29 -05:00
#include "mooutils/moofileicon.h"
#include "mooutils/moofilewatch.h"
#include "mooutils/mooencodings.h"
#include "mooutils/mooi18n.h"
2008-09-07 00:15:07 -05:00
#include "mooutils/mootype-macros.h"
2005-06-22 18:20:32 +00:00
#include <string.h>
2005-11-10 07:15:16 +00:00
#include <sys/types.h>
#include <sys/stat.h>
2006-12-22 02:16:50 -06:00
#include <fcntl.h>
2006-06-13 00:58:44 -05:00
#ifdef HAVE_UNISTD_H
2005-11-10 07:15:16 +00:00
#include <unistd.h>
2006-06-13 00:58:44 -05:00
#endif
2005-11-10 07:15:16 +00:00
#include <stdio.h>
#include <errno.h>
#if GLIB_CHECK_VERSION(2,6,0)
# include <glib/gstdio.h>
#endif
2005-06-22 18:20:32 +00:00
2006-06-18 05:43:25 -05:00
#undef LOAD_BINARY
#define ENCODING_LOCALE "LOCALE"
2005-06-22 18:20:32 +00:00
#ifndef O_BINARY
#define O_BINARY 0
#endif
2008-09-07 00:15:07 -05:00
MOO_DEFINE_QUARK (MooEditFileErrorQuark, _moo_edit_file_error_quark)
static GSList *UNTITLED = NULL;
static GHashTable *UNTITLED_NO = NULL;
2005-09-02 23:27:25 +00:00
static void block_buffer_signals (MooEdit *edit);
static void unblock_buffer_signals (MooEdit *edit);
static gboolean focus_in_cb (MooEdit *edit);
static void check_file_status (MooEdit *edit,
gboolean in_focus_only);
static void file_modified_on_disk (MooEdit *edit);
static void file_deleted (MooEdit *edit);
static void add_status (MooEdit *edit,
MooEditStatus s);
2005-09-02 23:27:25 +00:00
static gboolean moo_edit_load_local (MooEdit *edit,
2005-11-02 04:41:09 +00:00
const char *file,
const char *encoding,
GError **error);
static gboolean moo_edit_reload_local (MooEdit *edit,
2007-11-25 12:22:34 -06:00
const char *encoding,
2005-11-02 04:41:09 +00:00
GError **error);
static gboolean moo_edit_save_local (MooEdit *edit,
const char *filename,
2005-11-02 04:41:09 +00:00
const char *encoding,
2005-11-10 07:15:16 +00:00
MooEditSaveFlags flags,
2005-11-02 04:41:09 +00:00
GError **error);
static gboolean moo_edit_save_copy_local (MooEdit *edit,
const char *filename,
2005-11-02 04:41:09 +00:00
const char *encoding,
GError **error);
static char *_moo_edit_filename_to_utf8 (const char *filename);
static void _moo_edit_start_file_watch (MooEdit *edit);
2005-09-02 23:27:25 +00:00
2005-06-22 18:20:32 +00:00
static const char *
normalize_encoding (const char *encoding,
gboolean for_save)
2005-09-02 23:27:25 +00:00
{
2007-07-12 00:53:34 -05:00
if (!encoding || !encoding[0] || !strcmp (encoding, MOO_ENCODING_AUTO))
encoding = for_save ? MOO_ENCODING_UTF8 : NULL;
return encoding;
2005-09-02 23:27:25 +00:00
}
2005-11-11 20:09:30 +00:00
gboolean
_moo_edit_load_file (MooEdit *edit,
const char *filename,
const char *encoding,
GError **error)
2005-09-02 23:27:25 +00:00
{
char *filename_copy, *encoding_copy;
gboolean result;
GError *error_here = NULL;
2005-09-02 23:27:25 +00:00
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
g_return_val_if_fail (filename != NULL, FALSE);
filename_copy = g_strdup (filename);
encoding_copy = g_strdup (normalize_encoding (encoding, FALSE));
2005-09-02 23:27:25 +00:00
result = moo_edit_load_local (edit, filename_copy, encoding_copy, &error_here);
if (error_here)
g_propagate_error (error, error_here);
g_free (filename_copy);
g_free (encoding_copy);
return result;
2005-09-02 23:27:25 +00:00
}
2005-11-11 20:09:30 +00:00
gboolean
2007-11-25 12:22:34 -06:00
_moo_edit_reload_file (MooEdit *edit,
const char *encoding,
GError **error)
2005-09-02 23:27:25 +00:00
{
GError *error_here = NULL;
gboolean result;
2005-09-02 23:27:25 +00:00
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
2007-11-25 12:22:34 -06:00
result = moo_edit_reload_local (edit, encoding, &error_here);
if (error_here)
g_propagate_error (error, error_here);
return result;
2005-09-02 23:27:25 +00:00
}
2005-11-11 20:09:30 +00:00
gboolean
_moo_edit_save_file (MooEdit *edit,
const char *filename,
const char *encoding,
MooEditSaveFlags flags,
GError **error)
2005-09-02 23:27:25 +00:00
{
GError *error_here = NULL;
char *filename_copy, *encoding_copy;
gboolean result;
2005-09-02 23:27:25 +00:00
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
g_return_val_if_fail (filename != NULL, FALSE);
filename_copy = g_strdup (filename);
encoding_copy = g_strdup (normalize_encoding (encoding, TRUE));
2005-09-02 23:27:25 +00:00
result = moo_edit_save_local (edit, filename_copy, encoding_copy, flags, &error_here);
if (error_here)
g_propagate_error (error, error_here);
g_free (filename_copy);
g_free (encoding_copy);
return result;
2005-09-02 23:27:25 +00:00
}
2005-11-02 04:41:09 +00:00
gboolean
_moo_edit_save_file_copy (MooEdit *edit,
const char *filename,
const char *encoding,
GError **error)
2005-11-02 04:41:09 +00:00
{
char *filename_copy, *encoding_copy;
gboolean result;
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
g_return_val_if_fail (filename != NULL, FALSE);
filename_copy = g_strdup (filename);
encoding_copy = g_strdup (normalize_encoding (encoding, TRUE));
2005-11-02 04:41:09 +00:00
result = moo_edit_save_copy_local (edit, filename_copy, encoding_copy, error);
2005-11-02 04:41:09 +00:00
g_free (filename_copy);
g_free (encoding_copy);
return result;
}
static void
set_encoding_error (GError **error)
{
GError *tmp_error = NULL;
g_set_error (&tmp_error, MOO_EDIT_FILE_ERROR,
MOO_EDIT_FILE_ERROR_ENCODING,
"%s", *error ? (*error)->message : "ERROR");
g_clear_error (error);
g_propagate_error (error, tmp_error);
}
2005-09-02 23:27:25 +00:00
/***************************************************************************/
/* File loading
*/
typedef enum {
SUCCESS,
ERROR_FILE,
ERROR_ENCODING
} LoadResult;
static LoadResult do_load (MooEdit *edit,
const char *file,
const char *encoding,
2006-12-22 02:16:50 -06:00
struct stat *statbuf,
GError **error);
2006-06-18 05:43:25 -05:00
#ifdef LOAD_BINARY
static LoadResult load_binary (MooEdit *edit,
const char *file,
GError **error);
2006-06-18 03:38:28 -05:00
#endif
2005-09-02 23:27:25 +00:00
2007-04-10 02:20:06 -05:00
const char *
_moo_get_default_encodings (void)
{
/* Translators: if translated, it should be a comma-separated list
of encodings to try when opening files. Encodings names should be
those understood by iconv, or "LOCALE" which means user's locale
charset. For instance, the default value is "UTF-8,LOCALE,ISO_8859-15,ISO_8859-1".
You want to add common preferred non-UTF8 encodings used in your locale.
Do not remove ISO_8859-15 and ISO_8859-1, instead leave them at the end,
these are common source files encodings. */
const char *to_translate = N_("encodings_list");
const char *encodings;
2007-04-10 02:20:06 -05:00
encodings = _(to_translate);
if (!strcmp (encodings, to_translate))
encodings = "UTF-8," ENCODING_LOCALE ",ISO_8859-1,ISO_8859-15";
return encodings;
}
static GSList *
get_encodings (void)
{
const char *encodings;
char **raw, **p;
GSList *result;
encodings = moo_prefs_get_string (moo_edit_setting (MOO_EDIT_PREFS_ENCODINGS));
2007-04-10 02:20:06 -05:00
if (!encodings || !encodings[0])
encodings = _moo_get_default_encodings ();
result = NULL;
raw = g_strsplit (encodings, ",", 0);
for (p = raw; p && *p; ++p)
{
const char *enc;
if (!g_ascii_strcasecmp (*p, ENCODING_LOCALE))
{
if (g_get_charset (&enc))
enc = "UTF-8";
}
else
{
enc = *p;
}
if (!g_slist_find_custom (result, enc, (GCompareFunc) g_ascii_strcasecmp))
result = g_slist_prepend (result, g_strdup (enc));
}
if (!result)
{
g_critical ("%s: oops", G_STRLOC);
result = g_slist_prepend (NULL, g_strdup ("UTF-8"));
}
g_strfreev (raw);
return g_slist_reverse (result);
}
static LoadResult
try_load (MooEdit *edit,
const char *file,
const char *encoding,
2006-12-22 02:16:50 -06:00
struct stat *statbuf,
GError **error)
2005-09-02 23:27:25 +00:00
{
2005-09-06 16:21:05 +00:00
GtkTextBuffer *buffer;
gboolean enable_highlight;
LoadResult result;
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
g_return_val_if_fail (file && file[0], FALSE);
g_return_val_if_fail (encoding && encoding[0], FALSE);
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
2006-06-18 03:38:28 -05:00
gtk_text_buffer_set_text (buffer, "", 0);
2006-04-26 14:08:58 -05:00
g_object_get (edit, "enable-highlight", &enable_highlight, NULL);
g_object_set (edit, "enable-highlight", FALSE, NULL);
2006-12-22 02:16:50 -06:00
result = do_load (edit, file, encoding, statbuf, error);
2006-04-26 14:08:58 -05:00
g_object_set (edit, "enable-highlight", enable_highlight, NULL);
return result;
}
static gboolean
moo_edit_load_local (MooEdit *edit,
const char *file,
const char *encoding,
GError **error)
{
GtkTextIter start;
GtkTextBuffer *buffer;
MooTextView *view;
2005-09-02 23:27:25 +00:00
gboolean undo;
LoadResult result = ERROR_FILE;
2006-12-22 02:16:50 -06:00
struct stat statbuf;
char *freeme = NULL;
2005-09-02 23:27:25 +00:00
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
g_return_val_if_fail (file && file[0], FALSE);
2005-06-22 18:20:32 +00:00
if (moo_edit_is_empty (edit))
undo = FALSE;
else
undo = TRUE;
view = MOO_TEXT_VIEW (edit);
2005-09-06 16:21:05 +00:00
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
block_buffer_signals (edit);
2005-06-22 18:20:32 +00:00
if (undo)
2005-09-06 16:21:05 +00:00
gtk_text_buffer_begin_user_action (buffer);
2005-06-22 18:20:32 +00:00
else
moo_text_view_begin_not_undoable_action (view);
moo_text_buffer_begin_non_interactive_action (MOO_TEXT_BUFFER (buffer));
2005-06-22 18:20:32 +00:00
if (!encoding)
{
GSList *encodings;
encodings = get_encodings ();
result = ERROR_ENCODING;
while (encodings)
{
char *enc = encodings->data;
enc = encodings->data;
encodings = g_slist_delete_link (encodings, encodings);
2006-08-09 18:41:32 -05:00
g_clear_error (error);
result = try_load (edit, file, enc, &statbuf, error);
2006-08-09 18:41:32 -05:00
if (result != ERROR_ENCODING)
{
encoding = freeme = enc;
break;
}
g_free (enc);
}
g_slist_foreach (encodings, (GFunc) g_free, NULL);
g_slist_free (encodings);
}
else
{
2006-12-22 02:16:50 -06:00
result = try_load (edit, file, encoding, &statbuf, error);
2005-06-22 18:20:32 +00:00
}
2006-06-18 05:43:25 -05:00
#ifdef LOAD_BINARY
if (result == ERROR_ENCODING)
2006-06-18 03:38:28 -05:00
{
g_clear_error (error);
2006-12-22 02:16:50 -06:00
result = load_binary (edit, file, &statbuf, error);
2006-06-18 03:38:28 -05:00
}
#endif
if (result == ERROR_ENCODING)
set_encoding_error (error);
if (result != SUCCESS)
2006-06-18 03:38:28 -05:00
gtk_text_buffer_set_text (buffer, "", 0);
2005-06-22 18:20:32 +00:00
unblock_buffer_signals (edit);
if (result == SUCCESS)
{
2005-11-03 06:47:53 +00:00
/* XXX */
gtk_text_buffer_get_start_iter (buffer, &start);
gtk_text_buffer_place_cursor (buffer, &start);
2005-11-03 06:47:53 +00:00
edit->priv->status = 0;
moo_edit_set_modified (edit, FALSE);
_moo_edit_set_filename (edit, file, encoding);
2006-12-22 02:16:50 -06:00
edit->priv->mode = statbuf.st_mode;
edit->priv->mode_set = TRUE;
_moo_edit_start_file_watch (edit);
}
else
{
_moo_edit_stop_file_watch (edit);
}
2005-06-22 18:20:32 +00:00
if (undo)
2005-09-06 16:21:05 +00:00
gtk_text_buffer_end_user_action (buffer);
2005-06-22 18:20:32 +00:00
else
moo_text_view_end_not_undoable_action (view);
2005-06-22 18:20:32 +00:00
moo_text_buffer_end_non_interactive_action (MOO_TEXT_BUFFER (buffer));
g_free (freeme);
return result == SUCCESS;
2005-06-22 18:20:32 +00:00
}
static LoadResult
do_load (MooEdit *edit,
const char *filename,
const char *encoding,
2006-12-22 02:16:50 -06:00
struct stat *statbuf,
GError **error)
2005-06-22 18:20:32 +00:00
{
2005-09-02 23:27:25 +00:00
GIOChannel *file = NULL;
GIOStatus status;
2005-09-06 16:21:05 +00:00
GtkTextBuffer *buffer;
MooEditLineEndType le = MOO_EDIT_LINE_END_NONE;
2006-04-28 19:39:30 -05:00
GString *text = NULL;
char *line = NULL;
LoadResult result = ERROR_FILE;
2005-06-22 18:20:32 +00:00
g_return_val_if_fail (filename != NULL, ERROR_FILE);
g_return_val_if_fail (encoding != NULL, ERROR_FILE);
2005-06-22 18:20:32 +00:00
if (g_stat (filename, statbuf) == 0)
{
if (
#ifdef S_ISBLK
2006-12-22 02:16:50 -06:00
S_ISBLK (statbuf->st_mode) ||
#endif
#ifdef S_ISCHR
2006-12-22 02:16:50 -06:00
S_ISCHR (statbuf->st_mode) ||
#endif
#ifdef S_ISFIFO
2006-12-22 02:16:50 -06:00
S_ISFIFO (statbuf->st_mode) ||
#endif
#ifdef S_ISSOCK
2006-12-22 02:16:50 -06:00
S_ISSOCK (statbuf->st_mode) ||
#endif
0)
{
g_set_error (error, G_FILE_ERROR,
G_FILE_ERROR_FAILED,
"Not a regular file.");
return ERROR_FILE;
}
}
2005-09-06 16:21:05 +00:00
2007-04-09 21:59:31 -05:00
file = g_io_channel_new_file (filename, "r", error);
2005-06-22 18:20:32 +00:00
2005-09-02 23:27:25 +00:00
if (!file)
return ERROR_FILE;
2005-06-22 18:20:32 +00:00
2005-09-06 16:21:05 +00:00
if (g_io_channel_set_encoding (file, encoding, error) != G_IO_STATUS_NORMAL)
{
result = ERROR_ENCODING;
2006-04-28 19:39:30 -05:00
goto error_out;
}
2006-04-28 19:39:30 -05:00
text = g_string_new (NULL);
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
2005-06-22 18:20:32 +00:00
2005-09-02 23:27:25 +00:00
while (TRUE)
{
gboolean insert_line_term = FALSE;
gsize len, line_term_pos;
MooEditLineEndType le_here = 0;
2005-06-22 18:20:32 +00:00
status = g_io_channel_read_line (file, &line, &len, &line_term_pos, error);
2005-06-22 18:20:32 +00:00
if (status != G_IO_STATUS_NORMAL && status != G_IO_STATUS_EOF)
{
if ((*error)->domain == G_CONVERT_ERROR)
result = ERROR_ENCODING;
else
result = ERROR_FILE;
2006-04-28 19:39:30 -05:00
goto error_out;
}
2005-09-02 23:27:25 +00:00
if (line)
2005-07-29 11:51:36 +00:00
{
2005-09-02 23:27:25 +00:00
if (!g_utf8_validate (line, len, NULL))
{
result = ERROR_ENCODING;
2005-09-02 23:27:25 +00:00
g_set_error (error, G_CONVERT_ERROR,
G_CONVERT_ERROR_ILLEGAL_SEQUENCE,
"Invalid UTF8 data read from file");
2006-04-28 19:39:30 -05:00
goto error_out;
2005-09-02 23:27:25 +00:00
}
if (line_term_pos != len)
{
char *line_term = line + line_term_pos;
insert_line_term = TRUE;
len = line_term_pos;
if (!strcmp ("\xe2\x80\xa9", line_term))
{
insert_line_term = FALSE;
len = len + 3;
}
else if (!strcmp (line_term, "\r"))
{
le_here = MOO_EDIT_LINE_END_MAC;
}
else if (!strcmp (line_term, "\n"))
{
le_here = MOO_EDIT_LINE_END_UNIX;
}
else if (!strcmp (line_term, "\r\n"))
{
le_here = MOO_EDIT_LINE_END_WIN32;
}
if (le_here)
{
if (le && le != le_here)
le = MOO_EDIT_LINE_END_MIX;
else
le = le_here;
}
}
2006-04-28 19:39:30 -05:00
g_string_append_len (text, line, len);
if (insert_line_term)
2006-04-28 19:39:30 -05:00
g_string_append_c (text, '\n');
#define MAX_TEXT_BUF_LEN 1000000
if (text->len > MAX_TEXT_BUF_LEN)
{
gtk_text_buffer_insert_at_cursor (buffer, text->str, text->len);
g_string_truncate (text, 0);
}
#undef MAX_TEXT_BUF_LEN
2005-09-02 23:27:25 +00:00
g_free (line);
2005-07-29 11:51:36 +00:00
}
2005-09-02 23:27:25 +00:00
if (status == G_IO_STATUS_EOF)
2005-07-29 11:51:36 +00:00
{
g_io_channel_shutdown (file, TRUE, NULL);
g_io_channel_unref (file);
break;
2005-07-29 11:51:36 +00:00
}
2005-06-22 18:20:32 +00:00
}
2006-04-28 19:39:30 -05:00
if (text->len)
gtk_text_buffer_insert_at_cursor (buffer, text->str, text->len);
edit->priv->line_end_type = le;
2006-04-28 19:39:30 -05:00
g_string_free (text, TRUE);
2005-09-02 23:27:25 +00:00
g_clear_error (error);
return SUCCESS;
2006-04-28 19:39:30 -05:00
error_out:
if (text)
g_string_free (text, TRUE);
if (file)
{
g_io_channel_shutdown (file, TRUE, NULL);
g_io_channel_unref (file);
}
g_free (line);
return result;
2005-06-22 18:20:32 +00:00
}
2006-06-18 05:43:25 -05:00
#ifdef LOAD_BINARY
2006-06-18 03:38:28 -05:00
#define TEXT_BUF_LEN (1 << 16)
#define REPLACEMENT_CHAR '\1'
#define LINE_LEN 80
static LoadResult
2006-06-18 03:38:28 -05:00
load_binary (MooEdit *edit,
const char *filename,
GError **error)
{
GIOChannel *file = NULL;
GIOStatus status;
GtkTextBuffer *buffer;
g_return_val_if_fail (filename != NULL, FALSE);
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
file = g_io_channel_new_file (filename, "r", error);
if (!file)
return ERROR_FILE;
2006-06-18 03:38:28 -05:00
g_io_channel_set_encoding (file, NULL, NULL);
while (TRUE)
{
gsize len, line_len;
char buf[TEXT_BUF_LEN];
status = g_io_channel_read_chars (file, buf, TEXT_BUF_LEN, &len, error);
if (status != G_IO_STATUS_NORMAL && status != G_IO_STATUS_EOF)
goto error_out;
if (len)
{
guint i;
for (i = 0, line_len = 0; i < len; ++i, ++line_len)
{
if (buf[i] && !(buf[i] & 0x80))
continue;
if (line_len > LINE_LEN)
{
buf[i] = '\n';
line_len = 0;
}
else
{
buf[i] = REPLACEMENT_CHAR;
}
}
gtk_text_buffer_insert_at_cursor (buffer, buf, len);
}
if (status == G_IO_STATUS_EOF)
{
g_io_channel_shutdown (file, TRUE, NULL);
g_io_channel_unref (file);
break;
}
}
g_clear_error (error);
return SUCCESS;
2006-06-18 03:38:28 -05:00
error_out:
if (file)
{
g_io_channel_shutdown (file, TRUE, NULL);
g_io_channel_unref (file);
}
return ERROR_FILE;
2006-06-18 03:38:28 -05:00
}
#undef TEXT_BUF_LEN
#undef REPLACEMENT_CHAR
#undef LINE_LEN
#endif
2005-11-03 06:47:53 +00:00
/* XXX */
2005-11-11 20:09:30 +00:00
static gboolean
2007-11-25 12:22:34 -06:00
moo_edit_reload_local (MooEdit *edit,
const char *encoding,
GError **error)
2005-06-22 18:20:32 +00:00
{
GtkTextIter start, end;
2005-09-06 16:21:05 +00:00
GtkTextBuffer *buffer;
2006-04-26 14:08:58 -05:00
gboolean result, enable_highlight;
2005-06-22 18:20:32 +00:00
g_return_val_if_fail (edit->priv->filename != NULL, FALSE);
2005-09-06 16:21:05 +00:00
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
2005-06-22 18:20:32 +00:00
block_buffer_signals (edit);
2005-09-06 16:21:05 +00:00
gtk_text_buffer_begin_user_action (buffer);
2005-06-22 18:20:32 +00:00
2005-09-06 16:21:05 +00:00
gtk_text_buffer_get_bounds (buffer, &start, &end);
gtk_text_buffer_delete (buffer, &start, &end);
2006-04-26 14:08:58 -05:00
g_object_get (edit, "enable-highlight", &enable_highlight, NULL);
g_object_set (edit, "enable-highlight", FALSE, NULL);
2005-06-22 18:20:32 +00:00
result = _moo_edit_load_file (edit, edit->priv->filename,
2007-11-25 12:22:34 -06:00
encoding ? encoding : edit->priv->encoding,
error);
2006-04-26 14:08:58 -05:00
g_object_set (edit, "enable-highlight", enable_highlight, NULL);
gtk_text_buffer_end_user_action (buffer);
unblock_buffer_signals (edit);
if (result)
2005-09-02 23:27:25 +00:00
{
2005-06-22 18:20:32 +00:00
edit->priv->status = 0;
moo_edit_set_modified (edit, FALSE);
_moo_edit_start_file_watch (edit);
2005-09-02 23:27:25 +00:00
g_clear_error (error);
2005-06-22 18:20:32 +00:00
}
return result;
2005-06-22 18:20:32 +00:00
}
2005-09-02 23:27:25 +00:00
/***************************************************************************/
/* File saving
*/
2005-11-10 07:15:16 +00:00
#ifdef __WIN32__
#define BAK_SUFFIX ".bak"
#define DEFAULT_LE_TYPE MOO_EDIT_LINE_END_WIN32
2005-11-10 07:15:16 +00:00
#else
#define BAK_SUFFIX "~"
#define DEFAULT_LE_TYPE MOO_EDIT_LINE_END_UNIX
2005-11-10 07:15:16 +00:00
#endif
2005-09-02 23:27:25 +00:00
static gboolean do_write (MooEdit *edit,
const char *file,
const char *encoding,
GError **error);
2005-11-02 04:41:09 +00:00
static gboolean
do_save_local (MooEdit *edit,
const char *filename,
const char *encoding,
GError **error,
gboolean *retval)
2005-06-22 18:20:32 +00:00
{
*retval = TRUE;
2005-09-02 23:27:25 +00:00
if (!do_write (edit, filename, encoding, error))
{
if ((*error)->domain != G_CONVERT_ERROR ||
_moo_encodings_equal (encoding, MOO_ENCODING_UTF8))
{
*retval = FALSE;
return FALSE;
}
g_clear_error (error);
if (!do_write (edit, filename, MOO_ENCODING_UTF8, error))
{
*retval = FALSE;
return FALSE;
}
2005-09-02 23:27:25 +00:00
*retval = FALSE;
set_encoding_error (error);
}
2005-09-02 23:27:25 +00:00
return TRUE;
}
2005-11-02 04:41:09 +00:00
static gboolean
moo_edit_save_local (MooEdit *edit,
const char *filename,
const char *encoding,
MooEditSaveFlags flags,
GError **error)
2005-11-02 04:41:09 +00:00
{
gboolean result;
2005-11-02 04:41:09 +00:00
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
g_return_val_if_fail (filename && filename[0], FALSE);
if (do_save_local (edit, filename, encoding, error, &result))
{
edit->priv->status = 0;
_moo_edit_set_filename (edit, filename,
result ? encoding : MOO_ENCODING_UTF8);
moo_edit_set_modified (edit, FALSE);
_moo_edit_start_file_watch (edit);
}
return result;
}
static gboolean
moo_edit_save_copy_local (MooEdit *edit,
const char *filename,
const char *encoding,
GError **error)
{
gboolean result;
g_return_val_if_fail (MOO_IS_EDIT (edit), FALSE);
g_return_val_if_fail (filename && filename[0], FALSE);
do_save_local (edit, filename, encoding, error, &result);
return result;
2005-11-02 04:41:09 +00:00
}
static gboolean
do_write (MooEdit *edit,
const char *filename,
const char *encoding,
GError **error)
2005-09-02 23:27:25 +00:00
{
2006-12-22 02:16:50 -06:00
int fd = -1;
2005-09-02 23:27:25 +00:00
GIOChannel *file;
GIOStatus status;
GtkTextIter line_start;
2005-09-06 16:21:05 +00:00
GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
MooEditLineEndType le_type = DEFAULT_LE_TYPE;
const char *le = "\n";
gssize le_len = 1;
2006-12-22 02:16:50 -06:00
mode_t mode;
2005-09-02 23:27:25 +00:00
g_return_val_if_fail (filename != NULL, FALSE);
if (encoding && (!g_ascii_strcasecmp (encoding, "UTF-8") || !g_ascii_strcasecmp (encoding, "UTF8")))
2005-09-02 23:27:25 +00:00
encoding = NULL;
2006-12-22 13:25:06 -06:00
#ifdef __WIN32__
mode = S_IRUSR | S_IWUSR;
#else
2006-12-22 02:16:50 -06:00
if (edit->priv->mode_set)
mode = (edit->priv->mode & (S_IRWXU | S_IRWXG | S_IRWXO)) | S_IRUSR | S_IWUSR;
else
mode = S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH;
2006-12-22 13:25:06 -06:00
#endif
2006-12-22 02:16:50 -06:00
errno = 0;
fd = g_open (filename, O_BINARY | O_WRONLY | O_CREAT | O_TRUNC, mode);
2005-06-22 18:20:32 +00:00
2006-12-22 02:16:50 -06:00
if (fd == -1)
{
int err = errno;
g_set_error (error, G_FILE_ERROR,
g_file_error_from_errno (err),
"%s", g_strerror (err));
2005-09-02 23:27:25 +00:00
return FALSE;
2006-12-22 02:16:50 -06:00
}
#ifndef __WIN32__
file = g_io_channel_unix_new (fd);
#else
file = g_io_channel_win32_new_fd (fd);
#endif
g_io_channel_set_close_on_unref (file, TRUE);
2005-09-02 23:27:25 +00:00
if (encoding && g_io_channel_set_encoding (file, encoding, error) != G_IO_STATUS_NORMAL)
2005-06-22 18:20:32 +00:00
{
2005-09-02 23:27:25 +00:00
g_io_channel_shutdown (file, TRUE, NULL);
g_io_channel_unref (file);
return FALSE;
2005-06-22 18:20:32 +00:00
}
switch (edit->priv->line_end_type)
{
case MOO_EDIT_LINE_END_NONE:
le_type = DEFAULT_LE_TYPE;
break;
case MOO_EDIT_LINE_END_UNIX:
case MOO_EDIT_LINE_END_WIN32:
case MOO_EDIT_LINE_END_MAC:
le_type = edit->priv->line_end_type;
break;
case MOO_EDIT_LINE_END_MIX:
edit->priv->line_end_type = le_type = DEFAULT_LE_TYPE;
break;
}
switch (le_type)
{
case MOO_EDIT_LINE_END_UNIX:
le = "\n";
le_len = 1;
break;
case MOO_EDIT_LINE_END_WIN32:
le = "\r\n";
le_len = 2;
break;
case MOO_EDIT_LINE_END_MAC:
le = "\r";
le_len = 1;
break;
case MOO_EDIT_LINE_END_MIX:
case MOO_EDIT_LINE_END_NONE:
g_assert_not_reached ();
}
2005-09-06 16:21:05 +00:00
gtk_text_buffer_get_start_iter (buffer, &line_start);
2005-09-02 23:27:25 +00:00
do
2005-06-22 18:20:32 +00:00
{
2005-09-02 23:27:25 +00:00
gsize written;
GtkTextIter line_end = line_start;
if (!gtk_text_iter_ends_line (&line_start))
{
char *line;
gssize len = -1;
gtk_text_iter_forward_to_line_end (&line_end);
line = gtk_text_buffer_get_slice (buffer, &line_start, &line_end, TRUE);
2005-09-02 23:27:25 +00:00
status = g_io_channel_write_chars (file, line, len, &written, error);
g_free (line);
/* XXX */
if (status != G_IO_STATUS_NORMAL)
{
g_io_channel_shutdown (file, TRUE, NULL);
g_io_channel_unref (file);
return FALSE;
}
}
2006-04-05 15:29:45 -05:00
if (!gtk_text_iter_is_end (&line_end))
2005-09-02 23:27:25 +00:00
{
status = g_io_channel_write_chars (file, le, le_len, &written, error);
if (written < (gsize)le_len || status != G_IO_STATUS_NORMAL)
{
g_io_channel_shutdown (file, TRUE, NULL);
g_io_channel_unref (file);
return FALSE;
}
}
2005-06-22 18:20:32 +00:00
}
2005-09-02 23:27:25 +00:00
while (gtk_text_iter_forward_line (&line_start));
2005-06-22 18:20:32 +00:00
/* glib #320668 */
status = g_io_channel_flush (file, error);
if (status != G_IO_STATUS_NORMAL)
{
g_io_channel_shutdown (file, FALSE, NULL);
g_io_channel_unref (file);
return FALSE;
}
2005-09-02 23:27:25 +00:00
status = g_io_channel_shutdown (file, TRUE, error);
2005-06-22 18:20:32 +00:00
2005-09-02 23:27:25 +00:00
if (status != G_IO_STATUS_NORMAL)
{
g_io_channel_unref (file);
return FALSE;
}
g_io_channel_unref (file);
g_clear_error (error);
return TRUE;
2005-06-22 18:20:32 +00:00
}
2005-09-02 23:27:25 +00:00
/***************************************************************************/
/* Aux stuff
*/
static void
block_buffer_signals (MooEdit *edit)
2005-06-22 18:20:32 +00:00
{
2005-09-06 16:21:05 +00:00
GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
g_signal_handler_block (buffer, edit->priv->modified_changed_handler_id);
2005-06-22 18:20:32 +00:00
}
static void
unblock_buffer_signals (MooEdit *edit)
2005-06-22 18:20:32 +00:00
{
2005-09-06 16:21:05 +00:00
GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
g_signal_handler_unblock (buffer, edit->priv->modified_changed_handler_id);
2005-06-22 18:20:32 +00:00
}
static void
file_watch_callback (G_GNUC_UNUSED MooFileWatch *watch,
MooFileEvent *event,
gpointer data)
{
2006-12-10 16:48:19 -06:00
MooEdit *edit = data;
2006-12-10 16:48:19 -06:00
g_return_if_fail (MOO_IS_EDIT (data));
g_return_if_fail (event->monitor_id == edit->priv->file_monitor_id);
g_return_if_fail (edit->priv->filename != NULL);
g_return_if_fail (!(edit->priv->status & MOO_EDIT_CHANGED_ON_DISK));
switch (event->code)
{
case MOO_FILE_EVENT_CHANGED:
edit->priv->modified_on_disk = TRUE;
break;
case MOO_FILE_EVENT_DELETED:
edit->priv->deleted_from_disk = TRUE;
2005-11-16 11:23:49 +00:00
edit->priv->file_monitor_id = 0;
break;
case MOO_FILE_EVENT_ERROR:
/* XXX and what to do now? */
break;
case MOO_FILE_EVENT_CREATED:
g_critical ("%s: oops", G_STRLOC);
break;
}
check_file_status (edit, FALSE);
}
static void
2006-05-02 21:28:25 -05:00
_moo_edit_start_file_watch (MooEdit *edit)
2005-06-22 18:20:32 +00:00
{
MooFileWatch *watch;
GError *error = NULL;
2005-06-22 18:20:32 +00:00
watch = _moo_editor_get_file_watch (edit->priv->editor);
g_return_if_fail (watch != NULL);
if (edit->priv->file_monitor_id)
moo_file_watch_cancel_monitor (watch, edit->priv->file_monitor_id);
edit->priv->file_monitor_id = 0;
2005-06-22 18:20:32 +00:00
2005-09-02 23:27:25 +00:00
g_return_if_fail ((edit->priv->status & MOO_EDIT_CHANGED_ON_DISK) == 0);
2005-06-22 18:20:32 +00:00
g_return_if_fail (edit->priv->filename != NULL);
edit->priv->file_monitor_id =
moo_file_watch_create_monitor (watch, edit->priv->filename,
file_watch_callback,
edit, NULL, &error);
if (!edit->priv->file_monitor_id)
{
2006-05-02 21:28:25 -05:00
g_warning ("%s: could not start watch for '%s'",
G_STRLOC, edit->priv->filename);
if (error)
{
g_warning ("%s: %s", G_STRLOC, error->message);
g_error_free (error);
}
2006-05-02 21:28:25 -05:00
return;
}
2005-06-22 18:20:32 +00:00
if (!edit->priv->focus_in_handler_id)
edit->priv->focus_in_handler_id =
g_signal_connect (edit, "focus-in-event",
G_CALLBACK (focus_in_cb),
NULL);
}
void
_moo_edit_stop_file_watch (MooEdit *edit)
2005-06-22 18:20:32 +00:00
{
MooFileWatch *watch;
watch = _moo_editor_get_file_watch (edit->priv->editor);
g_return_if_fail (watch != NULL);
2005-09-02 23:27:25 +00:00
if (edit->priv->file_monitor_id)
moo_file_watch_cancel_monitor (watch, edit->priv->file_monitor_id);
edit->priv->file_monitor_id = 0;
2005-09-02 23:27:25 +00:00
if (edit->priv->focus_in_handler_id)
{
2005-06-22 18:20:32 +00:00
g_signal_handler_disconnect (edit, edit->priv->focus_in_handler_id);
edit->priv->focus_in_handler_id = 0;
}
}
2005-11-04 01:35:22 +00:00
static gboolean
focus_in_cb (MooEdit *edit)
2005-06-22 18:20:32 +00:00
{
check_file_status (edit, TRUE);
return FALSE;
}
static void
check_file_status (MooEdit *edit,
gboolean in_focus_only)
2005-06-22 18:20:32 +00:00
{
if (in_focus_only && !GTK_WIDGET_HAS_FOCUS (edit))
return;
2005-06-22 18:20:32 +00:00
g_return_if_fail (edit->priv->filename != NULL);
g_return_if_fail (!(edit->priv->status & MOO_EDIT_CHANGED_ON_DISK));
2005-06-22 18:20:32 +00:00
if (edit->priv->deleted_from_disk)
file_deleted (edit);
else if (edit->priv->modified_on_disk)
file_modified_on_disk (edit);
2005-06-22 18:20:32 +00:00
}
2005-11-04 01:35:22 +00:00
static void
file_modified_on_disk (MooEdit *edit)
2005-06-22 18:20:32 +00:00
{
g_return_if_fail (edit->priv->filename != NULL);
edit->priv->modified_on_disk = FALSE;
edit->priv->deleted_from_disk = FALSE;
_moo_edit_stop_file_watch (edit);
2005-09-02 23:27:25 +00:00
add_status (edit, MOO_EDIT_MODIFIED_ON_DISK);
2005-06-22 18:20:32 +00:00
}
2005-11-04 01:35:22 +00:00
static void
file_deleted (MooEdit *edit)
2005-06-22 18:20:32 +00:00
{
g_return_if_fail (edit->priv->filename != NULL);
edit->priv->modified_on_disk = FALSE;
edit->priv->deleted_from_disk = FALSE;
_moo_edit_stop_file_watch (edit);
2005-09-02 23:27:25 +00:00
add_status (edit, MOO_EDIT_DELETED);
2005-06-22 18:20:32 +00:00
}
2005-11-04 01:35:22 +00:00
static void
add_status (MooEdit *edit,
MooEditStatus s)
2005-06-22 18:20:32 +00:00
{
2005-09-02 23:27:25 +00:00
edit->priv->status |= s;
g_signal_emit_by_name (edit, "doc-status-changed", NULL);
2005-06-22 18:20:32 +00:00
}
2005-11-04 01:35:22 +00:00
static void
remove_untitled (gpointer destroyed,
MooEdit *edit)
{
gpointer n = g_hash_table_lookup (UNTITLED_NO, edit);
if (n)
{
UNTITLED = g_slist_remove (UNTITLED, n);
g_hash_table_remove (UNTITLED_NO, edit);
if (!destroyed)
g_object_weak_unref (G_OBJECT (edit),
(GWeakNotify) remove_untitled,
GINT_TO_POINTER (TRUE));
}
}
2005-11-04 01:35:22 +00:00
static int
add_untitled (MooEdit *edit)
{
int n;
if (!(n = GPOINTER_TO_INT (g_hash_table_lookup (UNTITLED_NO, edit))))
{
for (n = 1; ; ++n)
{
if (!g_slist_find (UNTITLED, GINT_TO_POINTER (n)))
{
UNTITLED = g_slist_prepend (UNTITLED, GINT_TO_POINTER (n));
break;
}
}
g_hash_table_insert (UNTITLED_NO, edit, GINT_TO_POINTER (n));
g_object_weak_ref (G_OBJECT (edit),
(GWeakNotify) remove_untitled,
GINT_TO_POINTER (TRUE));
}
return n;
}
2007-11-29 19:53:33 -06:00
const char *
_moo_edit_get_default_encoding (void)
{
2007-11-29 19:53:33 -06:00
return moo_prefs_get_string (moo_edit_setting (MOO_EDIT_PREFS_ENCODING_SAVE));
}
2005-11-04 01:35:22 +00:00
void
_moo_edit_set_filename (MooEdit *edit,
const char *file,
const char *encoding)
2005-06-22 18:20:32 +00:00
{
2007-11-29 19:53:33 -06:00
char *tmp1, *tmp3, *tmp4;
2007-05-02 00:37:36 -05:00
tmp1 = edit->priv->filename;
tmp3 = edit->priv->display_filename;
tmp4 = edit->priv->display_basename;
2005-06-22 18:20:32 +00:00
if (!UNTITLED_NO)
UNTITLED_NO = g_hash_table_new (g_direct_hash, g_direct_equal);
2005-09-02 23:27:25 +00:00
if (!file)
2005-06-22 18:20:32 +00:00
{
int n = add_untitled (edit);
2005-09-02 23:27:25 +00:00
edit->priv->filename = NULL;
if (n == 1)
edit->priv->display_filename = g_strdup (_("Untitled"));
else
edit->priv->display_filename = g_strdup_printf (_("Untitled %d"), n);
edit->priv->display_basename = g_strdup (edit->priv->display_filename);
2005-06-22 18:20:32 +00:00
}
2005-09-02 23:27:25 +00:00
else
{
char *basename;
remove_untitled (NULL, edit);
2005-09-02 23:27:25 +00:00
edit->priv->filename = g_strdup (file);
edit->priv->display_filename = _moo_edit_filename_to_utf8 (file);
basename = g_path_get_basename (file);
edit->priv->display_basename = _moo_edit_filename_to_utf8 (basename);
g_free (basename);
2005-06-22 18:20:32 +00:00
}
2007-05-02 00:37:36 -05:00
if (!encoding)
2007-11-29 19:53:33 -06:00
_moo_edit_set_encoding (edit, _moo_edit_get_default_encoding ());
else
2007-11-29 19:53:33 -06:00
_moo_edit_set_encoding (edit, encoding);
2005-06-22 18:20:32 +00:00
2005-09-02 23:27:25 +00:00
g_signal_emit_by_name (edit, "filename-changed", edit->priv->filename, NULL);
moo_edit_status_changed (edit);
2007-05-02 00:37:36 -05:00
g_free (tmp1);
g_free (tmp3);
g_free (tmp4);
2005-06-22 18:20:32 +00:00
}
static char *
2005-11-04 01:35:22 +00:00
_moo_edit_filename_to_utf8 (const char *filename)
{
GError *err = NULL;
char *utf_filename;
g_return_val_if_fail (filename != NULL, NULL);
utf_filename = g_filename_to_utf8 (filename, -1, NULL, NULL, &err);
if (!utf_filename)
{
g_critical ("%s: could not convert filename to UTF8", G_STRLOC);
if (err)
{
g_critical ("%s: %s", G_STRLOC, err->message);
g_error_free (err);
err = NULL;
}
utf_filename = g_strdup ("<Unknown>");
}
return utf_filename;
}
GdkPixbuf *
_moo_edit_get_icon (MooEdit *doc,
GtkWidget *widget,
GtkIconSize size)
{
2008-08-31 01:40:29 -05:00
return moo_get_icon_for_file (doc->priv->filename, widget, size);
}