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

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

204

205

206

207

208

209

210

211

212

213

214

215

216

217

218

219

220

221

222

223

224

225

226

227

228

229

230

231

232

233

234

235

236

237

238

239

240

241

242

243

244

245

246

247

248

249

250

251

252

253

254

255

256

257

258

259

260

261

262

263

264

265

266

267

268

269

270

271

272

273

274

275

276

277

278

279

280

281

282

283

284

285

286

287

288

289

290

291

292

293

294

295

296

297

298

299

300

301

302

303

304

305

306

307

308

309

310

311

312

313

314

315

316

317

318

319

320

321

322

323

324

325

326

327

328

329

330

331

332

333

334

335

336

337

338

339

340

341

342

343

344

345

346

347

348

349

350

351

352

353

354

355

356

357

358

359

360

361

362

363

364

365

366

367

368

369

370

371

372

373

374

375

376

377

378

379

380

381

382

383

384

385

386

387

388

389

390

391

392

393

394

395

396

397

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

 

# Copyright 2016-2018 Florian Bruhin (The Compiler) <mail@qutebrowser.org> 

# 

# 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/>. 

 

# We get various "abstract but not overridden" warnings 

# pylint: disable=abstract-method 

 

"""Bridge from QWebEngineSettings to our own settings. 

 

Module attributes: 

ATTRIBUTES: A mapping from internal setting names to QWebEngineSetting enum 

constants. 

""" 

 

import os 

 

import sip 

from PyQt5.QtGui import QFont 

from PyQt5.QtWebEngineWidgets import (QWebEngineSettings, QWebEngineProfile, 

QWebEngineScript) 

 

from qutebrowser.browser import shared 

from qutebrowser.browser.webengine import spell 

from qutebrowser.config import config, websettings 

from qutebrowser.utils import (utils, standarddir, javascript, qtutils, 

message, log, objreg) 

 

# The default QWebEngineProfile 

default_profile = None 

# The QWebEngineProfile used for private (off-the-record) windows 

private_profile = None 

 

 

class Base(websettings.Base): 

 

"""Base settings class with appropriate _get_global_settings.""" 

 

def _get_global_settings(self): 

return [default_profile.settings(), private_profile.settings()] 

 

 

class Attribute(Base, websettings.Attribute): 

 

"""A setting set via QWebEngineSettings::setAttribute.""" 

 

ENUM_BASE = QWebEngineSettings 

 

 

class Setter(Base, websettings.Setter): 

 

"""A setting set via a QWebEngineSettings setter method.""" 

 

pass 

 

 

class FontFamilySetter(Base, websettings.FontFamilySetter): 

 

"""A setter for a font family. 

 

Gets the default value from QFont. 

""" 

 

def __init__(self, font): 

# Mapping from WebEngineSettings::initDefaults in 

# qtwebengine/src/core/web_engine_settings.cpp 

font_to_qfont = { 

QWebEngineSettings.StandardFont: QFont.Serif, 

QWebEngineSettings.FixedFont: QFont.Monospace, 

QWebEngineSettings.SerifFont: QFont.Serif, 

QWebEngineSettings.SansSerifFont: QFont.SansSerif, 

QWebEngineSettings.CursiveFont: QFont.Cursive, 

QWebEngineSettings.FantasyFont: QFont.Fantasy, 

} 

super().__init__(setter=QWebEngineSettings.setFontFamily, font=font, 

qfont=font_to_qfont[font]) 

 

 

class DefaultProfileSetter(websettings.Base): 

 

"""A setting set on the QWebEngineProfile.""" 

 

def __init__(self, setter, converter=None, default=websettings.UNSET): 

super().__init__(default) 

self._setter = setter 

self._converter = converter 

 

def __repr__(self): 

return utils.get_repr(self, setter=self._setter, constructor=True) 

 

def _set(self, value, settings=None): 

if settings is not None: 

