• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

IJHack / QtPass / 24615779570

18 Apr 2026 10:56PM UTC coverage: 22.627%. First build
24615779570

Pull #1046

github

web-flow
Merge 1ee3cfbd5 into 3f2e420e3
Pull Request #1046: fix: pass QModelIndex by const ref, reorder MIME check, fix camelCase typo

5 of 6 new or added lines in 2 files covered. (83.33%)

1304 of 5763 relevant lines covered (22.63%)

8.53 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

53.78
/src/storemodel.cpp
1
// SPDX-FileCopyrightText: 2014 Anne Jan Brouwer
2
// SPDX-License-Identifier: GPL-3.0-or-later
3
#include "storemodel.h"
4
#include "qtpasssettings.h"
5

6
#include "util.h"
7
#include <QDebug>
8
#include <QFileSystemModel>
9
#include <QMessageBox>
10
#include <QMimeData>
11
#include <QRegularExpression>
12
#include <QtGlobal>
13
#include <utility>
14

15
auto operator<<(
1✔
16
    QDataStream &out,
17
    const dragAndDropInfoPasswordStore &dragAndDropInfoPasswordStore)
18
    -> QDataStream & {
19
  out << dragAndDropInfoPasswordStore.isDir
1✔
20
      << dragAndDropInfoPasswordStore.isFile
1✔
21
      << dragAndDropInfoPasswordStore.path;
1✔
22
  return out;
1✔
23
}
24

25
auto operator>>(QDataStream &in,
×
26
                dragAndDropInfoPasswordStore &dragAndDropInfoPasswordStore)
27
    -> QDataStream & {
28
  in >> dragAndDropInfoPasswordStore.isDir >>
×
29
      dragAndDropInfoPasswordStore.isFile >> dragAndDropInfoPasswordStore.path;
×
30
  return in;
×
31
}
32

33
/**
34
 * @brief StoreModel::StoreModel
35
 * SubClass of QSortFilterProxyModel via
36
 * http://www.qtcentre.org/threads/46471-QTreeView-Filter
37
 */
38
StoreModel::StoreModel() { fs = nullptr; }
19✔
39

40
/**
41
 * @brief StoreModel::filterAcceptsRow should row be shown, wrapper for
42
 * StoreModel::showThis method.
43
 * @param sourceRow
44
 * @param sourceParent
45
 * @return
46
 */
47
auto StoreModel::filterAcceptsRow(int sourceRow,
19✔
48
                                  const QModelIndex &sourceParent) const
49
    -> bool {
50
  QModelIndex index = sourceModel()->index(sourceRow, 0, sourceParent);
19✔
51
  return showThis(index);
19✔
52
}
53

54
/**
55
 * @brief StoreModel::showThis should a row be shown, based on our search
56
 * criteria.
57
 * @param index
58
 * @return
59
 */
60
auto StoreModel::showThis(const QModelIndex &index) const -> bool {
44✔
61
  bool retVal = false;
62
  if (fs == nullptr) {
44✔
63
    return retVal;
64
  }
65
  // Gives you the info for number of childs with a parent
66
  if (sourceModel()->rowCount(index) > 0) {
43✔
67
    for (int nChild = 0; nChild < sourceModel()->rowCount(index); ++nChild) {
24✔
68
      QModelIndex childIndex = sourceModel()->index(nChild, 0, index);
24✔
69
      if (!childIndex.isValid()) {
70
        break;
71
      }
72
      retVal = showThis(childIndex);
24✔
73
      if (retVal) {
24✔
74
        break;
75
      }
76
    }
77
  } else {
78
    QModelIndex useIndex = sourceModel()->index(index.row(), 0, index.parent());
19✔
79
    QString path = fs->filePath(useIndex);
19✔
80
    path = QDir(store).relativeFilePath(path);
38✔
81
    if (path.startsWith(".git")) {
38✔
82
      return false;
83
    }
84
    path.replace(Util::endsWithGpg(), "");
18✔
85
    retVal = path.contains(filterRegularExpression());
18✔
86
  }
87
  return retVal;
88
}
89

90
/**
91
 * @brief StoreModel::setModelAndStore update the source model and store.
92
 * @param sourceModel
93
 * @param passStore
94
 */
95
void StoreModel::setModelAndStore(QFileSystemModel *sourceModel,
14✔
96
                                  QString passStore) {
97
  setSourceModel(sourceModel);
14✔
98
  fs = sourceModel;
14✔
99
  store = std::move(passStore);
100
}
14✔
101

102
void StoreModel::setStore(const QString &passStore) {
×
103
  store = passStore;
×
104
#if QT_VERSION >= QT_VERSION_CHECK(6, 9, 0)
105
  beginFilterChange();
106
  endFilterChange();
107
#else
108
  invalidateFilter();
×
109
#endif
110
}
×
111

