githandler.cpp 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  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. void GitHandler::open(const QUrl &url, bool activate)
  49. {
  50. if(url.isLocalFile()) {
  51. open(url.toLocalFile(), activate);
  52. }
  53. }
  54. void GitHandler::open(const QString &path, bool activate)
  55. {
  56. qDebug() << "path" << path;
  57. git_buf root = {0,0,0};
  58. if(git_repository_discover(&root, path.toUtf8().data(), 0, NULL) != 0) {
  59. qDebug() << lastError();
  60. return;
  61. }
  62. GitRepository* repo = new GitRepository(QString::fromUtf8(root.ptr, root.size));
  63. Settings::instance()->add(repo);
  64. m_repositories->addRepository(repo);
  65. if(activate) {
  66. setActiveRepo(repo);
  67. }
  68. }
  69. void GitHandler::activateRepository(int i)
  70. {
  71. GitRepository* repo = m_repositories->at(i);
  72. setActiveRepo(repo);
  73. }
  74. void GitHandler::setActiveRepo(GitRepository* repo)
  75. {
  76. if(repo == nullptr || !repo->isValid()) {
  77. qDebug() << lastError();
  78. return;
  79. }
  80. if(m_activeRepo) {
  81. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readBranches);
  82. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readRemotes);
  83. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readTags);
  84. disconnect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, this, &GitHandler::updateModels);
  85. m_activeRepoWatcher->removePath(m_activeRepo->root());
  86. disconnect(m_activeRepo, &GitRepository::branchesChanged, this, &GitHandler::updateModels);
  87. }
  88. m_activeRepo = repo;
  89. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readBranches);
  90. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readRemotes);
  91. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, m_activeRepo, &GitRepository::readTags);
  92. connect(m_activeRepoWatcher, &QFileSystemWatcher::directoryChanged, this, &GitHandler::updateModels);
  93. connect(m_activeRepo, &GitRepository::branchesChanged, this, &GitHandler::updateModels);
  94. ColorHandler::instance().updateColors(m_activeRepo);
  95. // m_constantHead = m_activeRepo->head(); // TODO
  96. m_console->setRepository(m_activeRepo);
  97. updateModels();
  98. m_activeRepoWatcher->addPath(m_activeRepo->root());
  99. m_repositories->setActiveRepositoryIndex(m_repositories->indexOf(m_activeRepo));
  100. activeRepoChanged(m_activeRepo);
  101. Settings::instance()->saveLastRepo(m_activeRepo);
  102. }
  103. QString GitHandler::lastError() const
  104. {
  105. const git_error *e = giterr_last();
  106. if(e) {
  107. return QString("(%1): %2").arg(e->klass).arg(e->message);
  108. giterr_clear();
  109. }
  110. giterr_clear();
  111. return QString();
  112. }
  113. void GitHandler::diff(GitCommit* a, GitCommit* b)
  114. {
  115. if(!m_activeDiff.isNull()) {
  116. m_activeDiff->deleteLater();
  117. }
  118. Q_ASSERT_X(a->repository() == b->repository(), "GitHandler", "Cross repository diff requested");
  119. if(m_diffTask.isRunning()) {
  120. m_diffTask.cancel();
  121. }
  122. qDebug() << "Diff start thread: " << QThread::currentThreadId();
  123. QFuture<GitDiff*> future = QtConcurrent::run(&GitDiff::diff, a, b);
  124. m_diffTask.setFuture(future);
  125. }
  126. void GitHandler::diff()
  127. {
  128. if(!m_activeDiff.isNull()) {
  129. m_activeDiff->deleteLater();
  130. }
  131. GitCommit* commit = GitCommit::fromOid(activeRepo()->head());
  132. QFuture<GitDiff*> future = QtConcurrent::run(&GitDiff::diff, commit);
  133. m_diffTask.setFuture(future);
  134. }
  135. void GitHandler::diffReset()
  136. {
  137. delete m_activeDiff;
  138. }
  139. void GitHandler::onDiffReady()
  140. {
  141. setActiveDiff(m_diffTask.result());
  142. }
  143. void GitHandler::setActiveDiff(GitDiff* activeDiff)
  144. {
  145. if (m_activeDiff == activeDiff)
  146. return;
  147. m_activeDiff = activeDiff;
  148. emit activeDiffChanged(activeDiff);
  149. }
  150. GitDiff* GitHandler::activeDiff() const
  151. {
  152. return m_activeDiff.data();
  153. }
  154. void GitHandler::pull() const
  155. {
  156. // git_remote_fetch(m_activeRepo->remote)
  157. }
  158. void GitHandler::updateModels()
  159. {
  160. if(!m_activeRepo) {
  161. return;
  162. }
  163. BranchContainer &branches = m_activeRepo->branches();
  164. m_activeRepo->updateHead();
  165. m_graphTask.cancel();
  166. m_graphTask.setFuture(QtConcurrent::run(&GitHandler::updateGraph, m_activeRepo->head(), branches));
  167. m_branchList->reset(branches.values());
  168. m_tagList->reset(m_activeRepo->tags().values());
  169. }
  170. void GitHandler::copy(const QString& sha1)
  171. {
  172. QGuiApplication::clipboard()->setText(sha1);
  173. }
  174. CommitGraph* GitHandler::updateGraph(const GitOid &head, const BranchContainer &branches)
  175. {
  176. CommitGraph* graph = new CommitGraph();
  177. bool headIsBranch = false;
  178. //TODO: Need to think about constant head more deeply
  179. // foreach(GitBranch* branch, branches) {
  180. // if(branch->oid() == m_constantHead) {
  181. // graph->addHead(branch);
  182. // headIsBranch = true;
  183. // break;
  184. // }
  185. // }
  186. if(!headIsBranch) {
  187. graph->addHead(head);
  188. }
  189. foreach(GitBranch* branch, branches) {
  190. qDebug() << "Next head " << branch->fullName();
  191. graph->addHead(branch);
  192. }
  193. graph->addWorkdir();
  194. graph->moveToThread(head.repository()->thread());
  195. return graph;
  196. }
  197. void GitHandler::onGraphReady()
  198. {
  199. m_commits->deleteLater();
  200. m_commits = nullptr;
  201. emit commitsChanged(m_commits);
  202. m_graph->deleteLater();
  203. m_graph = nullptr;
  204. emit graphChanged(m_graph);
  205. m_graph = m_graphTask.result();
  206. m_commits = CommitModel::fromGraph(m_graph);
  207. emit graphChanged(m_graph);
  208. emit commitsChanged(m_commits);
  209. }
  210. void GitHandler::loadCachedRepos()
  211. {
  212. QString activeRepo = Settings::instance()->loadLastRepo();
  213. QStringList cachedRepos;
  214. Settings::instance()->load(cachedRepos);
  215. foreach (QString repoPath, cachedRepos) {
  216. open(repoPath, false);
  217. }
  218. // m_repositories->ac(m_repositories->indexOf(m_activeRepo));
  219. }