Hide keyboard shortcuts

Hot-keys on this page

r m x p   toggle line displays

j k   next/prev highlighted chunk

0   (zero) top of page

1   (one) first highlighted chunk

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: 

 

# Copyright 2017-2018 Ryan Roden-Corrent (rcorre) <ryan@rcorre.net> 

# 

# This file is part of qutebrowser. 

# 

# qutebrowser is free software: you can redistribute it and/or modify 

# it under the terms of the GNU General Public License as published by 

# the Free Software Foundation, either version 3 of the License, or 

# (at your option) any later version. 

# 

# qutebrowser is distributed in the hope that it will be useful, 

# but WITHOUT ANY WARRANTY; without even the implied warranty of 

# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 

# GNU General Public License for more details. 

# 

# You should have received a copy of the GNU General Public License 

# along with qutebrowser. If not, see <http://www.gnu.org/licenses/>. 

 

"""A completion category that queries the SQL History store.""" 

 

from PyQt5.QtSql import QSqlQueryModel 

 

from qutebrowser.misc import sql 

from qutebrowser.utils import debug 

from qutebrowser.config import config 

 

 

class HistoryCategory(QSqlQueryModel): 

 

"""A completion category that queries the SQL History store.""" 

 

def __init__(self, *, delete_func=None, parent=None): 

"""Create a new History completion category.""" 

super().__init__(parent=parent) 

self.name = "History" 

self._query = None 

 

# advertise that this model filters by URL and title 

self.columns_to_filter = [0, 1] 

self.delete_func = delete_func 

 

def _atime_expr(self): 

"""If max_items is set, return an expression to limit the query.""" 

max_items = config.val.completion.web_history_max_items 

# HistoryCategory should not be added to the completion in that case. 

assert max_items != 0 

 

if max_items < 0: 

return '' 

 

min_atime = sql.Query(' '.join([ 

'SELECT min(last_atime) FROM', 

'(SELECT last_atime FROM CompletionHistory', 

'ORDER BY last_atime DESC LIMIT :limit)', 

])).run(limit=max_items).value() 

 

if not min_atime: 

# if there are no history items, min_atime may be '' (issue #2849) 

return '' 

 

return "AND last_atime >= {}".format(min_atime) 

 

def set_pattern(self, pattern): 

"""Set the pattern used to filter results. 

 

Args: 

pattern: string pattern to filter by. 

""" 

# escape to treat a user input % or _ as a literal, not a wildcard 

pattern = pattern.replace('%', '\\%') 

pattern = pattern.replace('_', '\\_') 

words = ['%{}%'.format(w) for w in pattern.split(' ')] 

 

# build a where clause to match all of the words in any order 

# given the search term "a b", the WHERE clause would be: 

# ((url || title) LIKE '%a%') AND ((url || title) LIKE '%b%') 

where_clause = ' AND '.join( 

"(url || title) LIKE :{} escape '\\'".format(i) 

for i in range(len(words))) 

 

# replace ' in timestamp-format to avoid breaking the query 

timestamp_format = config.val.completion.timestamp_format 

timefmt = ("strftime('{}', last_atime, 'unixepoch', 'localtime')" 

.format(timestamp_format.replace("'", "`"))) 

 

if not self._query or len(words) != len(self._query.boundValues()): 

# if the number of words changed, we need to generate a new query 

# otherwise, we can reuse the prepared query for performance 

self._query = sql.Query(' '.join([ 

"SELECT url, title, {}".format(timefmt), 

"FROM CompletionHistory", 

# the incoming pattern will have literal % and _ escaped 

# we need to tell sql to treat '\' as an escape character 

'WHERE ({})'.format(where_clause), 

self._atime_expr(), 

"ORDER BY last_atime DESC", 

]), forward_only=False) 

 

with debug.log_time('sql', 'Running completion query'): 

self._query.run(**{ 

str(i): w for i, w in enumerate(words)}) 

self.setQuery(self._query) 

 

def removeRows(self, row, _count, _parent=None): 

"""Override QAbstractItemModel::removeRows to re-run sql query.""" 

# re-run query to reload updated table 

with debug.log_time('sql', 'Re-running completion query post-delete'): 

self._query.run() 

self.setQuery(self._query) 

while self.rowCount() < row: 

self.fetchMore() 

return True