112
/**
113
 * @brief StoreModel::data don't show the .gpg at the end of a file.
114
 * @param index
115
 * @param role
116
 * @return
117
 */
118
auto StoreModel::data(const QModelIndex &index, int role) const -> QVariant {
2✔
119
  if (!index.isValid()) {
120
    return {};
121
  }
122

123
  QVariant initial_value;
124
  initial_value = QSortFilterProxyModel::data(index, role);
1✔
125

126
  if (role == Qt::DisplayRole) {
1✔
127
    QString name = initial_value.toString();
1✔
128
    name.replace(Util::endsWithGpg(), "");
1✔
129
    initial_value.setValue(name);
1✔
130
  }
131

132
  return initial_value;
1✔
133
}
1✔
134

135
/**
136
 * @brief StoreModel::supportedDropActions enable drop.
137
 * @return
138
 */
139
auto StoreModel::supportedDropActions() const -> Qt::DropActions {
1✔
140
  return Qt::CopyAction | Qt::MoveAction;
1✔
141
}
142

143
/**
144
 * @brief StoreModel::supportedDragActions enable drag.
145
 * @return
146
 */
147
auto StoreModel::supportedDragActions() const -> Qt::DropActions {
1✔
148
  return Qt::CopyAction | Qt::MoveAction;
1✔
149
}
150

151
/**
152
 * @brief StoreModel::flags
153
 * @param index
154
 * @return
155
 */
156
auto StoreModel::flags(const QModelIndex &index) const -> Qt::ItemFlags {
2✔
157
  Qt::ItemFlags defaultFlags = QSortFilterProxyModel::flags(index);
2✔
158

159
  if (index.isValid()) {
160
    return Qt::ItemIsDragEnabled | Qt::ItemIsDropEnabled | defaultFlags;
161
  }
162
  return Qt::ItemIsDropEnabled | defaultFlags;
1✔
163
}
164

165
/**
166
 * @brief StoreModel::mimeTypes
167
 * @return
168
 */
169
auto StoreModel::mimeTypes() const -> QStringList {
1✔
170
  QStringList types;
1✔
171
  types << "application/vnd+qtpass.dragAndDropInfoPasswordStore";
1✔
172
  return types;
1✔
173
}
174

175
/**
176
 * @brief StoreModel::mimeData
177
 * @param indexes
178
 * @return
179
 */
180
auto StoreModel::mimeData(const QModelIndexList &indexes) const -> QMimeData * {
1✔
181
  dragAndDropInfoPasswordStore info;
182

183
  QByteArray encodedData;
1✔
184
  // only use the first, otherwise we should enable multiselection
185
  QModelIndex index = indexes.at(0);
1✔
186
  if (index.isValid()) {
187
    QModelIndex useIndex = mapToSource(index);
1✔
188

189
    info.isDir = fs->fileInfo(useIndex).isDir();
1✔
190
    info.isFile = fs->fileInfo(useIndex).isFile();
1✔
191
    info.path = fs->fileInfo(useIndex).absoluteFilePath();
2✔
192
    QDataStream stream(&encodedData, QIODevice::WriteOnly);
1✔
193
    stream << info;
1✔
194
  }
1✔
195

196
  auto *mimeData = new QMimeData();
1✔
197
  mimeData->setData("application/vnd+qtpass.dragAndDropInfoPasswordStore",
2✔
198
                    encodedData);
199
  return mimeData;
1✔
200
}
201

202
/**
203
 * @brief StoreModel::canDropMimeData
204
 * @param data
205
 * @param action
206
 * @param row
207
 * @param column
208
 * @param parent
209
 * @return
210
 */
211
auto StoreModel::canDropMimeData(const QMimeData *data, Qt::DropAction action,
×
212
                                 int row, int column,
213
                                 const QModelIndex &parent) const -> bool {
214
#ifdef QT_DEBUG
215
  qDebug() << action << row;
216
#else
217
  Q_UNUSED(action)
218
  Q_UNUSED(row)
219
#endif
220

NEW
221
  if (!data->hasFormat("application/vnd+qtpass.dragAndDropInfoPasswordStore")) {
×
222
    return false;
223
  }
224

225
  QModelIndex useIndex =
226
      this->index(parent.row(), parent.column(), parent.parent());
×
227
  QByteArray encodedData =
228
      data->data("application/vnd+qtpass.dragAndDropInfoPasswordStore");
×
229
  QDataStream stream(&encodedData, QIODevice::ReadOnly);
×
230
  dragAndDropInfoPasswordStore info;
231
  stream >> info;
×
232

233
  if (column > 0) {
×
234
    return false;
235
  }
236

237
  // you can drop a folder on a folder
238
  if (fs->fileInfo(mapToSource(useIndex)).isDir() && info.isDir) {
×
239
    return true;
240
  }
241
  // you can drop a file on a folder
242
  if (fs->fileInfo(mapToSource(useIndex)).isDir() && info.isFile) {
×
243
    return true;
244
  }
245
  // you can drop a file on a file
246
  if (fs->fileInfo(mapToSource(useIndex)).isFile() && info.isFile) {
×
247
    return true;
248
  }
249

250
  return false;
251
}
×
252

