Completed
Push — qml ( bc64a3...e96930 )
by Olivier
01:07
created

MainController.setFilters()   A

Complexity

Conditions 1

Size

Total Lines 3

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
c 0
b 0
f 0
dl 0
loc 3
rs 10
1
import logging
2
import os
3
import string
4
5
from PyQt5 import QtCore
6
7
from qtodotxt.lib import tasklib
8
from qtodotxt.lib.file import File
9
10
from qtodotxt.controllers.filters_tree_controller import FiltersTreeController
11
from qtodotxt.lib.filters import SimpleTextFilter, FutureFilter, IncompleteTasksFilter, CompleteTasksFilter
12
13
logger = logging.getLogger(__name__)
14
15
16
class MainController(QtCore.QObject):
17
18
    error = QtCore.pyqtSignal(str, arguments=["msg"])
19
    fileExternallyModified = QtCore.pyqtSignal()
20
21
    def __init__(self, args):
22
        super(MainController, self).__init__()
23
        self._args = args
24
        self._filteredTasks = []
25
        # use object variable for setting only used in this class
26
        # others are accessed through QSettings
27
        self._settings = QtCore.QSettings()
28
        self._showCompleted = self._settings.value("show_completed", False)
29
        self._showFuture = self._settings.value("show_completed", True)
30
        self._file = File()
31
        self._file.fileModified.connect(self.fileExternallyModified)
32
        self._modified = False
33
        self._filters_tree_controller = FiltersTreeController()
34
        self._title = "QTodoTxt"
35
        self._recentFiles = self._settings.value("recent_files", [])
36
        self._searchText = ""
37
        self._currentFilters = []
38
        # self._currentFilters = self._settings.value("current_filters", ["All"])  # move to QML
39
        self._updateCompletionStrings()
40
41
    def _taskModified(self, task):
42
        self.setModified()
43
        self.auto_save()
44
        if not task.text:
45
            self.deleteTask(task)
46
            return
47
        self.applyFilters()
48
49
    def showError(self, msg):
50
        logger.debug("ERROR: %s", msg)
51
        self.error.emit(msg)
52
53
    completionChanged = QtCore.pyqtSignal()
54
55
    @QtCore.pyqtProperty('QStringList', notify=completionChanged)
56
    def completionStrings(self):
57
        return self._completionStrings
58
    
59
    def _updateCompletionStrings(self):
60
        contexts = ['@' + name for name in self._file.getAllContexts()]
61
        projects = ['+' + name for name in self._file.getAllProjects()]
62
        lowest_priority = self._settings.value("lowest_priority", "D")
63
        idx = string.ascii_uppercase.index(lowest_priority) + 1
64
        priorities = ['(' + val +')' for val in string.ascii_uppercase[:idx]]
65
        self._completionStrings = contexts + projects + priorities + ['due:']
66
        self.completionChanged.emit()
67
68
    @QtCore.pyqtSlot('QModelIndexList')
69
    def filterByIndexes(self, idxs):
70
        filters = [self._filters_tree_controller.model.itemFromIndex(idx).filter for idx in idxs]
71
        self.setFilters(filters)
72
73
    def setFilters(self, filters):
74
        self._currentFilters = filters
75
        self.applyFilters()
76
77
    @QtCore.pyqtSlot('QString', 'int', result='int')
78
    def newTask(self, text='', after=None):
79
        task = tasklib.Task(text)
80
        if bool(self._settings.value("Preferences/add_creation_date", False, type=bool)):
81
            task.addCreationDate()
82
        task.modified.connect(self._taskModified)
83
        if after is None:
84
            after = len(self._filteredTasks) - 1
85
        self._file.tasks.append(task)
86
        self._filteredTasks.insert(after + 1, task)  # force the new task to be visible
87
        self.setModified()
88
        self.auto_save()
89
        self.filteredTasksChanged.emit()
90
        return after + 1
91
92
    @QtCore.pyqtSlot('QVariant')
93
    def deleteTask(self, task):
94
        if not isinstance(task, tasklib.Task):
95
            # if task is not a task assume it is an int
96
            task = self.filteredTasks[task]
97
        self._file.tasks.remove(task)
98
        self.setModified()
99
        self.auto_save()
100
        self.applyFilters()  # update filtered list for UI
101
102
    @property
103
    def allTasks(self):
104
        return self._file.tasks
105
106
    @allTasks.setter
107
    def allTasks(self, tasks):
108
        self._file.tasks = tasks
109
110
    filteredTasksChanged = QtCore.pyqtSignal()
111
112
    @QtCore.pyqtProperty('QVariant', notify=filteredTasksChanged)
113
    def filteredTasks(self):
114
        return self._filteredTasks
115
116
    showFutureChanged = QtCore.pyqtSignal('bool')
117
118
    @QtCore.pyqtProperty('bool', notify=showFutureChanged)
119
    def showFuture(self):
120
        return self._showFuture
121
122
    @showFuture.setter
123
    def showFuture(self, val):
124
        self._showFuture = val
125
        self.showFutureChanged.emit(val)
126
        self.applyFilters()
127
128
    searchTextChanged = QtCore.pyqtSignal(str)
129
130
    @QtCore.pyqtProperty('QString', notify=searchTextChanged)
131
    def searchText(self):
132
        return self._searchText
133
134
    @searchText.setter
135
    def searchText(self, txt):
136
        self._searchText = txt
137
        self.applyFilters()
138
        self.searchTextChanged.emit(txt)
139
140
    showCompletedChanged = QtCore.pyqtSignal('bool')
141
142
    @QtCore.pyqtProperty('bool', notify=showCompletedChanged)
143
    def showCompleted(self):
144
        return self._showCompleted
145
146
    @showCompleted.setter
