1
0

window-missing-files.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602
  1. /******************************************************************************
  2. Copyright (C) 2019 by Dillon Pentz <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. ******************************************************************************/
  14. #include "window-missing-files.hpp"
  15. #include "window-basic-main.hpp"
  16. #include "obs-app.hpp"
  17. #include <QLineEdit>
  18. #include <QToolButton>
  19. #include <QFileDialog>
  20. #include "qt-wrappers.hpp"
  21. enum MissingFilesColumn {
  22. Source,
  23. OriginalPath,
  24. NewPath,
  25. State,
  26. Count
  27. };
  28. enum MissingFilesRole { EntryStateRole = Qt::UserRole, NewPathsToProcessRole };
  29. /**********************************************************
  30. Delegate - Presents cells in the grid.
  31. **********************************************************/
  32. MissingFilesPathItemDelegate::MissingFilesPathItemDelegate(
  33. bool isOutput, const QString &defaultPath)
  34. : QStyledItemDelegate(),
  35. isOutput(isOutput),
  36. defaultPath(defaultPath)
  37. {
  38. }
  39. QWidget *MissingFilesPathItemDelegate::createEditor(
  40. QWidget *parent, const QStyleOptionViewItem & /* option */,
  41. const QModelIndex &) const
  42. {
  43. QSizePolicy buttonSizePolicy(QSizePolicy::Policy::Minimum,
  44. QSizePolicy::Policy::Expanding,
  45. QSizePolicy::ControlType::PushButton);
  46. QWidget *container = new QWidget(parent);
  47. auto browseCallback = [this, container]() {
  48. const_cast<MissingFilesPathItemDelegate *>(this)->handleBrowse(
  49. container);
  50. };
  51. auto clearCallback = [this, container]() {
  52. const_cast<MissingFilesPathItemDelegate *>(this)->handleClear(
  53. container);
  54. };
  55. QHBoxLayout *layout = new QHBoxLayout();
  56. layout->setContentsMargins(0, 0, 0, 0);
  57. layout->setSpacing(0);
  58. QLineEdit *text = new QLineEdit();
  59. text->setObjectName(QStringLiteral("text"));
  60. text->setSizePolicy(QSizePolicy(QSizePolicy::Policy::Expanding,
  61. QSizePolicy::Policy::Expanding,
  62. QSizePolicy::ControlType::LineEdit));
  63. layout->addWidget(text);
  64. QToolButton *browseButton = new QToolButton();
  65. browseButton->setText("...");
  66. browseButton->setSizePolicy(buttonSizePolicy);
  67. layout->addWidget(browseButton);
  68. container->connect(browseButton, &QToolButton::clicked, browseCallback);
  69. // The "clear" button is not shown in input cells
  70. if (isOutput) {
  71. QToolButton *clearButton = new QToolButton();
  72. clearButton->setText("X");
  73. clearButton->setSizePolicy(buttonSizePolicy);
  74. layout->addWidget(clearButton);
  75. container->connect(clearButton, &QToolButton::clicked,
  76. clearCallback);
  77. }
  78. container->setLayout(layout);
  79. container->setFocusProxy(text);
  80. return container;
  81. }
  82. void MissingFilesPathItemDelegate::setEditorData(QWidget *editor,
  83. const QModelIndex &index) const
  84. {
  85. QLineEdit *text = editor->findChild<QLineEdit *>();
  86. text->setText(index.data().toString());
  87. editor->setProperty(PATH_LIST_PROP, QVariant());
  88. }
  89. void MissingFilesPathItemDelegate::setModelData(QWidget *editor,
  90. QAbstractItemModel *model,
  91. const QModelIndex &index) const
  92. {
  93. // We use the PATH_LIST_PROP property to pass a list of
  94. // path strings from the editor widget into the model's
  95. // NewPathsToProcessRole. This is only used when paths
  96. // are selected through the "browse" or "delete" buttons
  97. // in the editor. If the user enters new text in the
  98. // text box, we simply pass that text on to the model
  99. // as normal text data in the default role.
  100. QVariant pathListProp = editor->property(PATH_LIST_PROP);
  101. if (pathListProp.isValid()) {
  102. QStringList list =
  103. editor->property(PATH_LIST_PROP).toStringList();
  104. if (isOutput) {
  105. model->setData(index, list);
  106. } else
  107. model->setData(index, list,
  108. MissingFilesRole::NewPathsToProcessRole);
  109. } else {
  110. QLineEdit *lineEdit = editor->findChild<QLineEdit *>();
  111. model->setData(index, lineEdit->text(), 0);
  112. }
  113. }
  114. void MissingFilesPathItemDelegate::paint(QPainter *painter,
  115. const QStyleOptionViewItem &option,
  116. const QModelIndex &index) const
  117. {
  118. QStyleOptionViewItem localOption = option;
  119. initStyleOption(&localOption, index);
  120. QApplication::style()->drawControl(QStyle::CE_ItemViewItem,
  121. &localOption, painter);
  122. }
  123. void MissingFilesPathItemDelegate::handleBrowse(QWidget *container)
  124. {
  125. QLineEdit *text = container->findChild<QLineEdit *>();
  126. QString currentPath = text->text();
  127. if (currentPath.isEmpty() ||
  128. currentPath.compare(QTStr("MissingFiles.Clear")) == 0)
  129. currentPath = defaultPath;
  130. bool isSet = false;
  131. if (isOutput) {
  132. QString newPath = QFileDialog::getOpenFileName(
  133. container, QTStr("MissingFiles.SelectFile"),
  134. currentPath, nullptr);
  135. #ifdef __APPLE__
  136. // TODO: Revisit when QTBUG-42661 is fixed
  137. container->window()->raise();
  138. #endif
  139. if (!newPath.isEmpty()) {
  140. container->setProperty(PATH_LIST_PROP,
  141. QStringList() << newPath);
  142. isSet = true;
  143. }
  144. }
  145. if (isSet)
  146. emit commitData(container);
  147. }
  148. void MissingFilesPathItemDelegate::handleClear(QWidget *container)
  149. {
  150. // An empty string list will indicate that the entry is being
  151. // blanked and should be deleted.
  152. container->setProperty(PATH_LIST_PROP,
  153. QStringList() << QTStr("MissingFiles.Clear"));
  154. container->findChild<QLineEdit *>()->clearFocus();
  155. ((QWidget *)container->parent())->setFocus();
  156. emit commitData(container);
  157. }
  158. /**
  159. Model
  160. **/
  161. MissingFilesModel::MissingFilesModel(QObject *parent)
  162. : QAbstractTableModel(parent)
  163. {
  164. QStyle *style = QApplication::style();
  165. warningIcon = style->standardIcon(QStyle::SP_MessageBoxWarning);
  166. }
  167. int MissingFilesModel::rowCount(const QModelIndex &) const
  168. {
  169. return files.length();
  170. }
  171. int MissingFilesModel::columnCount(const QModelIndex &) const
  172. {
  173. return MissingFilesColumn::Count;
  174. }
  175. int MissingFilesModel::found() const
  176. {
  177. int res = 0;
  178. for (int i = 0; i < files.length(); i++) {
  179. if (files[i].state != Missing && files[i].state != Cleared)
  180. res++;
  181. }
  182. return res;
  183. }
  184. QVariant MissingFilesModel::data(const QModelIndex &index, int role) const
  185. {
  186. QVariant result = QVariant();
  187. if (index.row() >= files.length()) {
  188. return QVariant();
  189. } else if (role == Qt::DisplayRole) {
  190. QFileInfo fi(files[index.row()].originalPath);
  191. switch (index.column()) {
  192. case MissingFilesColumn::Source:
  193. result = files[index.row()].source;
  194. break;
  195. case MissingFilesColumn::OriginalPath:
  196. result = fi.fileName();
  197. break;
  198. case MissingFilesColumn::NewPath:
  199. result = files[index.row()].newPath;
  200. break;
  201. case MissingFilesColumn::State:
  202. switch (files[index.row()].state) {
  203. case MissingFilesState::Missing:
  204. result = QTStr("MissingFiles.Missing");
  205. break;
  206. case MissingFilesState::Replaced:
  207. result = QTStr("MissingFiles.Replaced");
  208. break;
  209. case MissingFilesState::Found:
  210. result = QTStr("MissingFiles.Found");
  211. break;
  212. case MissingFilesState::Cleared:
  213. result = QTStr("MissingFiles.Cleared");
  214. break;
  215. }
  216. break;
  217. }
  218. } else if (role == Qt::DecorationRole &&
  219. index.column() == MissingFilesColumn::Source) {
  220. OBSBasic *main =
  221. reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
  222. OBSSourceAutoRelease source = obs_get_source_by_name(
  223. files[index.row()].source.toStdString().c_str());
  224. if (source) {
  225. result = main->GetSourceIcon(obs_source_get_id(source));
  226. }
  227. } else if (role == Qt::FontRole &&
  228. index.column() == MissingFilesColumn::State) {
  229. QFont font = QFont();
  230. font.setBold(true);
  231. result = font;
  232. } else if (role == Qt::ToolTipRole &&
  233. index.column() == MissingFilesColumn::State) {
  234. switch (files[index.row()].state) {
  235. case MissingFilesState::Missing:
  236. result = QTStr("MissingFiles.Missing");
  237. break;
  238. case MissingFilesState::Replaced:
  239. result = QTStr("MissingFiles.Replaced");
  240. break;
  241. case MissingFilesState::Found:
  242. result = QTStr("MissingFiles.Found");
  243. break;
  244. case MissingFilesState::Cleared:
  245. result = QTStr("MissingFiles.Cleared");
  246. break;
  247. default:
  248. break;
  249. }
  250. } else if (role == Qt::ToolTipRole) {
  251. switch (index.column()) {
  252. case MissingFilesColumn::OriginalPath:
  253. result = files[index.row()].originalPath;
  254. break;
  255. case MissingFilesColumn::NewPath:
  256. result = files[index.row()].newPath;
  257. break;
  258. default:
  259. break;
  260. }
  261. }
  262. return result;
  263. }
  264. Qt::ItemFlags MissingFilesModel::flags(const QModelIndex &index) const
  265. {
  266. Qt::ItemFlags flags = QAbstractTableModel::flags(index);
  267. if (index.column() == MissingFilesColumn::OriginalPath) {
  268. flags &= ~Qt::ItemIsEditable;
  269. } else if (index.column() == MissingFilesColumn::NewPath &&
  270. index.row() != files.length()) {
  271. flags |= Qt::ItemIsEditable;
  272. }
  273. return flags;
  274. }
  275. void MissingFilesModel::fileCheckLoop(QList<MissingFileEntry> files,
  276. QString path, bool skipPrompt)
  277. {
  278. loop = false;
  279. QUrl url = QUrl().fromLocalFile(path);
  280. QString dir =
  281. url.toDisplayString(QUrl::RemoveScheme | QUrl::RemoveFilename |
  282. QUrl::PreferLocalFile);
  283. bool prompted = skipPrompt;
  284. for (int i = 0; i < files.length(); i++) {
  285. if (files[i].state != MissingFilesState::Missing)
  286. continue;
  287. QUrl origFile = QUrl().fromLocalFile(files[i].originalPath);
  288. QString filename = origFile.fileName();
  289. QString testFile = dir + filename;
  290. if (os_file_exists(testFile.toStdString().c_str())) {
  291. if (!prompted) {
  292. QMessageBox::StandardButton button =
  293. QMessageBox::question(
  294. nullptr,
  295. QTStr("MissingFiles.AutoSearch"),
  296. QTStr("MissingFiles.AutoSearchText"));
  297. if (button == QMessageBox::No)
  298. break;
  299. prompted = true;
  300. }
  301. QModelIndex in = index(i, MissingFilesColumn::NewPath);
  302. setData(in, testFile, 0);
  303. }
  304. }
  305. loop = true;
  306. }
  307. bool MissingFilesModel::setData(const QModelIndex &index, const QVariant &value,
  308. int role)
  309. {
  310. bool success = false;
  311. if (role == MissingFilesRole::NewPathsToProcessRole) {
  312. QStringList list = value.toStringList();
  313. int row = index.row() + 1;
  314. beginInsertRows(QModelIndex(), row, row);
  315. MissingFileEntry entry;
  316. entry.originalPath = list[0].replace("\\", "/");
  317. entry.source = list[1];
  318. files.insert(row, entry);
  319. row++;
  320. endInsertRows();
  321. success = true;
  322. } else {
  323. QString path = value.toString();
  324. if (index.column() == MissingFilesColumn::NewPath) {
  325. files[index.row()].newPath = value.toString();
  326. QString fileName = QUrl(path).fileName();
  327. QString origFileName =
  328. QUrl(files[index.row()].originalPath).fileName();
  329. if (path.isEmpty()) {
  330. files[index.row()].state =
  331. MissingFilesState::Missing;
  332. } else if (path.compare(QTStr("MissingFiles.Clear")) ==
  333. 0) {
  334. files[index.row()].state =
  335. MissingFilesState::Cleared;
  336. } else if (fileName.compare(origFileName) == 0) {
  337. files[index.row()].state =
  338. MissingFilesState::Found;
  339. if (loop)
  340. fileCheckLoop(files, path, false);
  341. } else {
  342. files[index.row()].state =
  343. MissingFilesState::Replaced;
  344. if (loop)
  345. fileCheckLoop(files, path, false);
  346. }
  347. emit dataChanged(index, index);
  348. success = true;
  349. }
  350. }
  351. return success;
  352. }
  353. QVariant MissingFilesModel::headerData(int section, Qt::Orientation orientation,
  354. int role) const
  355. {
  356. QVariant result = QVariant();
  357. if (role == Qt::DisplayRole &&
  358. orientation == Qt::Orientation::Horizontal) {
  359. switch (section) {
  360. case MissingFilesColumn::State:
  361. result = QTStr("MissingFiles.State");
  362. break;
  363. case MissingFilesColumn::Source:
  364. result = QTStr("Basic.Main.Source");
  365. break;
  366. case MissingFilesColumn::OriginalPath:
  367. result = QTStr("MissingFiles.MissingFile");
  368. break;
  369. case MissingFilesColumn::NewPath:
  370. result = QTStr("MissingFiles.NewFile");
  371. break;
  372. }
  373. }
  374. return result;
  375. }
  376. OBSMissingFiles::OBSMissingFiles(obs_missing_files_t *files, QWidget *parent)
  377. : QDialog(parent),
  378. filesModel(new MissingFilesModel),
  379. ui(new Ui::OBSMissingFiles)
  380. {
  381. setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
  382. ui->setupUi(this);
  383. ui->tableView->setModel(filesModel);
  384. ui->tableView->setItemDelegateForColumn(
  385. MissingFilesColumn::OriginalPath,
  386. new MissingFilesPathItemDelegate(false, ""));
  387. ui->tableView->setItemDelegateForColumn(
  388. MissingFilesColumn::NewPath,
  389. new MissingFilesPathItemDelegate(true, ""));
  390. ui->tableView->horizontalHeader()->setSectionResizeMode(
  391. QHeaderView::ResizeMode::Stretch);
  392. ui->tableView->horizontalHeader()->setSectionResizeMode(
  393. MissingFilesColumn::Source,
  394. QHeaderView::ResizeMode::ResizeToContents);
  395. ui->tableView->horizontalHeader()->setMaximumSectionSize(width() / 3);
  396. ui->tableView->horizontalHeader()->setSectionResizeMode(
  397. MissingFilesColumn::State,
  398. QHeaderView::ResizeMode::ResizeToContents);
  399. ui->tableView->setEditTriggers(
  400. QAbstractItemView::EditTrigger::CurrentChanged);
  401. ui->warningIcon->setPixmap(
  402. filesModel->warningIcon.pixmap(QSize(32, 32)));
  403. for (size_t i = 0; i < obs_missing_files_count(files); i++) {
  404. obs_missing_file_t *f =
  405. obs_missing_files_get_file(files, (int)i);
  406. const char *oldPath = obs_missing_file_get_path(f);
  407. const char *name = obs_missing_file_get_source_name(f);
  408. addMissingFile(oldPath, name);
  409. }
  410. QString found =
  411. QTStr("MissingFiles.NumFound")
  412. .arg("0",
  413. QString::number(obs_missing_files_count(files)));
  414. ui->found->setText(found);
  415. fileStore = files;
  416. connect(ui->doneButton, &QPushButton::clicked, this,
  417. &OBSMissingFiles::saveFiles);
  418. connect(ui->browseButton, &QPushButton::clicked, this,
  419. &OBSMissingFiles::browseFolders);
  420. connect(ui->cancelButton, &QPushButton::clicked, this,
  421. &OBSMissingFiles::close);
  422. connect(filesModel, &MissingFilesModel::dataChanged, this,
  423. &OBSMissingFiles::dataChanged);
  424. QModelIndex index = filesModel->createIndex(0, 1);
  425. QMetaObject::invokeMethod(ui->tableView, "setCurrentIndex",
  426. Qt::QueuedConnection,
  427. Q_ARG(const QModelIndex &, index));
  428. }
  429. OBSMissingFiles::~OBSMissingFiles()
  430. {
  431. obs_missing_files_destroy(fileStore);
  432. }
  433. void OBSMissingFiles::addMissingFile(const char *originalPath,
  434. const char *sourceName)
  435. {
  436. QStringList list;
  437. list.append(originalPath);
  438. list.append(sourceName);
  439. QModelIndex insertIndex = filesModel->index(filesModel->rowCount() - 1,
  440. MissingFilesColumn::Source);
  441. filesModel->setData(insertIndex, list,
  442. MissingFilesRole::NewPathsToProcessRole);
  443. }
  444. void OBSMissingFiles::saveFiles()
  445. {
  446. for (int i = 0; i < filesModel->files.length(); i++) {
  447. MissingFilesState state = filesModel->files[i].state;
  448. if (state != MissingFilesState::Missing) {
  449. obs_missing_file_t *f =
  450. obs_missing_files_get_file(fileStore, i);
  451. QString path = filesModel->files[i].newPath;
  452. if (state == MissingFilesState::Cleared) {
  453. obs_missing_file_issue_callback(f, "");
  454. } else {
  455. char *p = bstrdup(path.toStdString().c_str());
  456. obs_missing_file_issue_callback(f, p);
  457. bfree(p);
  458. }
  459. }
  460. }
  461. QDialog::accept();
  462. }
  463. void OBSMissingFiles::browseFolders()
  464. {
  465. QString dir = QFileDialog::getExistingDirectory(
  466. this, QTStr("MissingFiles.SelectDir"), "",
  467. QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks);
  468. if (dir != "") {
  469. dir += "/";
  470. filesModel->fileCheckLoop(filesModel->files, dir, true);
  471. }
  472. }
  473. void OBSMissingFiles::dataChanged()
  474. {
  475. QString found = QTStr("MissingFiles.NumFound")
  476. .arg(QString::number(filesModel->found()),
  477. QString::number(obs_missing_files_count(
  478. fileStore)));
  479. ui->found->setText(found);
  480. ui->tableView->resizeColumnToContents(MissingFilesColumn::State);
  481. ui->tableView->resizeColumnToContents(MissingFilesColumn::Source);
  482. }
  483. QIcon OBSMissingFiles::GetWarningIcon()
  484. {
  485. return filesModel->warningIcon;
  486. }
  487. void OBSMissingFiles::SetWarningIcon(const QIcon &icon)
  488. {
  489. ui->warningIcon->setPixmap(icon.pixmap(QSize(32, 32)));
  490. filesModel->warningIcon = icon;
  491. }