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

IJHack / QtPass / 24616009682

18 Apr 2026 11:10PM UTC coverage: 22.627%. First build
24616009682

Pull #1046

github

web-flow
Merge 6f215fd67 into afd4f6fc3
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
  QT_WARNING_PUSH
105
  QT_WARNING_DISABLE_DEPRECATED
106
  invalidateFilter();
×
107
  QT_WARNING_POP
108
}
×
109

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

121
  QVariant initial_value;
122
  initial_value = QSortFilterProxyModel::data(index, role);
1✔
123

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

130
  return initial_value;
1✔
131
}
1✔
132

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

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

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

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

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

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

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

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

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

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

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

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

231
  if (column > 0) {
×
232
    return false;
233
  }
234

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

248
  return false;
249
}
×
250

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

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

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

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

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

338
  return QSortFilterProxyModel::lessThan(source_left, source_right);
1✔
339
}
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