githandler.cpp 7.5 KB

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