253
/**
254
 * @brief StoreModel::dropMimeData
255
 * @param data
256
 * @param action
257
 * @param row
258
 * @param column
259
 * @param parent
260
 * @return
261
 */
262
auto StoreModel::dropMimeData(const QMimeData *data, Qt::DropAction action,
×
263
                              int row, int column, const QModelIndex &parent)
264
    -> bool {
265
  if (!canDropMimeData(data, action, row, column, parent)) {
×
266
    return false;
267
  }
268

269
  if (action == Qt::IgnoreAction) {
×
270
    return true;
271
  }
272
  QByteArray encodedData =
273
      data->data("application/vnd+qtpass.dragAndDropInfoPasswordStore");
×
274

275
  QDataStream stream(&encodedData, QIODevice::ReadOnly);
×
276
  dragAndDropInfoPasswordStore info;
277
  stream >> info;
×
278
  QModelIndex destIndex =
279
      this->index(parent.row(), parent.column(), parent.parent());
×
280
  QFileInfo destFileinfo = fs->fileInfo(mapToSource(destIndex));
×
281
  QFileInfo srcFileInfo = QFileInfo(info.path);
×
282
  QString cleanedSrc = QDir::cleanPath(srcFileInfo.absoluteFilePath());
×
283
  QString cleanedDest = QDir::cleanPath(destFileinfo.absoluteFilePath());
×
284
  if (info.isDir) {
×
285
    // dropped dir onto dir
286
    if (destFileinfo.isDir()) {
×
287
      QDir destDir = QDir(cleanedDest).filePath(srcFileInfo.fileName());
×
288
      QString cleanedDestDir = QDir::cleanPath(destDir.absolutePath());
×
289
      if (action == Qt::MoveAction) {
×
290
        QtPassSettings::getPass()->Move(cleanedSrc, cleanedDestDir);
×
291
      } else if (action == Qt::CopyAction) {
×
292
        QtPassSettings::getPass()->Copy(cleanedSrc, cleanedDestDir);
×
293
      }
294
    }
×
295
  } else if (info.isFile) {
×
296
    // dropped file onto a directory
297
    if (destFileinfo.isDir()) {
×
298
      if (action == Qt::MoveAction) {
×
299
        QtPassSettings::getPass()->Move(cleanedSrc, cleanedDest);
×
300
      } else if (action == Qt::CopyAction) {
×
301
        QtPassSettings::getPass()->Copy(cleanedSrc, cleanedDest);
×
302
      }
303
    } else if (destFileinfo.isFile()) {
×
304
      // dropped file onto a file
305
      int answer = QMessageBox::question(
×
306
          nullptr, tr("force overwrite?"),
×
307
          tr("overwrite %1 with %2?").arg(cleanedDest, cleanedSrc),
×
308
          QMessageBox::Yes | QMessageBox::No);
309
      bool force = answer == QMessageBox::Yes;
×
310
      if (action == Qt::MoveAction) {
×
311
        QtPassSettings::getPass()->Move(cleanedSrc, cleanedDest, force);
×
312
      } else if (action == Qt::CopyAction) {
×
313
        QtPassSettings::getPass()->Copy(cleanedSrc, cleanedDest, force);
×
314
      }
315
    }
316
  }
317
  return true;
318
}
×
319

320
/**
321
 * @brief StoreModel::lessThan
322
 * @param source_left
323
 * @param source_right
324
 * @return
325
 */
326
auto StoreModel::lessThan(const QModelIndex &source_left,
2✔
327
                          const QModelIndex &source_right) const -> bool {
328
/* matches logic in QFileSystemModelSorter::compareNodes() */
329
#ifndef Q_OS_MAC
330
  if (fs && (source_left.column() == 0 || source_left.column() == 1)) {
2✔
331
    bool leftD = fs->isDir(source_left);
2✔
332
    bool rightD = fs->isDir(source_right);
2✔
333

334
    if (leftD ^ rightD) {
2✔
335
      return leftD;
336
    }
337
  }
338
#endif
339

340
  return QSortFilterProxyModel::lessThan(source_left, source_right);
1✔
341
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc