/*
* Copyright (C) 2012-2013 Robin Haberkorn
*
* 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 .
*/
#ifndef __RING_H
#define __RING_H
#include
#include
#include
#include
#include
#include
#include "sciteco.h"
#include "interface.h"
#include "undo.h"
#include "qregisters.h"
#include "parser.h"
/*
* Auxiliary functions
*/
static inline bool
is_glob_pattern(const gchar *str)
{
return strchr(str, '*') || strchr(str, '?');
}
/*
* Get absolute/full version of a possibly relative path.
* Works with existing and non-existing paths (in the latter case,
* heuristics may be applied.)
*/
gchar *get_absolute_path(const gchar *path);
/*
* Classes
*/
class Buffer {
class UndoTokenClose : public UndoToken {
Buffer *buffer;
public:
UndoTokenClose(Buffer *_buffer)
: UndoToken(), buffer(_buffer) {}
void run(void);
};
public:
TAILQ_ENTRY(Buffer) buffers;
gchar *filename;
gint dot;
gint savepoint_id;
bool dirty;
private:
typedef void document;
document *doc;
public:
Buffer() : filename(NULL), dot(0), savepoint_id(0), dirty(false)
{
doc = (document *)interface.ssm(SCI_CREATEDOCUMENT);
}
~Buffer()
{
interface.ssm(SCI_RELEASEDOCUMENT, 0, (sptr_t)doc);
g_free(filename);
}
inline Buffer *&
next(void)
{
return TAILQ_NEXT(this, buffers);
}
inline Buffer *&
prev(void)
{
TAILQ_HEAD(Head, Buffer);
return TAILQ_PREV(this, Head, buffers);
}
inline void
set_filename(const gchar *filename)
{
gchar *resolved = get_absolute_path(filename);
g_free(Buffer::filename);
Buffer::filename = resolved;
interface.info_update(this);
}
inline void
edit(void)
{
interface.ssm(SCI_SETDOCPOINTER, 0, (sptr_t)doc);
interface.ssm(SCI_GOTOPOS, dot);
interface.info_update(this);
}
inline void
undo_edit(void)
{
interface.undo_info_update(this);
undo.push_msg(SCI_GOTOPOS, dot);
undo.push_msg(SCI_SETDOCPOINTER, 0, (sptr_t)doc);
}
bool load(const gchar *filename);
inline void
undo_close(void)
{
undo.push(new UndoTokenClose(this));
}
};
extern class Ring {
/*
* Emitted after a buffer close
* The pointer is the only remaining reference to the buffer!
*/
class UndoTokenEdit : public UndoToken {
Ring *ring;
Buffer *buffer;
public:
UndoTokenEdit(Ring *_ring, Buffer *_buffer)
: UndoToken(), ring(_ring), buffer(_buffer) {}
~UndoTokenEdit()
{
if (buffer)
delete buffer;
}
void run(void);
};
class UndoTokenRemoveFile : public UndoToken {
gchar *filename;
public:
UndoTokenRemoveFile(const gchar *_filename)
: filename(g_strdup(_filename)) {}
~UndoTokenRemoveFile()
{
g_free(filename);
}
void
run(void)
{
g_unlink(filename);
}
};
TAILQ_HEAD(Head, Buffer) head;
public:
Buffer *current;
Ring() : current(NULL)
{
TAILQ_INIT(&head);
}
~Ring();
inline Buffer *
first(void)
{
return TAILQ_FIRST(&head);
}
inline Buffer *
last(void)
{
return TAILQ_LAST(&head, Head);
}
Buffer *find(const gchar *filename);
Buffer *find(gint64 id);
void dirtify(void);
bool is_any_dirty(void);
bool edit(gint64 id);
void edit(const gchar *filename);
inline void
undo_edit(void)
{
current->dot = interface.ssm(SCI_GETCURRENTPOS);
undo.push_var(current);
current->undo_edit();
}
bool save(const gchar *filename);
void close(Buffer *buffer);
void close(void);
inline void
undo_close(void)
{
current->undo_close();
}
} ring;
/*
* Command states
*/
class StateEditFile : public StateExpectString {
private:
bool allowFilename;
void do_edit(const gchar *filename) throw (Error);
void do_edit(gint64 id) throw (Error);
void initial(void) throw (Error);
State *done(const gchar *str) throw (Error);
};
class StateSaveFile : public StateExpectString {
private:
State *done(const gchar *str) throw (Error);
};
namespace States {
extern StateEditFile editfile;
extern StateSaveFile savefile;
}
/* FIXME: clean up current_save_dot() usage */
static inline void
current_save_dot(void)
{
gint dot = interface.ssm(SCI_GETCURRENTPOS);
if (ring.current)
ring.current->dot = dot;
else if (QRegisters::current)
QRegisters::current->dot = dot;
}
#endif