/usr/lib/python2.7/dist-packages/quodlibet/config.py is in exfalso 3.7.1-2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
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 | # -*- coding: utf-8 -*-
# Copyright 2004-2008 Joe Wreschnig
# 2009-2013 Nick Boultbee
# 2011-2014 Christoph Reiter
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License version 2 as
# published by the Free Software Foundation
import shutil
from . import const
from quodlibet.util.config import Config, Error
from quodlibet.util import print_d, print_w
from quodlibet.compat import PY2, iteritems
# Some plugins can be enabled on first install
AUTO_ENABLED_PLUGINS = ["Shuffle Playlist", "Remove Playlist Duplicates"]
# this defines the initial and default values
INITIAL = {
# User-defined tag name -> human name mappings
"header_maps": {
},
"player": {
"time_remaining": "false",
"replaygain": "on",
"fallback_gain": "0.0",
"pre_amp_gain": "0.0",
"backend": "gstbe",
"gst_pipeline": "",
"gst_buffer": "1.5", # stream buffer duration in seconds
"gst_device": "",
"gst_disable_gapless": "false",
},
"library": {
"exclude": "",
"refresh_on_start": "true",
},
# State about the player, to restore on startup
"memory": {
"song": "", # filename of last song
"seek": "0", # last song position, in milliseconds
"volume": "1.0", # internal volume, [0.0, 1.0]
"browser": "PanedBrowser", # browser name
"queue": "false", # on or off
"queue_expanded": "false", # on or off
"shufflequeue": "false", # on or off
"sortby": "0album", # <reversed?>tagname, song list sort
"order": "inorder",
"order_shuffle": "shuffle",
"shuffle": "false",
"plugin_selection": "", # selected plugin in manager
"column_widths": "", # column widths in c1,w1,c2,w2... format
"column_expands": "",
},
"browsers": {
"query_text": "", # none/search bar text
# panes in paned browser
"panes":
"~people <~year|[b][i]<~year>[/i][/b] - ><album>",
"pane_selection": "", # selected pane values
"pane_wide_mode": "0", # browser orientation
"background": "", # "global" filter for SearchBar
"albums": "", # album list
"album_sort": "0", # album sorting mode, default is title
"album_covers": "1", # album cover display, on/off
"album_substrings": "1", # include substrings in inline search
"collection_headers": "~people 0",
"radio": "", # radio filter selection
"rating_click": "true", # click to rate song, on/off
"rating_confirm_multiple": "false", # confirm rating multiple songs
"cover_size": "-1", # max cover height/width, <= 0 is default
"search_limit": "false", # Show the limit widgets for SearchBar
},
# Kind of a dumping ground right now, should probably be
# cleaned out later.
"settings": {
# scan directories, :-separated
"scan": "",
# scroll song list on current song change
"jump": "true",
# Unrated songs are given this value
"default_rating": "0.5",
# Rating scale i.e. maximum number of symbols
"ratings": "4",
# (0 = disabled i.e. arithmetic mean)
"bayesian_rating_factor": "0.0",
# rating symbol (black star)
"rating_symbol_full": "\xe2\x98\x85",
# rating symbol (hollow star)
"rating_symbol_blank": "\xe2\x98\x86",
# probably belongs in memory
"repeat": "false",
# Now deprecated: space-separated headers column
#"headers": " ".join(const.DEFAULT_COLUMNS),
# 2.6: this gets migrated from headers entry in code.
# TODO: re-instate columns here in > 2.6 or once most have migrated
#"columns": ",".join(const.DEFAULT_COLUMNS),
# hack to disable hints, see bug #526
"disable_hints": "false",
# search as soon as text is typed into search box
"eager_search": "true",
# tags which get searched in addition to the ones present in the
# song list, separate with ","
"search_tags": "",
# If set to "true" allow directly deleting files, even on systems that
# support sending them to the trash.
"bypass_trash": "false",
# osx implementation might be buggy so let users disable it
"disable_mmkeys": "false",
},
"rename": {
"spaces": "false",
"windows": "true",
"ascii": "false",
},
"tagsfrompath": {
"underscores": "false",
"titlecase": "false",
"split": "false",
"add": "false",
},
"plugins": {
# newline-separated plugin IDs
"active_plugins": "\n".join(AUTO_ENABLED_PLUGINS),
# Issue 1231: Maximum number of SongsMenu plugins to run at once
"default_max_plugin_invocations": 30,
},
"editing": {
"split_on": "/ & ,", # words to split on
"id3encoding": "", # ID3 encodings to try
"human_title_case": "true",
"save_to_songs": "true",
"save_email": const.EMAIL,
"alltags": "true", # show all tags, or just "human-readable" ones
# Skip dialog to save or revert changes
"auto_save_changes": "false",
"default_tags": "", # e.g. "title,artist"
},
"albumart": {
"prefer_embedded": "false",
"force_filename": "false",
"filename": "folder.jpg",
}
}
# global instance
_config = Config(version=0)
options = _config.options
get = _config.get
gettext = _config.gettext
getboolean = _config.getboolean
getint = _config.getint
getfloat = _config.getfloat
getstringlist = _config.getstringlist
setstringlist = _config.setstringlist
getlist = _config.getlist
setlist = _config.setlist
set = _config.set
settext = _config.settext
write = _config.write
reset = _config.reset
add_section = _config.add_section
has_option = _config.has_option
remove_option = _config.remove_option
register_upgrade_function = _config.register_upgrade_function
_filename = None
"""The filename last used for loading"""
def init_defaults():
"""Fills in the defaults, so they are guaranteed to be available"""
_config.defaults.clear()
for section, values in iteritems(INITIAL):
_config.defaults.add_section(section)
for key, value in iteritems(values):
_config.defaults.set(section, key, value)
def init(filename=None):
global _filename
if not _config.is_empty():
_config.clear()
_filename = filename
if filename is not None:
try:
_config.read(filename)
except (Error, EnvironmentError):
print_w("Reading config file %r failed." % filename)
# move the broken file out of the way
try:
shutil.copy(filename, filename + ".not-valid")
except EnvironmentError:
pass
def save(filename=None):
"""Writes the active config to filename, ignoring all possible errors.
If not filename is given the one used for loading is used.
"""
global _filename
if filename is None:
filename = _filename
if filename is None:
return
print_d("Writing config...")
try:
_config.write(filename)
except EnvironmentError:
print_w("Unable to write config.")
def quit():
"""Clears the active config"""
_config.clear()
def state(arg):
return _config.getboolean("settings", arg)
class RatingsPrefs(object):
"""
Models Ratings settings as configured by the user, with caching.
"""
def __init__(self):
self.__number = self.__default = None
self.__full_symbol = self.__blank_symbol = None
@property
def precision(self):
"""Returns the smallest ratings delta currently configured"""
return 1.0 / self.number
@property
def number(self):
if self.__number is None:
self.__number = getint("settings", "ratings")
return self.__number
@number.setter
def number(self, i):
"""The (maximum) integer number of ratings icons configured"""
self.__number = self.__save("ratings", int(i))
@property
def default(self):
"""The current default floating-point rating"""
if self.__default is None:
self.__default = getfloat("settings", "default_rating")
return self.__default
@default.setter
def default(self, f):
self.__default = self.__save("default_rating", float(f))
@property
def full_symbol(self):
"""The symbol to use for a full (active) rating"""
if self.__full_symbol is None:
self.__full_symbol = self.__get_symbol("full")
return self.__full_symbol
@full_symbol.setter
def full_symbol(self, s):
self.__full_symbol = self.__save("rating_symbol_full", s)
@property
def blank_symbol(self):
"""The symbol to use for a blank (inactive) rating, if needed"""
if self.__blank_symbol is None:
self.__blank_symbol = self.__get_symbol("blank")
return self.__blank_symbol
@blank_symbol.setter
def blank_symbol(self, s):
self.__blank_symbol = self.__save("rating_symbol_blank", s)
@property
def all(self):
"""Returns all the possible ratings currently available"""
return [float(i) / self.number for i in range(0, self.number + 1)]
@staticmethod
def __save(key, value):
set("settings", key, value)
return value
@staticmethod
def __get_symbol(variant="full"):
return gettext("settings", "rating_symbol_%s" % variant)
class HardCodedRatingsPrefs(RatingsPrefs):
number = int(INITIAL["settings"]["ratings"])
default = float(INITIAL["settings"]["default_rating"])
blank_symbol = INITIAL["settings"]["rating_symbol_blank"]
full_symbol = INITIAL["settings"]["rating_symbol_full"]
if PY2:
blank_symbol = blank_symbol.decode("utf-8")
full_symbol = full_symbol.decode("utf-8")
# Need an instance just for imports to work
RATINGS = RatingsPrefs()
|