window-missing-files.cpp 16 KB

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