githandler.cpp 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. #include "githandler.h"
  2. #include <QUrl>
  3. #include <QFileSystemWatcher>
  4. #include <QGuiApplication>
  5. #include <QClipboard>
  6. #include <QtConcurrentRun>
  7. #include <QDebug>
  8. #include <QDir>
  9. #include <qqml.h>
  10. #include <gitrepository.h>
  11. #include <gitbranch.h>
  12. #include <gitdiff.h>
  13. #include <commitmodel.h>
  14. #include <tagmodel.h>
  15. #include <git2.h>
  16. #include <colorhandler.h>
  17. #include <commitgraph.h>
  18. #include <graphpoint.h>
  19. #include <branchlistmodel.h>
  20. #include <taglistmodel.h>
  21. #include <gitconsole.h>
  22. #include <graphpoint.h>
  23. #include <settings.h>
  24. GitHandler::GitHandler() : QObject()
  25. ,m_repositories(new RepositoryModel(this))
  26. ,m_commits(nullptr)
  27. ,m_graph(nullptr)
  28. ,m_activeRepo(nullptr)
  29. ,m_activeDiff(nullptr)
  30. ,m_branchList(new BranchListModel(this))
  31. ,m_tagList(new TagListModel(this))
  32. ,m_activeRepoWatcher(new QFileSystemWatcher(this))
  33. ,m_console(new GitConsole(this))
  34. ,m_homePath(QUrl::fromLocalFile(QDir::homePath()))
  35. {
  36. git_libgit2_init();
  37. connect(&m_diffTask, &QFutureWatcher<GitDiff*>::finished, this, &GitHandler::onDiffReady);
  38. connect(&m_graphTask, &QFutureWatcher<CommitGraph*>::finished, this, &GitHandler::onGraphReady);
  39. connect(&m_graphTask, &QFutureWatcher<CommitGraph*>::started, this, &GitHandler::isBusyChanged);
  40. connect(&m_graphTask, &QFutureWatcher<CommitGraph*>::finished, this, &GitHandler::isBusyChanged);
  41. connect(&m_graphTask, &QFutureWatcher<CommitGraph*>::canceled, this, &GitHandler::isBusyChanged);
  42. connect(&m_graphTask, &QFutureWatcher<CommitGraph*>::paused, this, &GitHandler::isBusyChanged);
  43. connect(&m_graphTask, &QFutureWatcher<CommitGraph*>::resumed, this, &GitHandler::isBusyChanged);
  44. loadCachedRepos();
  45. }
  46. GitHandler::~GitHandler()
  47. {
  48. git_libgit2_shutdown();
  49. }
  50. void GitHandler::open(const QUrl &url, bool activate)
  51. {
  52. if(url.isLocalFile()) {
  53. open(url.toLocalFile(), activate);
  54. }
  55. }
  56. void GitHandler::open(const QString &path, bool activate)
  57. {
  58. qDebug() << "path" << path;
  59. git_buf root = {0, 0, 0};
  60. if(git_repository_discover(&root, path.toUtf8().data(), 0, NULL) != 0) {
  61. qDebug() << lastError();
  62. return;
  63. }
  64. QString rootStr = QString::fromUtf8(root.ptr, root.size);
  65. GitRepository* repo = new GitRepository(rootStr);
  66. if (!m_repositories->findByProperty("path", QVariant::fromValue(repo->path())).isNull()) {
  67. qDebug() << "Repository is already in list";
  68. delete repo;
  69. return;
  70. }
  71. Settings::instance()->add(repo);
  72. m_repositories->addRepository(repo);
  73. if (activate) {
  74. setActiveRepo(repo);
  75. }
  76. }
  77. void GitHandler::activateRepository(int i)
  78. {
  79. GitRepository* repo = m_repositories->at(i);
  80. setActiveRepo(repo);
  81. }
  82. void GitHandler::setActiveRepo(GitRepository* repo)
  83. {
  84. if (repo == nullptr || !repo->isValid()) {
  85. qDebug() << lastError();
  86. return;
  87. }
  88. if (m_activeRepo) {
  89. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readBranches);
  90. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readRemotes);
  91. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readTags);
  92. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, this, &GitHandler::updateModels);
  93. m_activeRepoWatcher->removePath(m_activeRepo->root());
  94. disconnect(m_activeRepo, &GitRepository::branchesChanged, this, &GitHandler::updateModels);
  95. }
  96. m_activeRepo = repo;
  97. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readBranches);
  98. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readRemotes);
  99. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readTags);
  100. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, this, &GitHandler::updateModels);
  101. connect(m_activeRepo, &GitRepository::branchesChanged, this, &GitHandler::updateModels);
  102. ColorHandler::instance().updateColors(m_activeRepo);
  103. // m_constantHead = m_activeRepo->head(); // TODO
  104. m_console->setRepository(m_activeRepo);
  105. updateModels();
  106. m_activeRepoWatcher->addPath(m_activeRepo->root());
  107. m_repositories->setActiveRepositoryIndex(m_repositories->indexOf(m_activeRepo));
  108. activeRepoChanged(m_activeRepo);
  109. Settings::instance()->saveLastRepo(m_activeRepo);
  110. }
  111. QString GitHandler::lastError() const
  112. {
  113. const git_error *e = giterr_last();
  114. if(e) {
  115. return QString("(%1): %2").arg(e->klass).arg(e->message);
  116. giterr_clear();
  117. }
  118. giterr_clear();
  119. return QString();
  120. }
  121. void GitHandler::diff(GitCommit* a, GitCommit* b)
  122. {
  123. if(!m_activeDiff.isNull()) {
  124. m_activeDiff->deleteLater();
  125. }
  126. Q_ASSERT_X(a->repository() == b->repository(), "GitHandler", "Cross repository diff requested");
  127. if(m_diffTask.isRunning()) {
  128. m_diffTask.cancel();
  129. }
  130. qDebug() << "Diff start thread: " << QThread::currentThreadId();
  131. QFuture<GitDiff*> future = QtConcurrent::run(&GitDiff::diff, a, b);
  132. m_diffTask.setFuture(future);
  133. }
  134. void GitHandler::diff()
  135. {
  136. if(!m_activeDiff.isNull()) {
  137. m_activeDiff->deleteLater();
  138. }
  139. GitCommit* commit = GitCommit::fromOid(activeRepo()->head());
  140. QFuture<GitDiff*> future = QtConcurrent::run(&GitDiff::diff, commit);
  141. m_diffTask.setFuture(future);
  142. }
  143. void GitHandler::diffReset()
  144. {
  145. delete m_activeDiff;
  146. }
  147. void GitHandler::onDiffReady()
  148. {
  149. setActiveDiff(m_diffTask.result());
  150. }
  151. void GitHandler::setActiveDiff(GitDiff* activeDiff)
  152. {
  153. if (m_activeDiff == activeDiff)
  154. return;
  155. m_activeDiff = activeDiff;
  156. emit activeDiffChanged(activeDiff);
  157. }
  158. GitDiff* GitHandler::activeDiff() const
  159. {
  160. return m_activeDiff.data();
  161. }
  162. void GitHandler::pull() const
  163. {
  164. // git_remote_fetch(m_activeRepo->remote)
  165. }
  166. void GitHandler::updateModels()
  167. {
  168. if(!m_activeRepo) {
  169. return;
  170. }
  171. BranchContainer &branches = m_activeRepo->branches();
  172. m_activeRepo->updateHead();
  173. m_graphTask.cancel();
  174. m_graphTask.setFuture(QtConcurrent::run(&GitHandler::updateGraph, m_activeRepo->head(), branches));
  175. m_branchList->reset(branches.values());
  176. m_tagList->reset(m_activeRepo->tags().values());
  177. }
  178. void GitHandler::copy(const QString& sha1)
  179. {
  180. QGuiApplication::clipboard()->setText(sha1);
  181. }
  182. CommitGraph* GitHandler::updateGraph(const GitOid &head, const BranchContainer &branches)
  183. {
  184. CommitGraph* graph = new CommitGraph();
  185. bool headIsBranch = false;
  186. //TODO: Need to think about constant head more deeply
  187. // foreach(GitBranch* branch, branches) {
  188. // if(branch->oid() == m_constantHead) {
  189. // graph->addHead(branch);
  190. // headIsBranch = true;
  191. // break;
  192. // }
  193. // }
  194. if(!headIsBranch) {
  195. graph->addHead(head);
  196. }
  197. foreach(GitBranch* branch, branches) {
  198. qDebug() << "Next head " << branch->fullName();
  199. graph->addHead(branch);
  200. }
  201. graph->addWorkdir();
  202. graph->moveToThread(head.repository()->thread());
  203. return graph;
  204. }
  205. void GitHandler::onGraphReady()
  206. {
  207. m_commits->deleteLater();
  208. m_commits = nullptr;
  209. emit commitsChanged(m_commits);
  210. m_graph->deleteLater();
  211. m_graph = nullptr;
  212. emit graphChanged(m_graph);
  213. m_graph = m_graphTask.result();
  214. m_commits = CommitModel::fromGraph(m_graph);
  215. emit graphChanged(m_graph);
  216. emit commitsChanged(m_commits);
  217. }
  218. void GitHandler::loadCachedRepos()
  219. {
  220. QString activeRepo = Settings::instance()->loadLastRepo();
  221. QStringList cachedRepos;
  222. Settings::instance()->load(cachedRepos);
  223. foreach (QString repoPath, cachedRepos) {
  224. open(repoPath, false);
  225. }
  226. QPointer<GitRepository> repo = m_repositories->findByProperty("id", QVariant::fromValue<QString>(activeRepo));
  227. setActiveRepo(repo);
  228. }