fceux/gfceux

637 lines
20 KiB
Plaintext
Raw Normal View History

2006-07-30 04:00:49 +00:00
#!/usr/bin/python
2008-08-03 00:50:40 +00:00
# gfceux - Graphical launcher for fceux.
2006-07-30 04:00:49 +00:00
# Designed on Ubuntu, with platfrom independence in mind.
version = "2.0.2"
2008-08-03 00:50:40 +00:00
title = "gfceux"
# Copyright (C) 2008 Lukas Sabota <ltsmooth42 _at_ gmail.com>
2006-07-30 04:00:49 +00:00
##
"""
This program 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 2
of the License, or (at your option) any later version.
This program 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 this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
"""
2006-10-14 22:13:58 +00:00
# # # # # # # #
2006-07-30 04:00:49 +00:00
# Python imports
2006-10-14 22:13:58 +00:00
2006-07-30 04:00:49 +00:00
import sys
import os
import pickle
2006-10-14 21:41:08 +00:00
import shutil
2006-07-30 04:00:49 +00:00
from optparse import OptionParser
2008-06-24 16:43:26 +00:00
#from subprocess import Popen
2006-07-30 04:00:49 +00:00
try:
2008-06-17 16:02:45 +00:00
import pygtk
pygtk.require("2.0")
2006-07-30 04:00:49 +00:00
import gtk
except ImportError:
2008-06-17 16:02:45 +00:00
print "The PyGTK libraries cannot be found.\n\
Ensure that PyGTK (>=2.12) is installed on this system.\n\
2006-07-30 04:00:49 +00:00
On Debian based systems (like Ubuntu), try this command:\n\
2008-06-17 16:02:45 +00:00
sudo apt-get install python-gtk2 libgtk2.0-0"
2008-08-21 05:07:56 +00:00
2008-06-17 16:02:45 +00:00
2008-07-02 05:58:29 +00:00
class GameOptions:
# sound
2006-07-30 04:00:49 +00:00
sound_check = True
soundq_check = True
soundrate_entry = "11000"
soundbufsize_entry = "48"
# video
2006-07-30 04:00:49 +00:00
fullscreen_check = False
xscale_spin = 2
yscale_spin = 2
bpp_combo = 32
2006-07-30 04:00:49 +00:00
opengl_check = False
2008-07-02 05:58:29 +00:00
autoscale_check = True
2008-06-17 16:02:45 +00:00
# main
extra_entry = ''
romfile = ''
moviefile = ''
luafile = ''
2008-06-17 16:02:45 +00:00
# network
2006-07-30 04:00:49 +00:00
join_radio = False
join_add = ''
join_port = 4046
join_pass = ''
host_radio = False
host_port = 4046
host_pass = ''
no_network_radio = True
2006-10-14 22:13:58 +00:00
2006-07-30 04:00:49 +00:00
def load_options():
global options, optionsfile
try:
ifile = file(optionsfile, 'r')
options = pickle.load(ifile)
pickle.load(ifile)
except:
return
ifile.close()
def save_options():
global options, optionsfile
if os.path.exists(os.path.dirname(optionsfile)) == 0:
os.mkdir(os.path.dirname(optionsfile))
ofile = open(optionsfile, 'w')
2006-07-30 04:00:49 +00:00
pickle.dump(options, ofile)
ofile.close()
def give_widgets():
"""
give_widgets()
This function takes data from the options struct and relays it to
the GTK window
"""
2008-06-17 16:02:45 +00:00
global options, widgets
2006-07-30 04:00:49 +00:00
try:
2008-06-17 16:02:45 +00:00
widgets.get_object("rom_entry").set_text(options.romfile)
widgets.get_object("movie_entry").set_text(options.moviefile)
widgets.get_object("lua_entry").set_text(options.luafile)
# sound
2008-06-17 16:02:45 +00:00
widgets.get_object("sound_check").set_active(options.sound_check)
widgets.get_object("soundq_check").set_active(options.soundq_check)
widgets.get_object("soundrate_entry").set_text(options.soundrate_entry)
widgets.get_object("soundbufsize_entry").set_text(options.soundbufsize_entry)
2006-07-30 04:00:49 +00:00
# video
2008-06-17 16:02:45 +00:00
widgets.get_object("fullscreen_check").set_active(options.fullscreen_check)
widgets.get_object("opengl_check").set_active(options.opengl_check)
2008-07-02 05:58:29 +00:00
widgets.get_object("autoscale_check").set_active(options.autoscale_check)
2008-06-17 16:02:45 +00:00
widgets.get_object("xscale_spin").set_value(options.xscale_spin)
widgets.get_object("yscale_spin").set_value(options.yscale_spin)
2006-07-30 04:00:49 +00:00
2008-06-17 16:02:45 +00:00
widgets.get_object("extra_entry").set_text(options.extra_entry)
2006-07-30 04:00:49 +00:00
# Usability point:
# Users will probably not want to remember their previous network setting.
# Users may accidently be connecting to a remote server/hosting a game when
# they were unaware.
# No network is being set by default
2008-06-17 16:02:45 +00:00
widgets.get_object("no_network_radio").set_active(True)
widgets.get_object("join_add").set_text(options.join_add)
widgets.get_object("join_port").set_value(float(options.join_port))
widgets.get_object("join_pass").set_text(options.join_pass)
widgets.get_object("host_port").set_value(float(options.host_port))
widgets.get_object("host_pass").set_text(options.host_pass)
2006-07-30 04:00:49 +00:00
except AttributeError:
# When new widgets are added, old pickle files might break.
2008-07-02 05:58:29 +00:00
options = GameOptions()
give_widgets()
2006-10-14 21:41:08 +00:00
2006-07-30 04:00:49 +00:00
def set_options():
"""
set_options()
This function grabs all of the data from the GTK widgets
and stores it in the options object.
"""
2008-06-17 16:02:45 +00:00
options.romfile = widgets.get_object("rom_entry").get_text()
options.moviefile = widgets.get_object("movie_entry").get_text()
options.luafile = widgets.get_object("lua_entry").get_text()
# sound
2008-06-17 16:02:45 +00:00
options.sound_check = widgets.get_object("sound_check").get_active()
options.soundq_check = widgets.get_object("soundq_check").get_active()
options.soundrate_entry = widgets.get_object("soundrate_entry").get_text()
options.soundbufsize_entry = widgets.get_object("soundbufsize_entry").get_text()
2006-07-30 04:00:49 +00:00
# video
2008-06-17 16:02:45 +00:00
options.fullscreen_check = widgets.get_object("fullscreen_check").get_active()
options.opengl_check = widgets.get_object("opengl_check").get_active()
2008-07-02 05:58:29 +00:00
options.autoscale_check = widgets.get_object("autoscale_check").get_active()
2008-06-17 16:02:45 +00:00
options.xscale_spin = widgets.get_object("xscale_spin").get_value()
options.yscale_spin = widgets.get_object("yscale_spin").get_value()
2006-07-30 04:00:49 +00:00
2008-06-17 16:02:45 +00:00
options.extra_entry = widgets.get_object("extra_entry").get_text()
2006-07-30 04:00:49 +00:00
2008-06-17 16:02:45 +00:00
options.join_radio = widgets.get_object("join_radio").get_active()
options.host_radio = widgets.get_object("host_radio").get_active()
options.no_network_radio = widgets.get_object("no_network_radio").get_active()
options.join_add = widgets.get_object("join_add").get_text()
options.join_port = int(widgets.get_object("join_port").get_value())
2008-06-17 16:02:45 +00:00
options.join_pass = widgets.get_object("join_pass").get_text()
options.host_port = widgets.get_object("host_port").get_value()
options.host_pass = widgets.get_object("host_pass").get_text()
2006-07-30 04:00:49 +00:00
2008-06-06 00:32:42 +00:00
def find_binary(file):
# first check the script directory
if os.path.isfile(os.path.join(os.path.dirname(sys.argv[0]),file)):
return os.path.join(os.path.dirname(sys.argv[0]), file)
2008-06-06 00:32:42 +00:00
# if not in the script directory, check the $PATH
2006-07-30 04:00:49 +00:00
path = os.getenv('PATH')
directories= []
directory = ''
# check for '$' so last entry is processed
for x in path + '$':
if x != ':' and x != '$':
directory = directory + x
else:
directories.append(directory)
directory = ''
for x in directories:
2008-06-06 00:32:42 +00:00
if os.path.isfile(os.path.join(x, file)):
return os.path.join(x,file)
2006-07-30 04:00:49 +00:00
return None
2008-06-17 16:02:45 +00:00
2008-06-24 16:43:26 +00:00
# # # # # # # # # # # # # # # # #
2008-06-17 16:02:45 +00:00
# Globals
options = None
optionsfile = os.getenv('HOME') + '/.fceultra/gfceux_options.dat'
2008-06-17 16:02:45 +00:00
widgets = None
class GfceuApp:
def __init__(self):
self.fceux_binary = self.find_fceu()
self.load_ui()
2008-07-02 05:58:29 +00:00
options = GameOptions()
2008-06-17 16:02:45 +00:00
load_options()
give_widgets()
try:
gtk.main()
except KeyboardInterrupt:
sys.exit(0)
def msg(self, text, use_gtk=False):
"""
GfceuApp.msg()
This function prints messages to the user. This is generally used for status
messages. If a GTK message_box is requried, the use_gtk flag can be enabled.
"""
print text
if use_gtk:
msgbox = gtk.MessageDialog(parent=None, flags=0, type=gtk.MESSAGE_INFO,
buttons=gtk.BUTTONS_CLOSE)
msgbox.set_markup(text)
msgbox.run()
msgbox.destroy()
def print_error(self, message, code, use_gtk=True, fatal=True):
"""
GfceuApp.print_error()
2008-06-17 16:02:45 +00:00
Presents the user with an error message and optionally quits the program.
"""
print title + ' error code '+str(code)+': ' + message
if use_gtk:
msgbox = gtk.MessageDialog(parent=None, flags=0, type=gtk.MESSAGE_ERROR,
buttons=gtk.BUTTONS_CLOSE)
msgbox.set_markup(title + ' ERROR Code '+str(code)+':\n'+message)
msgbox.run()
msgbox.destroy()
if fatal:
sys.exit(code)
2008-06-17 16:02:45 +00:00
def find_fceu(self):
bin = find_binary('fceux')
if bin == None:
2008-08-24 07:23:24 +00:00
self.print_error('Could not find the fceux binary.\n\
Ensure that fceux is installed and in the $PATH.\n', 4, True)
2008-06-17 16:02:45 +00:00
else:
self.msg('Using: ' + bin)
return bin
def load_ui(self):
global widgets
""" Search for the glade XML file and load it """
# Check first in the directory of this script.
2008-08-03 00:50:40 +00:00
if os.path.isfile('gfceux.xml'):
glade_file = 'gfceux.xml'
2008-06-17 16:02:45 +00:00
# Then check to see if its installed on a *nix system
2008-08-03 00:50:40 +00:00
elif os.path.isfile(os.path.join(os.path.dirname(sys.argv[0]), '../share/gfceux/gfceux.xml')):
glade_file = os.path.join(os.path.dirname(sys.argv[0]), '../share/gfceux/gfceux.xml')
2008-06-17 16:02:45 +00:00
else:
print 'ERROR.'
print 'Could not find the ' + glade_file + ' file.'
2008-06-17 16:02:45 +00:00
print 'Try reinstalling the application.'
sys.exit(1)
2008-06-29 01:13:50 +00:00
try:
print "Using: " + glade_file
widgets = gtk.Builder()
widgets.add_from_file(glade_file)
widgets.connect_signals(self)
except:
self.print_error("Couldn't load the UI data.", 24)
2008-06-17 16:02:45 +00:00
widgets.get_object("main_window").show_all()
def launch(self, rom_name, local=False):
2006-10-14 22:13:58 +00:00
global options
2008-06-17 16:02:45 +00:00
set_options()
sound_options = ''
if options.sound_check:
sound_options += '--sound 1 '
else:
sound_options += '--sound 0 '
if options.soundq_check:
sound_options += '--soundq 1 '
else:
sound_options += '--soundq 0 '
if options.soundrate_entry:
sound_options += '--soundrate ' + options.soundrate_entry + ' '
if options.soundbufsize_entry:
sound_options += '--soundbufsize ' + options.soundbufsize_entry + ' '
2008-06-17 16:02:45 +00:00
# video
video_options = ''
if options.fullscreen_check:
video_options += '--fullscreen 1 '
else:
video_options += '--fullscreen 0 '
if options.opengl_check:
video_options += '--opengl 1 '
else:
video_options += '--opengl 0 '
2008-07-02 05:58:29 +00:00
if options.autoscale_check:
video_options += '--autoscale 1 '
else:
video_options += '--autoscale 0 '
2008-06-17 16:02:45 +00:00
video_options += ' --xscale ' + str(options.xscale_spin)
video_options += ' --yscale ' + str(options.yscale_spin)
video_options += ' '
# lua/movie
other_options = ''
if options.luafile:
other_options += '--loadlua ' + options.luafile + ' '
if options.moviefile:
other_options += '--playmov ' + options.moviefile + ' '
2008-06-17 16:02:45 +00:00
# Netplay is fucked right now
2008-06-17 16:02:45 +00:00
if options.join_radio:
if options.join_pass == '':
netpass = ''
else:
netpass = '--pass ' + options.join_pass
network = '--net ' + options.join_add +\
2008-06-17 16:02:45 +00:00
' --port '+ str(options.join_port) + ' ' + netpass
else:
network = ''
2008-06-17 16:02:45 +00:00
if options.host_radio:
"""
2008-06-17 16:02:45 +00:00
if options.host_pass == '':
netpass = ' '
else:
# netpass = ' --pass ' + '"' + options.host_pass + '" '
netpass = ' --pass ' + options.host_pass
2008-06-17 16:02:45 +00:00
network = '--net localhost --port '+\
str(options.host_port) + netpass + ' '
"""
network = ''
2008-06-17 16:02:45 +00:00
if local:
network = ''
2008-06-17 16:02:45 +00:00
command = self.fceux_binary + ' ' + sound_options + video_options +\
network + other_options + options.extra_entry + ' '+ rom_name
2008-06-17 16:02:45 +00:00
self.msg('Command: ' + command)
# more code to disable because netplay is fucked
"""
2008-06-17 16:02:45 +00:00
if options.host_radio:
xterm_binary = find_binary("xterm")
if xterm_binary == None:
gfceu_error("Cannot find xterm on this system. You will not \n\
be informed of server output.", 102, True, False)
args = [self.server_binary]
else:
args = [xterm_binary, "-e", self.server_binary]
args.append('--port')
args.append(str(options.host_port))
if options.host_pass:
args.append("--password")
args.append(options.host_pass)
pid = Popen(args).pid
"""
widgets.get_object("main_window").hide()
# os.system() is a blocker, so we must force
# gtk to process our events.
while gtk.events_pending():
gtk.main_iteration_do()
os.system(command)
widgets.get_object("main_window").show()
# another part of netplay code
"""
if options.host_radio:
os.kill(pid, 9)
"""
2008-06-17 16:02:45 +00:00
### Callbacks
def launch_button_clicked(self, arg1):
global options
options.romfile = widgets.get_object("rom_entry").get_text()
if widgets.get_object("rom_entry").get_text() == '':
self.msg('Please specify a ROM to open in the main tab.', True)
2006-07-30 04:00:49 +00:00
return
2006-10-14 22:13:58 +00:00
self.launch('"'+ options.romfile +'"')
2008-06-17 16:02:45 +00:00
def about_button_clicked(self, menuitem, data=None):
2008-08-13 23:45:12 +00:00
widgets.get_object("about_dialog").set_name('GFCE UltraX '+version)
2008-06-17 16:02:45 +00:00
widgets.get_object("about_dialog").run()
widgets.get_object("about_dialog").hide()
def lua_browse_button_clicked(self, menuitem, data=None):
global options
set_options()
chooser = gtk.FileChooserDialog("Open...", None,
gtk.FILE_CHOOSER_ACTION_OPEN,
(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
gtk.STOCK_OPEN, gtk.RESPONSE_OK))
chooser.set_property("local-only", False)
chooser.set_default_response(gtk.RESPONSE_OK)
filter=gtk.FileFilter()
filter.set_name("Lua scripts")
filter.add_pattern("*.lua")
chooser.add_filter(filter)
filter = gtk.FileFilter()
filter.set_name("All files")
filter.add_pattern("*")
chooser.add_filter(filter)
if options.luafile == '':
folder = os.getenv('HOME')
else:
folder = os.path.split(options.luafile)[0]
chooser.set_current_folder (folder)
response = chooser.run()
chooser.hide()
if response == gtk.RESPONSE_OK:
if chooser.get_filename():
x = chooser.get_filename()
widgets.get_object("lua_entry").set_text(x)
options.luafile = x
def movie_browse_button_clicked(self, menuitem, data=None):
global options
set_options()
chooser = gtk.FileChooserDialog("Open...", None,
gtk.FILE_CHOOSER_ACTION_OPEN,
(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
gtk.STOCK_OPEN, gtk.RESPONSE_OK))
chooser.set_property("local-only", False)
chooser.set_default_response(gtk.RESPONSE_OK)
filter=gtk.FileFilter()
filter.set_name("FM2 movies")
filter.add_pattern("*.fm2")
chooser.add_filter(filter)
filter = gtk.FileFilter()
filter.set_name("All files")
filter.add_pattern("*")
chooser.add_filter(filter)
if options.moviefile == '':
folder = os.getenv('HOME')
else:
folder = os.path.split(options.moviefile)[0]
chooser.set_current_folder (folder)
response = chooser.run()
chooser.hide()
if response == gtk.RESPONSE_OK:
if chooser.get_filename():
x = chooser.get_filename()
widgets.get_object("movie_entry").set_text(x)
options.moviefile = x
def rom_browse_button_clicked(self, menuitem, data=None):
2008-06-17 16:02:45 +00:00
global options
2006-10-14 22:13:58 +00:00
set_options()
chooser = gtk.FileChooserDialog("Open...", None,
gtk.FILE_CHOOSER_ACTION_OPEN,
(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
gtk.STOCK_OPEN, gtk.RESPONSE_OK))
chooser.set_property("local-only", False)
2006-10-14 22:13:58 +00:00
chooser.set_default_response(gtk.RESPONSE_OK)
2006-07-30 04:00:49 +00:00
2006-10-14 22:13:58 +00:00
filter=gtk.FileFilter()
filter.set_name("NES Roms")
filter.add_mime_type("application/x-nes-rom")
filter.add_mime_type("application/zip")
filter.add_pattern("*.nes")
filter.add_pattern("*.zip")
chooser.add_filter(filter)
filter = gtk.FileFilter()
filter.set_name("All files")
filter.add_pattern("*")
chooser.add_filter(filter)
2006-07-30 04:00:49 +00:00
2006-10-14 22:13:58 +00:00
if options.romfile == '':
folder = os.getenv('HOME')
else:
folder = os.path.split(options.romfile)[0]
chooser.set_current_folder (folder)
response = chooser.run()
chooser.hide()
if response == gtk.RESPONSE_OK:
if chooser.get_filename():
x = chooser.get_filename()
2008-06-17 16:02:45 +00:00
widgets.get_object("rom_entry").set_text(x)
# reset lua and movie entries on rom change
widgets.get_object("movie_entry").set_text("")
widgets.get_object("lua_entry").set_text("")
2006-10-14 22:13:58 +00:00
options.romfile = x
2008-08-21 05:07:56 +00:00
# fix this global its ugly
gamepad_config = "0"
def gamepad_clicked_new(self, widget, data=None):
widgets.get_object("gamepad_config_window").show_all()
d = {'gp1_button' : "0",
'gp2_button' : "1",
'gp3_button' : "2",
'gp4_button' : "3"}
self.gamepad_config = d[widget.name]
def right_button_clicked(self, widget, data=None):
print "right!"
def left_button_clicked(self, widget, data=None):
print "left!"
def down_button_clicked(self, widget, data=None):
print "down!"
def up_button_clicked(self, widget, data=None):
print "up!"
def button_clicked(self, widget, data=None):
d = {'right_button' : "SDL.Input.GamePad." + self.gamepad_config + "Right",
'left_button' : "SDL.Input.GamePad." + self.gamepad_config + "Left",
'up_button' : "SDL.Input.GamePad." + self.gamepad_config + "Up",
'down_button' : "SDL.Input.GamePad." + self.gamepad_config + "Down",
'select_button' : "SDL.Input.GamePad." + self.gamepad_config + "Select",
'start_button' : "SDL.Input.GamePad." + self.gamepad_config + "Start",
'a_button' : "SDL.Input.GamePad." + self.gamepad_config + "A",
'b_button' : "SDL.Input.GamePad." + self.gamepad_config + "B"}
print d[widget.name]
2008-06-17 16:02:45 +00:00
def gamepad_clicked(self, widget, data=None):
2006-10-14 22:13:58 +00:00
d = {'gp1_button' : '1',
'gp2_button' : '2',
'gp3_button' : '3',
'gp4_button' : '4'}
command = '-inputcfg gamepad' + d[widget.name] + ' /dev/null'
2008-06-17 17:36:01 +00:00
self.launch(command, True)
2006-10-14 22:13:58 +00:00
2008-06-17 16:02:45 +00:00
def config_help_button_clicked(self, menuitem, data=None):
2006-10-14 22:13:58 +00:00
msgbox = gtk.MessageDialog(parent=None, flags=0,
type=gtk.MESSAGE_INFO, buttons=gtk.BUTTONS_CLOSE)
msgbox.set_markup("Once a gamepad is seleceted, a titlebar will be displayed\
indicating a NES button. Press the button or key you would like to have\
associated with the button indicated on the titlebar. This process\
will repeat until all buttons on the gamepad are configured.")
msgbox.run()
msgbox.hide()
2006-07-30 04:00:49 +00:00
2008-06-17 16:02:45 +00:00
def join_radio_clicked(self, menuitem, data=None):
2006-10-14 22:13:58 +00:00
global options
2008-06-17 16:02:45 +00:00
widgets.get_object("join_frame").set_sensitive(True)
widgets.get_object("host_frame").set_sensitive(False)
2006-10-14 22:13:58 +00:00
options.join_radio = True
options.host_radio = False
options.no_network_radio = False
2006-07-30 04:00:49 +00:00
2008-06-17 16:02:45 +00:00
def host_radio_clicked(self, menuitem, data=None):
if widgets.get_object("host_radio").get_active():
self.server_binary = find_binary('fceu-server')
if self.server_binary == None:
2006-10-14 22:13:58 +00:00
if os.name == 'nt':
self.print_error("The fceu server software cannot be found. \n\
2006-10-14 22:13:58 +00:00
Ensure that it is installed in the same directory as \n\
GFCE Ultra.", 102, True, False)
else:
self.print_error("The fceu server software cannot be found on \n\
2006-10-14 22:13:58 +00:00
this system. Ensure that it is installed and in your path.",
101, True, False)
2008-06-17 16:02:45 +00:00
widgets.get_object("no_network_radio").set_active(True)
2006-10-14 22:13:58 +00:00
options.no_network_radio = True
return False
2008-06-17 16:02:45 +00:00
widgets.get_object("join_frame").set_sensitive(False)
widgets.get_object("host_frame").set_sensitive(True)
2006-10-14 22:13:58 +00:00
options.join_radio = False
options.host_radio = True
options.no_network_radio = False
2008-06-17 16:02:45 +00:00
def no_network_radio_clicked(self, menuitem, data=None):
widgets.get_object("join_frame").set_sensitive(False)
widgets.get_object("host_frame").set_sensitive(False)
2006-07-30 04:00:49 +00:00
options.join_radio = False
2006-10-14 22:13:58 +00:00
options.host_radio = False
options.no_network_radio = True
2008-06-17 16:02:45 +00:00
def end(self, menuitem, data=None):
2006-10-14 22:13:58 +00:00
set_options()
save_options()
gtk.main_quit()
2006-07-30 04:00:49 +00:00
2006-10-14 22:13:58 +00:00
2006-07-30 04:00:49 +00:00
if __name__ == '__main__':
parser = OptionParser(version='%prog '+ version)
parser.parse_args()
2008-06-17 16:02:45 +00:00
app = GfceuApp()