mirror of
https://github.com/nomic-ai/gpt4all.git
synced 2024-10-01 01:06:10 -04:00
fbbf810020
* select the existing new chat if there already is one when "New Chat" is clicked * scroll to the new chat when "New Chat" is clicked * fix the "New Chat" being scrolled past the top of the chat list Signed-off-by: Jared Van Bortel <jared@nomic.ai>
265 lines
8.1 KiB
C++
265 lines
8.1 KiB
C++
#include "chatlistmodel.h"
|
|
#include "mysettings.h"
|
|
|
|
#include <QFile>
|
|
#include <QDataStream>
|
|
|
|
#define CHAT_FORMAT_MAGIC 0xF5D553CC
|
|
#define CHAT_FORMAT_VERSION 7
|
|
|
|
class MyChatListModel: public ChatListModel { };
|
|
Q_GLOBAL_STATIC(MyChatListModel, chatListModelInstance)
|
|
ChatListModel *ChatListModel::globalInstance()
|
|
{
|
|
return chatListModelInstance();
|
|
}
|
|
|
|
ChatListModel::ChatListModel()
|
|
: QAbstractListModel(nullptr) {}
|
|
|
|
void ChatListModel::loadChats()
|
|
{
|
|
addChat();
|
|
|
|
ChatsRestoreThread *thread = new ChatsRestoreThread;
|
|
connect(thread, &ChatsRestoreThread::chatRestored, this, &ChatListModel::restoreChat, Qt::QueuedConnection);
|
|
connect(thread, &ChatsRestoreThread::finished, this, &ChatListModel::chatsRestoredFinished, Qt::QueuedConnection);
|
|
connect(thread, &ChatsRestoreThread::finished, thread, &QObject::deleteLater);
|
|
thread->start();
|
|
|
|
connect(MySettings::globalInstance(), &MySettings::serverChatChanged, this, &ChatListModel::handleServerEnabledChanged);
|
|
}
|
|
|
|
void ChatListModel::removeChatFile(Chat *chat) const
|
|
{
|
|
Q_ASSERT(chat != m_serverChat);
|
|
const QString savePath = MySettings::globalInstance()->modelPath();
|
|
QFile file(savePath + "/gpt4all-" + chat->id() + ".chat");
|
|
if (!file.exists())
|
|
return;
|
|
bool success = file.remove();
|
|
if (!success)
|
|
qWarning() << "ERROR: Couldn't remove chat file:" << file.fileName();
|
|
}
|
|
|
|
ChatSaver::ChatSaver()
|
|
: QObject(nullptr)
|
|
{
|
|
moveToThread(&m_thread);
|
|
m_thread.start();
|
|
}
|
|
|
|
void ChatListModel::saveChats()
|
|
{
|
|
const QString savePath = MySettings::globalInstance()->modelPath();
|
|
QVector<Chat*> toSave;
|
|
for (Chat *chat : m_chats) {
|
|
if (chat == m_serverChat)
|
|
continue;
|
|
if (chat->isNewChat())
|
|
continue;
|
|
toSave.append(chat);
|
|
}
|
|
if (toSave.isEmpty()) {
|
|
emit saveChatsFinished();
|
|
return;
|
|
}
|
|
|
|
ChatSaver *saver = new ChatSaver;
|
|
connect(this, &ChatListModel::requestSaveChats, saver, &ChatSaver::saveChats, Qt::QueuedConnection);
|
|
connect(saver, &ChatSaver::saveChatsFinished, this, &ChatListModel::saveChatsFinished, Qt::QueuedConnection);
|
|
emit requestSaveChats(toSave);
|
|
}
|
|
|
|
void ChatSaver::saveChats(const QVector<Chat *> &chats)
|
|
{
|
|
QElapsedTimer timer;
|
|
timer.start();
|
|
const QString savePath = MySettings::globalInstance()->modelPath();
|
|
for (Chat *chat : chats) {
|
|
QString fileName = "gpt4all-" + chat->id() + ".chat";
|
|
QString filePath = savePath + "/" + fileName;
|
|
QFile originalFile(filePath);
|
|
QFile tempFile(filePath + ".tmp"); // Temporary file
|
|
|
|
bool success = tempFile.open(QIODevice::WriteOnly);
|
|
if (!success) {
|
|
qWarning() << "ERROR: Couldn't save chat to temporary file:" << tempFile.fileName();
|
|
continue;
|
|
}
|
|
QDataStream out(&tempFile);
|
|
|
|
out << (quint32)CHAT_FORMAT_MAGIC;
|
|
out << (qint32)CHAT_FORMAT_VERSION;
|
|
out.setVersion(QDataStream::Qt_6_2);
|
|
|
|
qDebug() << "serializing chat" << fileName;
|
|
if (!chat->serialize(out, CHAT_FORMAT_VERSION)) {
|
|
qWarning() << "ERROR: Couldn't serialize chat to file:" << tempFile.fileName();
|
|
tempFile.remove();
|
|
continue;
|
|
}
|
|
|
|
if (originalFile.exists())
|
|
originalFile.remove();
|
|
tempFile.rename(filePath);
|
|
}
|
|
|
|
qint64 elapsedTime = timer.elapsed();
|
|
qDebug() << "serializing chats took:" << elapsedTime << "ms";
|
|
emit saveChatsFinished();
|
|
}
|
|
|
|
void ChatsRestoreThread::run()
|
|
{
|
|
QElapsedTimer timer;
|
|
timer.start();
|
|
struct FileInfo {
|
|
bool oldFile;
|
|
qint64 creationDate;
|
|
QString file;
|
|
};
|
|
QList<FileInfo> files;
|
|
{
|
|
// Look for any files in the original spot which was the settings config directory
|
|
QSettings settings;
|
|
QFileInfo settingsInfo(settings.fileName());
|
|
QString settingsPath = settingsInfo.absolutePath();
|
|
QDir dir(settingsPath);
|
|
dir.setNameFilters(QStringList() << "gpt4all-*.chat");
|
|
QStringList fileNames = dir.entryList();
|
|
for (const QString &f : fileNames) {
|
|
QString filePath = settingsPath + "/" + f;
|
|
QFile file(filePath);
|
|
bool success = file.open(QIODevice::ReadOnly);
|
|
if (!success) {
|
|
qWarning() << "ERROR: Couldn't restore chat from file:" << file.fileName();
|
|
continue;
|
|
}
|
|
QDataStream in(&file);
|
|
FileInfo info;
|
|
info.oldFile = true;
|
|
info.file = filePath;
|
|
in >> info.creationDate;
|
|
files.append(info);
|
|
file.close();
|
|
}
|
|
}
|
|
{
|
|
const QString savePath = MySettings::globalInstance()->modelPath();
|
|
QDir dir(savePath);
|
|
dir.setNameFilters(QStringList() << "gpt4all-*.chat");
|
|
QStringList fileNames = dir.entryList();
|
|
for (const QString &f : fileNames) {
|
|
QString filePath = savePath + "/" + f;
|
|
QFile file(filePath);
|
|
bool success = file.open(QIODevice::ReadOnly);
|
|
if (!success) {
|
|
qWarning() << "ERROR: Couldn't restore chat from file:" << file.fileName();
|
|
continue;
|
|
}
|
|
QDataStream in(&file);
|
|
// Read and check the header
|
|
quint32 magic;
|
|
in >> magic;
|
|
if (magic != CHAT_FORMAT_MAGIC) {
|
|
qWarning() << "ERROR: Chat file has bad magic:" << file.fileName();
|
|
continue;
|
|
}
|
|
|
|
// Read the version
|
|
qint32 version;
|
|
in >> version;
|
|
if (version < 1) {
|
|
qWarning() << "ERROR: Chat file has non supported version:" << file.fileName();
|
|
continue;
|
|
}
|
|
|
|
if (version <= 1)
|
|
in.setVersion(QDataStream::Qt_6_2);
|
|
|
|
FileInfo info;
|
|
info.oldFile = false;
|
|
info.file = filePath;
|
|
in >> info.creationDate;
|
|
files.append(info);
|
|
file.close();
|
|
}
|
|
}
|
|
std::sort(files.begin(), files.end(), [](const FileInfo &a, const FileInfo &b) {
|
|
return a.creationDate > b.creationDate;
|
|
});
|
|
|
|
for (FileInfo &f : files) {
|
|
QFile file(f.file);
|
|
bool success = file.open(QIODevice::ReadOnly);
|
|
if (!success) {
|
|
qWarning() << "ERROR: Couldn't restore chat from file:" << file.fileName();
|
|
continue;
|
|
}
|
|
QDataStream in(&file);
|
|
|
|
qint32 version = 0;
|
|
if (!f.oldFile) {
|
|
// Read and check the header
|
|
quint32 magic;
|
|
in >> magic;
|
|
if (magic != CHAT_FORMAT_MAGIC) {
|
|
qWarning() << "ERROR: Chat file has bad magic:" << file.fileName();
|
|
continue;
|
|
}
|
|
|
|
// Read the version
|
|
in >> version;
|
|
if (version < 1) {
|
|
qWarning() << "ERROR: Chat file has non supported version:" << file.fileName();
|
|
continue;
|
|
}
|
|
|
|
if (version <= 1)
|
|
in.setVersion(QDataStream::Qt_6_2);
|
|
}
|
|
|
|
qDebug() << "deserializing chat" << f.file;
|
|
|
|
Chat *chat = new Chat;
|
|
chat->moveToThread(qApp->thread());
|
|
if (!chat->deserialize(in, version)) {
|
|
qWarning() << "ERROR: Couldn't deserialize chat from file:" << file.fileName();
|
|
} else {
|
|
emit chatRestored(chat);
|
|
}
|
|
if (f.oldFile)
|
|
file.remove(); // No longer storing in this directory
|
|
file.close();
|
|
}
|
|
|
|
qint64 elapsedTime = timer.elapsed();
|
|
qDebug() << "deserializing chats took:" << elapsedTime << "ms";
|
|
}
|
|
|
|
void ChatListModel::restoreChat(Chat *chat)
|
|
{
|
|
chat->setParent(this);
|
|
connect(chat, &Chat::nameChanged, this, &ChatListModel::nameChanged);
|
|
|
|
beginInsertRows(QModelIndex(), m_chats.size(), m_chats.size());
|
|
m_chats.append(chat);
|
|
endInsertRows();
|
|
}
|
|
|
|
void ChatListModel::chatsRestoredFinished()
|
|
{
|
|
addServerChat();
|
|
}
|
|
|
|
void ChatListModel::handleServerEnabledChanged()
|
|
{
|
|
if (MySettings::globalInstance()->serverChat() || m_serverChat != m_currentChat)
|
|
return;
|
|
|
|
Chat *nextChat = get(0);
|
|
Q_ASSERT(nextChat && nextChat != m_serverChat);
|
|
setCurrentChat(nextChat);
|
|
}
|