550 lines
23 KiB
Python
550 lines
23 KiB
Python
# Copyright (C) 2003-2005 Vincent Hanquez <tab AT snarc.org>
|
|
# Copyright (C) 2003-2014 Yann Leboulanger <asterix AT lagaule.org>
|
|
# Copyright (C) 2005 Alex Podaras <bigpod AT gmail.com>
|
|
# Stéphan Kochen <stephan AT kochen.nl>
|
|
# Copyright (C) 2005-2006 Dimitur Kirov <dkirov AT gmail.com>
|
|
# Nikos Kouremenos <kourem AT gmail.com>
|
|
# Copyright (C) 2006 Junglecow J <junglecow AT gmail.com>
|
|
# Copyright (C) 2006-2007 Travis Shirk <travis AT pobox.com>
|
|
# Stefan Bethge <stefan AT lanpartei.de>
|
|
# Copyright (C) 2006-2008 Jean-Marie Traissard <jim AT lapin.org>
|
|
# Copyright (C) 2007 James Newton <redshodan AT gmail.com>
|
|
# Julien Pivotto <roidelapluie AT gmail.com>
|
|
# Copyright (C) 2007-2008 Stephan Erb <steve-e AT h3c.de>
|
|
# Copyright (C) 2008 Jonathan Schleifer <js-gajim AT webkeks.org>
|
|
#
|
|
# This file is part of Gajim.
|
|
#
|
|
# Gajim 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; version 3 only.
|
|
#
|
|
# Gajim 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 Gajim. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
import os
|
|
|
|
from gi.repository import Gtk
|
|
from gi.repository import Gdk
|
|
from gi.repository import GObject
|
|
|
|
from gajim.common import helpers
|
|
from gajim.common import app
|
|
from gajim.common.i18n import _
|
|
|
|
from gajim import gtkgui_helpers
|
|
from gajim import dialogs
|
|
from gajim import dataforms_widget
|
|
from gajim import gui_menu_builder
|
|
|
|
from gajim.gtk.dialogs import ConfirmationDialog
|
|
from gajim.gtk.dialogs import ConfirmationDialogDoubleRadio
|
|
from gajim.gtk.dialogs import ErrorDialog
|
|
from gajim.gtk.dialogs import InputDialog
|
|
|
|
|
|
class FakeDataForm(Gtk.Table):
|
|
"""
|
|
Class for forms that are in XML format <entry1>value1</entry1> infos in a
|
|
table {entry1: value1}
|
|
"""
|
|
|
|
def __init__(self, infos, selectable=False):
|
|
GObject.GObject.__init__(self)
|
|
self.infos = infos
|
|
self.selectable = selectable
|
|
self.entries = {}
|
|
self._draw_table()
|
|
|
|
def _draw_table(self):
|
|
"""
|
|
Draw the table
|
|
"""
|
|
nbrow = 0
|
|
if 'instructions' in self.infos:
|
|
nbrow = 1
|
|
self.resize(rows=nbrow, columns=2)
|
|
label = Gtk.Label(label=self.infos['instructions'])
|
|
if self.selectable:
|
|
label.set_selectable(True)
|
|
self.attach(label, 0, 2, 0, 1, 0, 0, 0, 0)
|
|
for name in self.infos.keys():
|
|
if name in ('key', 'instructions', 'x', 'registered'):
|
|
continue
|
|
if not name:
|
|
continue
|
|
|
|
nbrow = nbrow + 1
|
|
self.resize(rows=nbrow, columns=2)
|
|
label = Gtk.Label(label=name.capitalize() + ':')
|
|
self.attach(label, 0, 1, nbrow - 1, nbrow, 0, 0, 0, 0)
|
|
entry = Gtk.Entry()
|
|
entry.set_activates_default(True)
|
|
if self.infos[name]:
|
|
entry.set_text(self.infos[name])
|
|
if name == 'password':
|
|
entry.set_visibility(False)
|
|
self.attach(entry, 1, 2, nbrow - 1, nbrow, 0, 0, 0, 0)
|
|
self.entries[name] = entry
|
|
if nbrow == 1:
|
|
entry.grab_focus()
|
|
|
|
def get_infos(self):
|
|
for name in self.entries:
|
|
self.infos[name] = self.entries[name].get_text()
|
|
return self.infos
|
|
|
|
class GroupchatConfigWindow:
|
|
|
|
def __init__(self, account, room_jid, form=None):
|
|
self.account = account
|
|
self.room_jid = room_jid
|
|
self.form = form
|
|
self.remove_button = {}
|
|
self.affiliation_treeview = {}
|
|
self.start_users_dict = {} # list at the beginning
|
|
self.affiliation_labels = {'outcast': _('Ban List'),
|
|
'member': _('Member List'), 'owner': _('Owner List'),
|
|
'admin':_('Administrator List')}
|
|
|
|
self.xml = gtkgui_helpers.get_gtk_builder('data_form_window.ui',
|
|
'data_form_window')
|
|
self.window = self.xml.get_object('data_form_window')
|
|
self.window.set_transient_for(app.interface.roster.window)
|
|
|
|
if self.form:
|
|
config_vbox = self.xml.get_object('config_vbox')
|
|
self.data_form_widget = dataforms_widget.DataFormWidget(self.form)
|
|
# hide scrollbar of this data_form_widget, we already have in this
|
|
# widget
|
|
sw = self.data_form_widget.xml.get_object(
|
|
'single_form_scrolledwindow')
|
|
sw.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.NEVER)
|
|
if self.form.title:
|
|
self.xml.get_object('title_label').set_text(self.form.title)
|
|
else:
|
|
self.xml.get_object('title_hseparator').set_no_show_all(True)
|
|
self.xml.get_object('title_hseparator').hide()
|
|
|
|
self.data_form_widget.show()
|
|
config_vbox.pack_start(self.data_form_widget, True, True, 0)
|
|
else:
|
|
self.xml.get_object('title_label').set_no_show_all(True)
|
|
self.xml.get_object('title_label').hide()
|
|
self.xml.get_object('title_hseparator').set_no_show_all(True)
|
|
self.xml.get_object('title_hseparator').hide()
|
|
self.xml.get_object('config_hseparator').set_no_show_all(True)
|
|
self.xml.get_object('config_hseparator').hide()
|
|
|
|
# Draw the edit affiliation list things
|
|
add_on_vbox = self.xml.get_object('add_on_vbox')
|
|
|
|
for affiliation in self.affiliation_labels:
|
|
self.start_users_dict[affiliation] = {}
|
|
hbox = Gtk.HBox(spacing=5)
|
|
add_on_vbox.pack_start(hbox, False, True, 0)
|
|
|
|
label = Gtk.Label(label=self.affiliation_labels[affiliation])
|
|
hbox.pack_start(label, False, True, 0)
|
|
|
|
bb = Gtk.HButtonBox()
|
|
bb.set_layout(Gtk.ButtonBoxStyle.END)
|
|
bb.set_spacing(5)
|
|
hbox.pack_start(bb, True, True, 0)
|
|
add_button = Gtk.Button(stock=Gtk.STOCK_ADD)
|
|
add_button.connect('clicked', self.on_add_button_clicked,
|
|
affiliation)
|
|
bb.pack_start(add_button, True, True, 0)
|
|
self.remove_button[affiliation] = Gtk.Button(stock=Gtk.STOCK_REMOVE)
|
|
self.remove_button[affiliation].set_sensitive(False)
|
|
self.remove_button[affiliation].connect('clicked',
|
|
self.on_remove_button_clicked, affiliation)
|
|
bb.pack_start(self.remove_button[affiliation], True, True, 0)
|
|
|
|
# jid, reason, nick, role
|
|
liststore = Gtk.ListStore(str, str, str, str)
|
|
self.affiliation_treeview[affiliation] = Gtk.TreeView(liststore)
|
|
self.affiliation_treeview[affiliation].get_selection().set_mode(
|
|
Gtk.SelectionMode.MULTIPLE)
|
|
self.affiliation_treeview[affiliation].connect('cursor-changed',
|
|
self.on_affiliation_treeview_cursor_changed, affiliation)
|
|
renderer = Gtk.CellRendererText()
|
|
col = Gtk.TreeViewColumn(_('JID'), renderer)
|
|
col.add_attribute(renderer, 'text', 0)
|
|
col.set_resizable(True)
|
|
col.set_sort_column_id(0)
|
|
self.affiliation_treeview[affiliation].append_column(col)
|
|
|
|
if affiliation == 'outcast':
|
|
renderer = Gtk.CellRendererText()
|
|
renderer.set_property('editable', True)
|
|
renderer.connect('edited', self.on_cell_edited)
|
|
col = Gtk.TreeViewColumn(_('Reason'), renderer)
|
|
col.add_attribute(renderer, 'text', 1)
|
|
col.set_resizable(True)
|
|
col.set_sort_column_id(1)
|
|
self.affiliation_treeview[affiliation].append_column(col)
|
|
elif affiliation == 'member':
|
|
renderer = Gtk.CellRendererText()
|
|
col = Gtk.TreeViewColumn(_('Nick'), renderer)
|
|
col.add_attribute(renderer, 'text', 2)
|
|
col.set_resizable(True)
|
|
col.set_sort_column_id(2)
|
|
self.affiliation_treeview[affiliation].append_column(col)
|
|
renderer = Gtk.CellRendererText()
|
|
col = Gtk.TreeViewColumn(_('Role'), renderer)
|
|
col.add_attribute(renderer, 'text', 3)
|
|
col.set_resizable(True)
|
|
col.set_sort_column_id(3)
|
|
self.affiliation_treeview[affiliation].append_column(col)
|
|
|
|
sw = Gtk.ScrolledWindow()
|
|
sw.add(self.affiliation_treeview[affiliation])
|
|
add_on_vbox.pack_start(sw, True, True, 0)
|
|
con = app.connections[self.account]
|
|
con.get_module('MUC').get_affiliation(self.room_jid, affiliation)
|
|
|
|
self.xml.connect_signals(self)
|
|
self.window.connect('delete-event', self.on_cancel_button_clicked)
|
|
self.window.show_all()
|
|
|
|
def on_cancel_button_clicked(self, *args):
|
|
if self.form:
|
|
con = app.connections[self.account]
|
|
con.get_module('MUC').cancel_config(self.room_jid)
|
|
self.window.destroy()
|
|
|
|
def on_cell_edited(self, cell, path, new_text):
|
|
model = self.affiliation_treeview['outcast'].get_model()
|
|
new_text = new_text
|
|
iter_ = model.get_iter(path)
|
|
model[iter_][1] = new_text
|
|
|
|
def on_add_button_clicked(self, widget, affiliation):
|
|
if affiliation == 'outcast':
|
|
title = _('Banning…')
|
|
#You can move '\n' before user@domain if that line is TOO BIG
|
|
prompt = _('<b>Whom do you want to ban?</b>\n\n')
|
|
elif affiliation == 'member':
|
|
title = _('Adding Member…')
|
|
prompt = _('<b>Whom do you want to make a member?</b>\n\n')
|
|
elif affiliation == 'owner':
|
|
title = _('Adding Owner…')
|
|
prompt = _('<b>Whom do you want to make an owner?</b>\n\n')
|
|
else:
|
|
title = _('Adding Administrator…')
|
|
prompt = _('<b>Whom do you want to make an administrator?</b>\n\n')
|
|
prompt += _('Can be one of the following:\n'
|
|
'1. user@domain/resource (only that resource matches).\n'
|
|
'2. user@domain (any resource matches).\n'
|
|
'3. domain/resource (only that resource matches).\n'
|
|
'4. domain (the domain itself matches, as does any user@domain,\n'
|
|
'domain/resource, or address containing a subdomain).')
|
|
|
|
def on_ok(jid):
|
|
if not jid:
|
|
return
|
|
model = self.affiliation_treeview[affiliation].get_model()
|
|
model.append((jid, '', '', ''))
|
|
InputDialog(title, prompt, ok_handler=on_ok)
|
|
|
|
def on_remove_button_clicked(self, widget, affiliation):
|
|
selection = self.affiliation_treeview[affiliation].get_selection()
|
|
model, paths = selection.get_selected_rows()
|
|
row_refs = []
|
|
for path in paths:
|
|
row_refs.append(Gtk.TreeRowReference.new(model, path))
|
|
for row_ref in row_refs:
|
|
path = row_ref.get_path()
|
|
iter_ = model.get_iter(path)
|
|
model.remove(iter_)
|
|
self.remove_button[affiliation].set_sensitive(False)
|
|
|
|
def on_affiliation_treeview_cursor_changed(self, widget, affiliation):
|
|
self.remove_button[affiliation].set_sensitive(True)
|
|
|
|
def affiliation_list_received(self, users_dict):
|
|
"""
|
|
Fill the affiliation treeview
|
|
"""
|
|
for jid in users_dict:
|
|
affiliation = users_dict[jid]['affiliation']
|
|
if affiliation not in self.affiliation_labels.keys():
|
|
# Unknown affiliation or 'none' affiliation, do not show it
|
|
continue
|
|
self.start_users_dict[affiliation][jid] = users_dict[jid]
|
|
tv = self.affiliation_treeview[affiliation]
|
|
model = tv.get_model()
|
|
reason = users_dict[jid].get('reason', '')
|
|
nick = users_dict[jid].get('nick', '')
|
|
role = users_dict[jid].get('role', '')
|
|
model.append((jid, reason, nick, role))
|
|
|
|
def on_data_form_window_destroy(self, widget):
|
|
del app.interface.instances[self.account]['gc_config'][self.room_jid]
|
|
|
|
def on_ok_button_clicked(self, widget):
|
|
if self.form:
|
|
form = self.data_form_widget.data_form
|
|
con = app.connections[self.account]
|
|
con.get_module('MUC').set_config(self.room_jid, form)
|
|
for affiliation in self.affiliation_labels:
|
|
users_dict = {}
|
|
actual_jid_list = []
|
|
model = self.affiliation_treeview[affiliation].get_model()
|
|
iter_ = model.get_iter_first()
|
|
# add new jid
|
|
while iter_:
|
|
jid = model[iter_][0]
|
|
actual_jid_list.append(jid)
|
|
if jid not in self.start_users_dict[affiliation] or \
|
|
(affiliation == 'outcast' and 'reason' in self.start_users_dict[
|
|
affiliation][jid] and self.start_users_dict[affiliation][jid]\
|
|
['reason'] != model[iter_][1]):
|
|
users_dict[jid] = {'affiliation': affiliation}
|
|
if affiliation == 'outcast':
|
|
users_dict[jid]['reason'] = model[iter_][1]
|
|
iter_ = model.iter_next(iter_)
|
|
# remove removed one
|
|
for jid in self.start_users_dict[affiliation]:
|
|
if jid not in actual_jid_list:
|
|
users_dict[jid] = {'affiliation': 'none'}
|
|
if users_dict:
|
|
con = app.connections[self.account]
|
|
con.get_module('MUC').set_affiliation(
|
|
self.room_jid, users_dict)
|
|
self.window.destroy()
|
|
|
|
#---------- RemoveAccountWindow class -------------#
|
|
class RemoveAccountWindow:
|
|
"""
|
|
Ask for removing from gajim only or from gajim and server too and do
|
|
removing of the account given
|
|
"""
|
|
|
|
def on_remove_account_window_destroy(self, widget):
|
|
if self.account in app.interface.instances:
|
|
del app.interface.instances[self.account]['remove_account']
|
|
|
|
def on_cancel_button_clicked(self, widget):
|
|
self.window.destroy()
|
|
|
|
def __init__(self, account):
|
|
self.account = account
|
|
xml = gtkgui_helpers.get_gtk_builder('remove_account_window.ui')
|
|
self.window = xml.get_object('remove_account_window')
|
|
active_window = app.app.get_active_window()
|
|
self.window.set_transient_for(active_window)
|
|
self.remove_and_unregister_radiobutton = xml.get_object(
|
|
'remove_and_unregister_radiobutton')
|
|
self.window.set_title(_('Removing %s account') % self.account)
|
|
xml.connect_signals(self)
|
|
self.window.show_all()
|
|
|
|
def on_remove_button_clicked(self, widget):
|
|
def remove():
|
|
if self.account in app.connections and \
|
|
app.connections[self.account].connected and \
|
|
not self.remove_and_unregister_radiobutton.get_active():
|
|
# change status to offline only if we will not remove this JID from
|
|
# server
|
|
app.connections[self.account].change_status('offline', 'offline')
|
|
if self.remove_and_unregister_radiobutton.get_active():
|
|
if not self.account in app.connections:
|
|
ErrorDialog(
|
|
_('Account is disabled'),
|
|
_('To unregister from a server, account must be '
|
|
'enabled.'),
|
|
transient_for=self.window)
|
|
return
|
|
if not app.connections[self.account].password:
|
|
def on_ok(passphrase, checked):
|
|
if passphrase == -1:
|
|
# We don't remove account cause we canceled pw window
|
|
return
|
|
app.connections[self.account].password = passphrase
|
|
app.connections[self.account].unregister_account(
|
|
self._on_remove_success)
|
|
|
|
dialogs.PassphraseDialog(
|
|
_('Password Required'),
|
|
_('Enter your password for account %s') % self.account,
|
|
_('Save password'), ok_handler=on_ok,
|
|
transient_for=self.window)
|
|
return
|
|
app.connections[self.account].unregister_account(
|
|
self._on_remove_success)
|
|
else:
|
|
self._on_remove_success(True)
|
|
|
|
if self.account in app.connections and \
|
|
app.connections[self.account].connected:
|
|
ConfirmationDialog(
|
|
_('Account "%s" is connected to the server') % self.account,
|
|
_('If you remove it, the connection will be lost.'),
|
|
on_response_ok=remove,
|
|
transient_for=self.window)
|
|
else:
|
|
remove()
|
|
|
|
def on_remove_responce_ok(self, is_checked):
|
|
if is_checked[0]:
|
|
self._on_remove_success(True)
|
|
|
|
def _on_remove_success(self, res):
|
|
# action of unregistration has failed, we don't remove the account
|
|
# Error message is send by connect_and_auth()
|
|
if not res:
|
|
ConfirmationDialogDoubleRadio(
|
|
_('Connection to server %s failed') % self.account,
|
|
_('What would you like to do?'),
|
|
_('Remove only from Gajim'),
|
|
_('Don\'t remove anything. I\'ll try again later'),
|
|
on_response_ok=self.on_remove_responce_ok, is_modal=False,
|
|
transient_for=self.window)
|
|
return
|
|
# Close all opened windows
|
|
app.interface.roster.close_all(self.account, force=True)
|
|
if self.account in app.connections:
|
|
app.connections[self.account].disconnect(on_purpose=True)
|
|
app.connections[self.account].cleanup()
|
|
del app.connections[self.account]
|
|
app.logger.remove_roster(app.get_jid_from_account(self.account))
|
|
app.config.del_per('accounts', self.account)
|
|
del app.interface.instances[self.account]
|
|
if self.account in app.nicks:
|
|
del app.interface.minimized_controls[self.account]
|
|
del app.nicks[self.account]
|
|
del app.block_signed_in_notifications[self.account]
|
|
del app.groups[self.account]
|
|
app.contacts.remove_account(self.account)
|
|
del app.gc_connected[self.account]
|
|
del app.automatic_rooms[self.account]
|
|
del app.to_be_removed[self.account]
|
|
del app.newly_added[self.account]
|
|
del app.sleeper_state[self.account]
|
|
del app.last_message_time[self.account]
|
|
del app.status_before_autoaway[self.account]
|
|
del app.gajim_optional_features[self.account]
|
|
del app.caps_hash[self.account]
|
|
if len(app.connections) >= 2: # Do not merge accounts if only one exists
|
|
app.interface.roster.regroup = app.config.get('mergeaccounts')
|
|
else:
|
|
app.interface.roster.regroup = False
|
|
app.interface.roster.setup_and_draw_roster()
|
|
app.app.remove_account_actions(self.account)
|
|
gui_menu_builder.build_accounts_menu()
|
|
|
|
window = app.get_app_window('AccountsWindow')
|
|
if window is not None:
|
|
window.remove_account(self.account)
|
|
self.window.destroy()
|
|
|
|
|
|
class ManageSoundsWindow:
|
|
def __init__(self, transient):
|
|
self._builder = gtkgui_helpers.get_gtk_builder(
|
|
'manage_sounds_window.ui')
|
|
self.window = self._builder.get_object('manage_sounds_window')
|
|
self.window.set_transient_for(transient)
|
|
|
|
self.sound_button = self._builder.get_object('filechooser')
|
|
|
|
filter_ = Gtk.FileFilter()
|
|
filter_.set_name(_('All files'))
|
|
filter_.add_pattern('*')
|
|
self.sound_button.add_filter(filter_)
|
|
|
|
filter_ = Gtk.FileFilter()
|
|
filter_.set_name(_('Wav Sounds'))
|
|
filter_.add_pattern('*.wav')
|
|
self.sound_button.add_filter(filter_)
|
|
self.sound_button.set_filter(filter_)
|
|
|
|
self.sound_tree = self._builder.get_object('sounds_treeview')
|
|
|
|
self._fill_sound_treeview()
|
|
|
|
self._builder.connect_signals(self)
|
|
|
|
self.window.show_all()
|
|
|
|
def _on_row_changed(self, model, path, iter_):
|
|
sound_event = model[iter_][3]
|
|
app.config.set_per('soundevents', sound_event,
|
|
'enabled', bool(model[path][0]))
|
|
app.config.set_per('soundevents', sound_event,
|
|
'path', model[iter_][2])
|
|
|
|
def _on_toggle(self, cell, path):
|
|
if self.sound_button.get_filename() is None:
|
|
return
|
|
model = self.sound_tree.get_model()
|
|
model[path][0] = not model[path][0]
|
|
|
|
def _fill_sound_treeview(self):
|
|
model = self.sound_tree.get_model()
|
|
model.clear()
|
|
|
|
# NOTE: sounds_ui_names MUST have all items of
|
|
# sounds = app.config.get_per('soundevents') as keys
|
|
sounds_dict = {
|
|
'attention_received': _('Attention Message Received'),
|
|
'first_message_received': _('First Message Received'),
|
|
'next_message_received_focused': _('Next Message Received Focused'),
|
|
'next_message_received_unfocused': _('Next Message Received Unfocused'),
|
|
'contact_connected': _('Contact Connected'),
|
|
'contact_disconnected': _('Contact Disconnected'),
|
|
'message_sent': _('Message Sent'),
|
|
'muc_message_highlight': _('Group Chat Message Highlight'),
|
|
'muc_message_received': _('Group Chat Message Received'),
|
|
}
|
|
|
|
for config_name, sound_name in sounds_dict.items():
|
|
enabled = app.config.get_per('soundevents', config_name, 'enabled')
|
|
path = app.config.get_per('soundevents', config_name, 'path')
|
|
model.append((enabled, sound_name, path, config_name))
|
|
|
|
def _on_cursor_changed(self, treeview):
|
|
model, iter_ = treeview.get_selection().get_selected()
|
|
path_to_snd_file = helpers.check_soundfile_path(model[iter_][2])
|
|
if path_to_snd_file is None:
|
|
self.sound_button.unselect_all()
|
|
else:
|
|
self.sound_button.set_filename(path_to_snd_file)
|
|
|
|
def _on_file_set(self, button):
|
|
model, iter_ = self.sound_tree.get_selection().get_selected()
|
|
|
|
filename = button.get_filename()
|
|
directory = os.path.dirname(filename)
|
|
app.config.set('last_sounds_dir', directory)
|
|
path_to_snd_file = helpers.strip_soundfile_path(filename)
|
|
|
|
# set new path to sounds_model
|
|
model[iter_][2] = path_to_snd_file
|
|
# set the sound to enabled
|
|
model[iter_][0] = True
|
|
|
|
def _on_clear(self, *args):
|
|
self.sound_button.unselect_all()
|
|
model, iter_ = self.sound_tree.get_selection().get_selected()
|
|
model[iter_][2] = ''
|
|
model[iter_][0] = False
|
|
|
|
def _on_play(self, *args):
|
|
model, iter_ = self.sound_tree.get_selection().get_selected()
|
|
snd_event_config_name = model[iter_][3]
|
|
helpers.play_sound(snd_event_config_name)
|
|
|
|
def _on_destroy(self, *args):
|
|
self.window.destroy()
|
|
window = app.get_app_window('Preferences')
|
|
if window is not None:
|
|
window.sounds_preferences = None
|