githandler.cpp 7.5 KB

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