raise ValueError("'settings' may not be set with " 

"DefaultProfileSetters!") 

 

setter = getattr(default_profile, self._setter) 

if self._converter is not None: 

value = self._converter(value) 

 

setter(value) 

 

 

class PersistentCookiePolicy(DefaultProfileSetter): 

 

"""The content.cookies.store setting is different from other settings.""" 

 

def __init__(self): 

super().__init__('setPersistentCookiesPolicy') 

 

def _set(self, value, settings=None): 

if settings is not None: 

raise ValueError("'settings' may not be set with " 

"PersistentCookiePolicy!") 

setter = getattr(QWebEngineProfile.defaultProfile(), self._setter) 

setter( 

QWebEngineProfile.AllowPersistentCookies if value else 

QWebEngineProfile.NoPersistentCookies 

) 

 

 

class DictionaryLanguageSetter(DefaultProfileSetter): 

 

"""Sets paths to dictionary files based on language codes.""" 

 

def __init__(self): 

super().__init__('setSpellCheckLanguages', default=[]) 

 

def _find_installed(self, code): 

local_filename = spell.local_filename(code) 

if not local_filename: 

message.warning( 

"Language {} is not installed - see scripts/dictcli.py " 

"in qutebrowser's sources".format(code)) 

return local_filename 

 

def _set(self, value, settings=None): 

if settings is not None: 

raise ValueError("'settings' may not be set with " 

"DictionaryLanguageSetter!") 

filenames = [self._find_installed(code) for code in value] 

log.config.debug("Found dicts: {}".format(filenames)) 

super()._set([f for f in filenames if f], settings) 

 

 

def _init_stylesheet(profile): 

"""Initialize custom stylesheets. 

 

Partially inspired by QupZilla: 

https://github.com/QupZilla/qupzilla/blob/v2.0/src/lib/app/mainapplication.cpp#L1063-L1101 

""" 

old_script = profile.scripts().findScript('_qute_stylesheet') 

if not old_script.isNull(): 

profile.scripts().remove(old_script) 

 

css = shared.get_user_stylesheet() 

source = '\n'.join([ 

'"use strict";', 

'window._qutebrowser = window._qutebrowser || {};', 

utils.read_file('javascript/stylesheet.js'), 

javascript.assemble('stylesheet', 'set_css', css), 

]) 

 

script = QWebEngineScript() 

script.setName('_qute_stylesheet') 

script.setInjectionPoint(QWebEngineScript.DocumentCreation) 

script.setWorldId(QWebEngineScript.ApplicationWorld) 

script.setRunsOnSubFrames(True) 

script.setSourceCode(source) 

profile.scripts().insert(script) 

 

 

def _update_stylesheet(): 

"""Update the custom stylesheet in existing tabs.""" 

css = shared.get_user_stylesheet() 

code = javascript.assemble('stylesheet', 'set_css', css) 

for win_id, window in objreg.window_registry.items(): 

# We could be in the middle of destroying a window here 

if sip.isdeleted(window): 

continue 

tab_registry = objreg.get('tab-registry', scope='window', 

window=win_id) 

for tab in tab_registry.values(): 

tab.run_js_async(code) 

 

 

def _set_http_headers(profile): 

"""Set the user agent and accept-language for the given profile. 

 

We override those per request in the URL interceptor (to allow for 

per-domain values), but this one still gets used for things like 

window.navigator.userAgent/.languages in JS. 

""" 

profile.setHttpUserAgent(config.val.content.headers.user_agent) 

accept_language = config.val.content.headers.accept_language 

if accept_language is not None: 

profile.setHttpAcceptLanguage(accept_language) 

 

 

def _update_settings(option): 

"""Update global settings when qwebsettings changed.""" 

websettings.update_mappings(MAPPINGS, option) 

if option in ['scrolling.bar', 'content.user_stylesheets']: 

_init_stylesheet(default_profile) 