147
    def showCompleted(self, val):
148
        self._showCompleted = val
149
        self.showCompletedChanged.emit(val)
150
        self.applyFilters()
151
152
    def auto_save(self):
153
        if bool(self._settings.value("Preferences/auto_save", True, type=bool)):
154
            self.save()
155
156
    def start(self):
157
        if self._args.file:
158
            filename = self._args.file
159
        else:
160
            filename = self._settings.value("last_open_file")
161
162
        if filename:
163
            try:
164
                self.open(filename)
165
            except OSError as ex:
166
                self.showError(str(ex))
167
        
168
        self.applyFilters()
169
        self._updateTitle()
170
171
    filtersUpdated = QtCore.pyqtSignal()  
172
173
    @QtCore.pyqtProperty('QVariant', notify=filtersUpdated)
174
    def filtersModel(self):
175
        return self._filters_tree_controller.model
176
   
177
    def _updateFilterTree(self):
178
        self._filters_tree_controller.showFilters(self._file)
179
        self.filtersUpdated.emit()
180
181
    def applyFilters(self):
182
        # First we filter with filters tree
183
        tasks = tasklib.filterTasks(self._currentFilters, self._file.tasks)
184
        # Then with our search text
185
        if self._searchText:
186
            tasks = tasklib.filterTasks([SimpleTextFilter(self._searchText)], tasks)
187
        # with future filter if needed
188
        if not self._showFuture:
189
            tasks = tasklib.filterTasks([FutureFilter()], tasks)
190
        # with complete filter if needed
191
        if not self._showCompleted and not CompleteTasksFilter() in self._currentFilters:
192
            tasks = tasklib.filterTasks([IncompleteTasksFilter()], tasks)
193
        self._filteredTasks = tasks
194
        self.filteredTasksChanged.emit()
195
196
197
    @QtCore.pyqtSlot()
198
    def archiveCompletedTasks(self):
199
        done = [task for task in self._file.tasks if task.is_complete]
200
        for task in done:
201
            self._file.saveDoneTask(task)
202
            self._file.tasks.remove(task)
203
        self.applyFilters()
204
        self.setModified()
205
        self.auto_save()
206
207
    modifiedChanged = QtCore.pyqtSignal(bool)
208
209
    @QtCore.pyqtProperty('bool', notify=modifiedChanged)
210
    def modified(self):
211
        return self._modified
212
213
    def setModified(self, val=True):
214
        self._modified = val
215
        self._updateTitle()
216
        if val:
217
            self._updateCompletionStrings()
218
            self._updateFilterTree()
219
        self.modifiedChanged.emit(val)
220
221
    @QtCore.pyqtSlot("QUrl")
222
    @QtCore.pyqtSlot()
223
    def save(self, path=None):
224
        if not path:
225
            path = self._file.filename
226
        elif isinstance(path, QtCore.QUrl):
227
            path = path.toLocalFile()
228
        self._file.filename = path
229
230
        logger.debug('MainController, saving file: %s.', path)
231
        try:
232
            self._file.save(path)
233
        except OSError as ex:
234
            logger.exception("Error saving file %s", path)
235
            self.showError(ex)
236
            return
237
        self._settings.setValue("last_open_file", path)
238
        self._settings.sync()
239
        self.setModified(False)
240
241
    def _updateTitle(self):
242
        title = 'QTodoTxt - '
243
        if self._file.filename:
244
            filename = os.path.basename(self._file.filename)
245
            title += filename
246
        else:
247
            title += 'Untitled'
248
        if self._modified:
249
            title += ' (*)'
250
        self._title = title
251
        self.titleChanged.emit(self._title)
252
253
    titleChanged = QtCore.pyqtSignal(str)
254
255
    @QtCore.pyqtProperty('QString', notify=titleChanged)
256
    def title(self):
257
        return self._title
258
259
    @QtCore.pyqtSlot(result='bool')
260
    def canExit(self):
261
        self.auto_save()
262
        return not self._modified
263
264
    def new(self):
265
        if self.canExit():
266
            self._file = File()
267
            self._loadFileToUI()
268
269
    @QtCore.pyqtSlot()
270
    def reload(self):
271
        self.open(self._file.filename)
272
273
    @QtCore.pyqtSlot('QUrl')
274
    @QtCore.pyqtSlot('QString')
275
    def open(self, filename):
276
        if isinstance(filename, QtCore.QUrl):
277
            filename = filename.toLocalFile()
278
        logger.debug('MainController.open called with filename="%s"', filename)
279
        try:
280
            self._file.load(filename)
281
        except Exception as ex:
282
            self.showError(self.tr("Error opening file: {}.\n Exception:{}").format(filename, ex))
283
            return
284
        self._loadFileToUI()
285
        self._settings.setValue("last_open_file", filename)
286
        for task in self._file.tasks:
287
            task.modified.connect(self._taskModified)
288
        self.applyFilters()
289
        self.updateRecentFile()
290
291
    recentFilesChanged = QtCore.pyqtSignal()
292
293
    @QtCore.pyqtProperty('QVariant', notify=recentFilesChanged)
294
    def recentFiles(self):
295
        return self._recentFiles
296
297
    def updateRecentFile(self):
298
        if self._file.filename in self._recentFiles:
299
            self._recentFiles.remove(self._file.filename)
300
        self._recentFiles.insert(0, self._file.filename)
301
        self._recentFiles = self._recentFiles[:int(self._settings.value("max_recent_files", 6))]
302
        self._settings.setValue("recent_files", self._recentFiles)
303
        self.recentFilesChanged.emit()
304
305
    def _loadFileToUI(self):
306
        self.setModified(False)
307
        self.applyFilters()
308
        self._updateCompletionStrings()
309
        self._updateFilterTree()
310