mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-19 05:53:35 +00:00
045aff3d9f
The removal of duplicates is done in LastCommandsSection::add and uses the erase-remove idiom for performance. Most of the patch is a cleanup of GuiCommandBuffer: * remove history_ member, that was a copy of the session lastcommands vector. Use instead a wrapper history() around it and a addHistory wrapper for adding new entries. * Make sure that there is only one place where commands are added to history. The code used to maintain a list for interactive editing, and a list for saving the session. They could be different in terms of leading/trailing spaces. * [unrelated] remove command_ member, which is just a copy of LyXAction list of commmands. Use directly lyxaction instead.
466 lines
11 KiB
C++
466 lines
11 KiB
C++
// -*- C++ -*-
|
|
/**
|
|
* \file Session.h
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author Lars Gullik Bjønnes
|
|
* \author Bo Peng
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*/
|
|
|
|
#ifndef SESSION_H
|
|
#define SESSION_H
|
|
|
|
#include "support/FileName.h"
|
|
#include "support/types.h"
|
|
|
|
#include <list>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
/** This session file maintains
|
|
1. the latest documents loaded (lastfiles)
|
|
2. cursor positions of files closed (lastfilepos)
|
|
3. opened files when a lyx session is closed (lastopened)
|
|
4. bookmarks
|
|
5. general purpose session info in the form of key/value pairs
|
|
6. the latest commands entered in the command buffer (lastcommands)
|
|
*/
|
|
namespace lyx {
|
|
|
|
/* base class for all sections in the session file
|
|
*/
|
|
class SessionSection
|
|
{
|
|
public:
|
|
///
|
|
SessionSection() {}
|
|
///
|
|
virtual ~SessionSection() {}
|
|
|
|
/// read section from std::istream
|
|
virtual void read(std::istream & is) = 0;
|
|
|
|
/// write to std::ostream
|
|
virtual void write(std::ostream & os) const = 0;
|
|
|
|
private:
|
|
/// uncopiable
|
|
SessionSection(SessionSection const &);
|
|
void operator=(SessionSection const &);
|
|
};
|
|
|
|
|
|
class LastFilesSection : SessionSection
|
|
{
|
|
public:
|
|
///
|
|
typedef std::vector<support::FileName> LastFiles;
|
|
|
|
public:
|
|
///
|
|
explicit LastFilesSection(unsigned int num = 4);
|
|
|
|
///
|
|
void read(std::istream & is) override;
|
|
|
|
///
|
|
void write(std::ostream & os) const override;
|
|
|
|
/// Return lastfiles container (vector)
|
|
LastFiles const lastFiles() const { return lastfiles; }
|
|
|
|
/** Insert #file# into the lastfile vector.
|
|
This funtion inserts #file# into the last files list. If the file
|
|
already exists it is moved to the top of the list, else exist it
|
|
is placed on the top of the list. If the list is full the last
|
|
file in the list is popped from the end.
|
|
@param file the file to insert in the lastfile list.
|
|
*/
|
|
void add(support::FileName const & file);
|
|
|
|
private:
|
|
/// Default number of lastfiles.
|
|
unsigned int const default_num_last_files;
|
|
|
|
/// Max number of lastfiles.
|
|
unsigned int const absolute_max_last_files;
|
|
|
|
/// a list of lastfiles
|
|
LastFiles lastfiles;
|
|
|
|
/// number of files in the lastfiles list.
|
|
unsigned int num_lastfiles;
|
|
|
|
/** Used by the constructor to set the number of stored last files.
|
|
@param num the number of lastfiles to set.
|
|
*/
|
|
void setNumberOfLastFiles(unsigned int num);
|
|
};
|
|
|
|
|
|
class LastOpenedSection : SessionSection
|
|
{
|
|
public:
|
|
///
|
|
struct LastOpenedFile {
|
|
LastOpenedFile() : file_name(), active(false) {}
|
|
|
|
LastOpenedFile(support::FileName file_name_, bool active_)
|
|
: file_name(file_name_), active(active_) {}
|
|
|
|
support::FileName file_name;
|
|
bool active;
|
|
};
|
|
///
|
|
typedef std::vector<LastOpenedFile> LastOpened;
|
|
|
|
public:
|
|
///
|
|
void read(std::istream & is) override;
|
|
|
|
///
|
|
void write(std::ostream & os) const override;
|
|
|
|
/// Return lastopened container (vector)
|
|
LastOpened const getfiles() const { return lastopened; }
|
|
|
|
/** add file to lastopened file list
|
|
@param file filename to add
|
|
*/
|
|
void add(support::FileName const & file, bool active = false);
|
|
|
|
/** clear lastopened file list
|
|
*/
|
|
void clear();
|
|
|
|
private:
|
|
/// a list of lastopened files
|
|
LastOpened lastopened;
|
|
};
|
|
|
|
|
|
class LastFilePosSection : SessionSection
|
|
{
|
|
public:
|
|
///
|
|
struct FilePos {
|
|
FilePos() : pit(0), pos(0) {}
|
|
support::FileName file;
|
|
pit_type pit;
|
|
pos_type pos;
|
|
};
|
|
|
|
///
|
|
typedef std::list<FilePos> FilePosList;
|
|
|
|
public:
|
|
///
|
|
LastFilePosSection() : num_lastfilepos(100) {}
|
|
|
|
///
|
|
void read(std::istream & is) override;
|
|
|
|
///
|
|
void write(std::ostream & os) const override;
|
|
|
|
/** add cursor position to the fname entry in the filepos list
|
|
@param pos file name and position of the cursor when the BufferView is closed.
|
|
*/
|
|
void save(FilePos const & pos);
|
|
|
|
/** load saved cursor position from the fname entry in the filepos list
|
|
@param fname file entry for which to load position information
|
|
*/
|
|
FilePos load(support::FileName const & fname) const;
|
|
|
|
private:
|
|
/// default number of lastfilepos to save */
|
|
unsigned int const num_lastfilepos;
|
|
|
|
|
|
/// a list of file positions
|
|
FilePosList lastfilepos;
|
|
};
|
|
|
|
|
|
class BookmarksSection : SessionSection
|
|
{
|
|
public:
|
|
/// A bookmark is composed of three parts
|
|
/// 1. filename
|
|
/// 2. bottom (whole document) level pit and pos, used to (inaccurately) save/restore a bookmark
|
|
/// 3. top level id and pos, used to accurately locate bookmark when lyx is running
|
|
/// top and bottom level information sometimes needs to be sync'ed. In particular,
|
|
/// top_id is determined when a bookmark is restored from session; and
|
|
/// bottom_pit and bottom_pos are determined from top_id when a bookmark
|
|
/// is saved to session. (What a mess! :-)
|
|
///
|
|
/// TODO: bottom level pit and pos will be replaced by StableDocIterator
|
|
class Bookmark {
|
|
public:
|
|
/// Filename
|
|
support::FileName filename;
|
|
/// Bottom level cursor pit, will be saved/restored by .lyx/session
|
|
pit_type bottom_pit;
|
|
/// Bottom level cursor position, will be saved/restore by .lyx/session
|
|
pos_type bottom_pos;
|
|
/// Top level cursor id, used to lcoate bookmarks for opened files
|
|
int top_id;
|
|
/// Top level cursor position within a paragraph
|
|
pos_type top_pos;
|
|
///
|
|
Bookmark() : bottom_pit(0), bottom_pos(0), top_id(0), top_pos(0) {}
|
|
///
|
|
Bookmark(support::FileName const & f, pit_type pit, pos_type pos, int id, pos_type tpos)
|
|
: filename(f), bottom_pit(pit), bottom_pos(pos), top_id(id), top_pos(tpos) {}
|
|
/// set bookmark top_id, this is because newly loaded bookmark
|
|
/// may have zero par_id and par_pit can change during editing, see bug 3092
|
|
void updatePos(pit_type pit, pos_type pos, int id) {
|
|
bottom_pit = pit;
|
|
bottom_pos = pos;
|
|
top_id = id;
|
|
}
|
|
};
|
|
|
|
///
|
|
typedef std::vector<Bookmark> BookmarkList;
|
|
|
|
public:
|
|
///
|
|
BookmarksSection() : bookmarks(max_bookmarks + 1) {}
|
|
|
|
/// Save the current position as bookmark
|
|
void save(support::FileName const & fname, pit_type bottom_pit, pos_type bottom_pos,
|
|
int top_id, pos_type top_pos, unsigned int idx);
|
|
|
|
/// return bookmark 0-9, bookmark 0 is the temporary bookmark
|
|
Bookmark const & bookmark(unsigned int i) const;
|
|
|
|
/// does the given bookmark have a saved position ?
|
|
bool isValid(unsigned int i) const;
|
|
|
|
/// is there at least one bookmark that has a saved position ?
|
|
bool hasValid() const;
|
|
|
|
///
|
|
unsigned int size() const { return bookmarks.size(); }
|
|
|
|
/// clear all bookmarks
|
|
void clear();
|
|
|
|
///
|
|
void read(std::istream & is) override;
|
|
|
|
///
|
|
void write(std::ostream & os) const override;
|
|
|
|
/** return bookmark list. Non-const container is used since
|
|
bookmarks will be cleaned after use.
|
|
*/
|
|
BookmarkList & load() { return bookmarks; }
|
|
|
|
///
|
|
typedef std::vector<std::pair<unsigned int, pos_type>> BookmarkPosList;
|
|
|
|
/// return a list of bookmarks and position for this paragraph
|
|
BookmarkPosList bookmarksInPar(support::FileName const & fn, int par_id) const;
|
|
|
|
/* An insertion/deletion in paragraph \c par_id of buffer \c fn
|
|
* lead to an offset \c offset after position \c pos. Update
|
|
* bookmarks accordingly.
|
|
*/
|
|
void adjustPosAfterPos(support::FileName const & fn,
|
|
int const par_id, pos_type pos, int offset);
|
|
|
|
private:
|
|
|
|
/// allow 9 regular bookmarks, bookmark 0 is temporary
|
|
unsigned int const max_bookmarks = 9;
|
|
|
|
/// a list of bookmarks
|
|
BookmarkList bookmarks;
|
|
|
|
};
|
|
|
|
|
|
class LastCommandsSection : SessionSection
|
|
{
|
|
public:
|
|
///
|
|
typedef std::vector<std::string> LastCommands;
|
|
|
|
public:
|
|
///
|
|
LastCommandsSection(unsigned int num);
|
|
///
|
|
void read(std::istream & is) override;
|
|
|
|
///
|
|
void write(std::ostream & os) const override;
|
|
|
|
/// Return lastcommands container (vector)
|
|
LastCommands const & getcommands() const { return lastcommands; }
|
|
|
|
/** add command to lastcommands list
|
|
@param command command to add
|
|
*/
|
|
void add(std::string const & command);
|
|
|
|
/** clear lastcommands list
|
|
*/
|
|
void clear();
|
|
|
|
private:
|
|
/// number of commands in the lastcommands list.
|
|
unsigned int num_lastcommands;
|
|
|
|
/** Used by the constructor to set the number of stored last commands.
|
|
@param num the number of lastcommands to set.
|
|
*/
|
|
void setNumberOfLastCommands(unsigned int num);
|
|
|
|
/// a list of lastopened commands
|
|
LastCommands lastcommands;
|
|
|
|
/// Default number of lastcommands.
|
|
unsigned int const default_num_last_commands;
|
|
|
|
/// Max number of lastcommands.
|
|
unsigned int const absolute_max_last_commands;
|
|
};
|
|
|
|
|
|
class AuthFilesSection : SessionSection
|
|
{
|
|
public:
|
|
///
|
|
explicit AuthFilesSection();
|
|
|
|
///
|
|
void read(std::istream & is) override;
|
|
|
|
///
|
|
void write(std::ostream & os) const override;
|
|
|
|
///
|
|
bool find(std::string const & name) const;
|
|
|
|
///
|
|
void insert(std::string const & name);
|
|
|
|
private:
|
|
/// set of document files authorized for external conversion
|
|
std::set<std::string> auth_files_;
|
|
};
|
|
|
|
|
|
class ShellEscapeSection : SessionSection
|
|
{
|
|
public:
|
|
///
|
|
explicit ShellEscapeSection() {}
|
|
|
|
///
|
|
void read(std::istream & is) override;
|
|
|
|
///
|
|
void write(std::ostream & os) const override;
|
|
|
|
///
|
|
bool find(std::string const & name) const;
|
|
|
|
///
|
|
bool findAuth(std::string const & name) const;
|
|
|
|
///
|
|
void insert(std::string const & name, bool auth = false);
|
|
|
|
///
|
|
void remove(std::string const & name);
|
|
|
|
private:
|
|
/// set of document files authorized for external conversion
|
|
std::set<std::string> shellescape_files_;
|
|
};
|
|
|
|
|
|
class Session
|
|
{
|
|
public:
|
|
/// Read the session file. @param num length of lastfiles
|
|
explicit Session(unsigned int num_last_files = 4,
|
|
unsigned int num_last_commands = 30);
|
|
/// Write the session file.
|
|
void writeFile() const;
|
|
///
|
|
LastFilesSection & lastFiles() { return last_files; }
|
|
///
|
|
LastFilesSection const & lastFiles() const { return last_files; }
|
|
///
|
|
LastOpenedSection & lastOpened() { return last_opened; }
|
|
///
|
|
LastOpenedSection const & lastOpened() const { return last_opened; }
|
|
///
|
|
LastFilePosSection & lastFilePos() { return last_file_pos; }
|
|
///
|
|
LastFilePosSection const & lastFilePos() const { return last_file_pos; }
|
|
///
|
|
BookmarksSection & bookmarks() { return bookmarks_; }
|
|
///
|
|
BookmarksSection const & bookmarks() const { return bookmarks_; }
|
|
///
|
|
LastCommandsSection & lastCommands() { return last_commands; }
|
|
///
|
|
LastCommandsSection const & lastCommands() const { return last_commands; }
|
|
///
|
|
AuthFilesSection & authFiles() { return auth_files; }
|
|
///
|
|
AuthFilesSection const & authFiles() const { return auth_files; }
|
|
///
|
|
ShellEscapeSection & shellescapeFiles() { return shellescape_files; }
|
|
///
|
|
ShellEscapeSection const & shellescapeFiles() const { return shellescape_files; }
|
|
|
|
private:
|
|
friend class LyX;
|
|
/// uncopiable
|
|
Session(Session const &);
|
|
void operator=(Session const &);
|
|
|
|
/// file to save session, determined in the constructor.
|
|
support::FileName session_file;
|
|
|
|
/** Read the session file.
|
|
Reads the #.lyx/session# at the beginning of the LyX session.
|
|
This will read the session file (usually #.lyx/session#).
|
|
@param file the file containing the session.
|
|
*/
|
|
void readFile();
|
|
|
|
///
|
|
LastFilesSection last_files;
|
|
///
|
|
LastOpenedSection last_opened;
|
|
///
|
|
LastFilePosSection last_file_pos;
|
|
///
|
|
BookmarksSection bookmarks_;
|
|
///
|
|
LastCommandsSection last_commands;
|
|
///
|
|
AuthFilesSection auth_files;
|
|
///
|
|
ShellEscapeSection shellescape_files;
|
|
};
|
|
|
|
/// This is a singleton class. Get the instance.
|
|
/// Implemented in LyX.cpp.
|
|
Session & theSession();
|
|
|
|
} // namespace lyx
|
|
|
|
#endif
|