_init_stylesheet(private_profile) 

_update_stylesheet() 

elif option in ['content.headers.user_agent', 

'content.headers.accept_language']: 

_set_http_headers(default_profile) 

_set_http_headers(private_profile) 

 

 

def _init_profiles(): 

"""Init the two used QWebEngineProfiles.""" 

global default_profile, private_profile 

default_profile = QWebEngineProfile.defaultProfile() 

default_profile.setCachePath( 

os.path.join(standarddir.cache(), 'webengine')) 

default_profile.setPersistentStoragePath( 

os.path.join(standarddir.data(), 'webengine')) 

_init_stylesheet(default_profile) 

_set_http_headers(default_profile) 

 

private_profile = QWebEngineProfile() 

assert private_profile.isOffTheRecord() 

_init_stylesheet(private_profile) 

_set_http_headers(private_profile) 

 

if qtutils.version_check('5.8'): 

default_profile.setSpellCheckEnabled(True) 

private_profile.setSpellCheckEnabled(True) 

 

 

def inject_userscripts(): 

"""Register user JavaScript files with the global profiles.""" 

# The Greasemonkey metadata block support in QtWebEngine only starts at 

# Qt 5.8. With 5.7.1, we need to inject the scripts ourselves in response 

# to urlChanged. 

if not qtutils.version_check('5.8'): 

return 

 

# Since we are inserting scripts into profile.scripts they won't 

# just get replaced by new gm scripts like if we were injecting them 

# ourselves so we need to remove all gm scripts, while not removing 

# any other stuff that might have been added. Like the one for 

# stylesheets. 

greasemonkey = objreg.get('greasemonkey') 

for profile in [default_profile, private_profile]: 

scripts = profile.scripts() 

for script in scripts.toList(): 

if script.name().startswith("GM-"): 

log.greasemonkey.debug('Removing script: {}' 

.format(script.name())) 

removed = scripts.remove(script) 

assert removed, script.name() 

 

# Then add the new scripts. 

for script in greasemonkey.all_scripts(): 

# @run-at (and @include/@exclude/@match) is parsed by 

# QWebEngineScript. 

new_script = QWebEngineScript() 

new_script.setWorldId(QWebEngineScript.MainWorld) 

new_script.setSourceCode(script.code()) 

new_script.setName("GM-{}".format(script.name)) 

new_script.setRunsOnSubFrames(script.runs_on_sub_frames) 

log.greasemonkey.debug('adding script: {}' 

.format(new_script.name())) 

scripts.insert(new_script) 

 

 

def init(args): 

"""Initialize the global QWebSettings.""" 

if args.enable_webengine_inspector: 

os.environ['QTWEBENGINE_REMOTE_DEBUGGING'] = str(utils.random_port()) 

 

_init_profiles() 

 

# We need to do this here as a WORKAROUND for 

# https://bugreports.qt.io/browse/QTBUG-58650 

if not qtutils.version_check('5.9', compiled=False): 

PersistentCookiePolicy().set(config.val.content.cookies.store) 

Attribute(QWebEngineSettings.FullScreenSupportEnabled).set(True) 

 

websettings.init_mappings(MAPPINGS) 

config.instance.changed.connect(_update_settings) 

 

 

def shutdown(): 

# FIXME:qtwebengine do we need to do something for a clean shutdown here? 

pass 

 

 

# Missing QtWebEngine attributes: 

# - ScreenCaptureEnabled 

# - Accelerated2dCanvasEnabled 

# - AutoLoadIconsForPage 

# - TouchIconsEnabled 

# - FocusOnNavigationEnabled (5.8) 

# - AllowRunningInsecureContent (5.8) 

# 

# Missing QtWebEngine fonts: 

# - PictographFont 

 

 

318 ↛ exitline 342 didn't finish the lambda on line 342MAPPINGS = { 

'content.images': 

Attribute(QWebEngineSettings.AutoLoadImages), 

'content.javascript.enabled': 

Attribute(QWebEngineSettings.JavascriptEnabled), 

'content.javascript.can_open_tabs_automatically': 

Attribute(QWebEngineSettings.JavascriptCanOpenWindows), 

'content.javascript.can_access_clipboard': 

Attribute(QWebEngineSettings.JavascriptCanAccessClipboard), 

'content.plugins': 

Attribute(QWebEngineSettings.PluginsEnabled), 

'content.hyperlink_auditing': 

Attribute(QWebEngineSettings.HyperlinkAuditingEnabled), 

'content.local_content_can_access_remote_urls': 

Attribute(QWebEngineSettings.LocalContentCanAccessRemoteUrls), 

'content.local_content_can_access_file_urls': 

Attribute(QWebEngineSettings.LocalContentCanAccessFileUrls), 

'content.webgl': 

Attribute(QWebEngineSettings.WebGLEnabled), 

'content.local_storage': 

Attribute(QWebEngineSettings.LocalStorageEnabled), 

'content.cache.size': 

# 0: automatically managed by QtWebEngine 

DefaultProfileSetter('setHttpCacheMaximumSize', default=0, 

converter=lambda val: 

qtutils.check_overflow(val, 'int', fatal=False)), 

'content.xss_auditing': 

Attribute(QWebEngineSettings.XSSAuditingEnabled), 

'content.default_encoding': 

Setter(QWebEngineSettings.setDefaultTextEncoding), 

 

'input.spatial_navigation': 

Attribute(QWebEngineSettings.SpatialNavigationEnabled), 

'input.links_included_in_focus_chain': 

Attribute(QWebEngineSettings.LinksIncludedInFocusChain), 

 

'fonts.web.family.standard': 

FontFamilySetter(QWebEngineSettings.StandardFont), 

'fonts.web.family.fixed': 

FontFamilySetter(QWebEngineSettings.FixedFont), 

'fonts.web.family.serif': 

FontFamilySetter(QWebEngineSettings.SerifFont), 

'fonts.web.family.sans_serif': 

FontFamilySetter(QWebEngineSettings.SansSerifFont), 

'fonts.web.family.cursive': 

FontFamilySetter(QWebEngineSettings.CursiveFont), 

'fonts.web.family.fantasy': 

FontFamilySetter(QWebEngineSettings.FantasyFont), 

'fonts.web.size.minimum': 

Setter(QWebEngineSettings.setFontSize, 

args=[QWebEngineSettings.MinimumFontSize]), 

'fonts.web.size.minimum_logical': 

Setter(QWebEngineSettings.setFontSize, 

args=[QWebEngineSettings.MinimumLogicalFontSize]), 

'fonts.web.size.default': 

Setter(QWebEngineSettings.setFontSize, 

args=[QWebEngineSettings.DefaultFontSize]), 

'fonts.web.size.default_fixed': 

Setter(QWebEngineSettings.setFontSize, 

args=[QWebEngineSettings.DefaultFixedFontSize]), 

 

'scrolling.smooth': 

Attribute(QWebEngineSettings.ScrollAnimatorEnabled), 

} 

 

try: 

MAPPINGS['content.print_element_backgrounds'] = Attribute( 

QWebEngineSettings.PrintElementBackgrounds) 

except AttributeError: 

# Added in Qt 5.8 

pass 

 

 

391 ↛ 395line 391 didn't jump to line 395, because the condition on line 391 was never falseif qtutils.version_check('5.8'): 

MAPPINGS['spellcheck.languages'] = DictionaryLanguageSetter() 

 

 

395 ↛ exitline 395 didn't exit the module, because the condition on line 395 was never falseif qtutils.version_check('5.9', compiled=False): 

# https://bugreports.qt.io/browse/QTBUG-58650 

MAPPINGS['content.cookies.store'] = PersistentCookiePolicy()