2014-06-30 04:17:04 +02:00
/ *
* opsu ! - an open - source osu ! client
2015-01-16 18:05:44 +01:00
* Copyright ( C ) 2014 , 2015 Jeffrey Han
2014-06-30 04:17:04 +02:00
*
* opsu ! 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 .
*
* opsu ! 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 opsu ! . If not , see < http : //www.gnu.org/licenses/>.
* /
2015-01-21 05:56:10 +01:00
package itdelatrisu.opsu ;
2014-06-30 04:17:04 +02:00
2016-12-12 19:48:03 +01:00
import awlex.ospu.polymover.factory.PolyMoverFactory ;
2015-08-29 04:29:21 +02:00
import itdelatrisu.opsu.audio.MusicController ;
2017-01-21 14:41:51 +01:00
import itdelatrisu.opsu.audio.SoundController ;
2015-08-29 04:29:21 +02:00
import itdelatrisu.opsu.beatmap.Beatmap ;
2016-11-19 22:00:29 +01:00
import itdelatrisu.opsu.beatmap.TimingPoint ;
2015-08-29 04:29:21 +02:00
import itdelatrisu.opsu.skins.Skin ;
import itdelatrisu.opsu.skins.SkinLoader ;
2016-12-12 19:48:03 +01:00
import itdelatrisu.opsu.states.Game ;
2015-08-29 04:29:21 +02:00
import itdelatrisu.opsu.ui.Fonts ;
2014-06-30 04:17:04 +02:00
import java.io.BufferedReader ;
import java.io.BufferedWriter ;
import java.io.File ;
import java.io.FileOutputStream ;
import java.io.FileReader ;
import java.io.IOException ;
import java.io.OutputStreamWriter ;
2015-01-16 19:42:54 +01:00
import java.net.URI ;
2014-06-30 04:17:04 +02:00
import java.text.SimpleDateFormat ;
2016-09-27 17:36:10 +02:00
import java.util.* ;
2014-07-09 04:17:48 +02:00
import java.util.concurrent.TimeUnit ;
2015-08-27 21:14:04 +02:00
import java.util.jar.Attributes ;
import java.util.jar.JarFile ;
import java.util.jar.Manifest ;
2015-11-01 03:20:08 +01:00
import java.util.regex.Matcher ;
import java.util.regex.Pattern ;
2014-06-30 04:17:04 +02:00
2014-07-18 05:58:37 +02:00
import org.lwjgl.input.Keyboard ;
2014-06-30 04:17:04 +02:00
import org.newdawn.slick.Input ;
import org.newdawn.slick.SlickException ;
2016-12-11 16:32:21 +01:00
import org.newdawn.slick.openal.SoundStore ;
2015-05-24 05:48:28 +02:00
import org.newdawn.slick.util.ClasspathLocation ;
import org.newdawn.slick.util.FileSystemLocation ;
2014-06-30 04:17:04 +02:00
import org.newdawn.slick.util.Log ;
2015-05-24 05:48:28 +02:00
import org.newdawn.slick.util.ResourceLoader ;
2014-06-30 04:17:04 +02:00
2015-11-01 03:20:08 +01:00
import com.sun.jna.platform.win32.Advapi32Util ;
import com.sun.jna.platform.win32.Win32Exception ;
import com.sun.jna.platform.win32.WinReg ;
2016-10-01 11:53:33 +02:00
import yugecin.opsudance.* ;
2017-01-17 23:18:12 +01:00
import yugecin.opsudance.core.DisplayContainer ;
2017-01-21 01:16:27 +01:00
import yugecin.opsudance.core.errorhandling.ErrorHandler ;
import yugecin.opsudance.core.events.EventBus ;
2017-01-21 23:52:19 +01:00
import yugecin.opsudance.events.BarNotificationEvent ;
2017-01-17 23:18:12 +01:00
import yugecin.opsudance.events.BubbleNotificationEvent ;
2017-01-21 15:30:07 +01:00
import yugecin.opsudance.events.ResolutionOrSkinChangedEvent ;
2017-01-01 23:47:11 +01:00
import yugecin.opsudance.movers.factories.ExgonMoverFactory ;
2016-12-04 13:40:41 +01:00
import yugecin.opsudance.movers.factories.QuadraticBezierMoverFactory ;
2016-12-04 15:32:38 +01:00
import yugecin.opsudance.movers.slidermovers.DefaultSliderMoverController ;
2017-02-05 23:36:39 +01:00
import yugecin.opsudance.utils.CachedVariable ;
2015-11-01 03:20:08 +01:00
2014-06-30 04:17:04 +02:00
/ * *
2015-01-21 05:56:10 +01:00
* Handles all user options .
2014-06-30 04:17:04 +02:00
* /
2015-01-21 05:56:10 +01:00
public class Options {
2015-08-27 21:14:04 +02:00
/** Whether to use XDG directories. */
2015-09-03 02:23:23 +02:00
public static final boolean USE_XDG = checkXDGFlag ( ) ;
2015-08-27 21:14:04 +02:00
2015-02-13 21:03:17 +01:00
/** The config directory. */
private static final File CONFIG_DIR = getXDGBaseDir ( " XDG_CONFIG_HOME " , " .config " ) ;
/** The data directory. */
private static final File DATA_DIR = getXDGBaseDir ( " XDG_DATA_HOME " , " .local/share " ) ;
2015-08-30 21:31:01 +02:00
/** The cache directory. */
private static final File CACHE_DIR = getXDGBaseDir ( " XDG_CACHE_HOME " , " .cache " ) ;
2015-01-22 06:44:45 +01:00
/** File for logging errors. */
2015-02-13 21:03:17 +01:00
public static final File LOG_FILE = new File ( CONFIG_DIR , " .opsu.log " ) ;
2014-06-30 04:17:04 +02:00
2015-01-22 06:44:45 +01:00
/** File for storing user options. */
2015-02-13 21:03:17 +01:00
private static final File OPTIONS_FILE = new File ( CONFIG_DIR , " .opsu.cfg " ) ;
2014-07-02 07:53:42 +02:00
2015-11-01 03:20:08 +01:00
/** The default beatmap directory (unless an osu! installation is detected). */
private static final File BEATMAP_DIR = new File ( DATA_DIR , " Songs/ " ) ;
2014-06-30 04:17:04 +02:00
2015-11-01 03:20:08 +01:00
/** The default skin directory (unless an osu! installation is detected). */
private static final File SKIN_ROOT_DIR = new File ( DATA_DIR , " Skins/ " ) ;
2015-05-24 05:48:28 +02:00
2015-03-05 03:03:06 +01:00
/** Cached beatmap database name. */
2015-05-17 03:25:19 +02:00
public static final File BEATMAP_DB = new File ( DATA_DIR , " .opsu.db " ) ;
2015-03-05 03:03:06 +01:00
2015-01-28 09:47:24 +01:00
/** Score database name. */
2015-02-13 21:03:17 +01:00
public static final File SCORE_DB = new File ( DATA_DIR , " .opsu_scores.db " ) ;
2015-01-28 09:47:24 +01:00
2015-08-30 21:31:01 +02:00
/** Directory where natives are unpacked. */
public static final File NATIVE_DIR = new File ( CACHE_DIR , " Natives/ " ) ;
2016-10-13 10:12:47 +02:00
/** Directory where temporary files are stored (deleted on exit). */
public static final File TEMP_DIR = new File ( CACHE_DIR , " Temp/ " ) ;
2015-01-22 06:44:45 +01:00
/** Font file name. */
2015-05-15 07:33:53 +02:00
public static final String FONT_NAME = " DroidSansFallback.ttf " ;
2014-08-25 18:47:10 +02:00
2015-03-07 10:17:19 +01:00
/** Version file name. */
public static final String VERSION_FILE = " version " ;
2015-01-22 06:44:45 +01:00
/** Repository address. */
2015-03-07 10:17:19 +01:00
public static final URI REPOSITORY_URI = URI . create ( " https://github.com/itdelatrisu/opsu " ) ;
2015-01-16 19:42:54 +01:00
2016-09-27 22:44:35 +02:00
/** Dance repository address. */
public static final URI DANCE_REPOSITORY_URI = URI . create ( " https://github.com/yugecin/opsu-dance " ) ;
2015-01-22 06:44:45 +01:00
/** Issue reporting address. */
2016-10-04 13:05:10 +02:00
public static final String ISSUES_URL = " https://github.com/yugecin/opsu-dance/issues/new?title=%s&body=%s " ;
2015-03-07 10:17:19 +01:00
/** Address containing the latest version file. */
2016-10-04 13:05:10 +02:00
public static final String VERSION_REMOTE = " https://raw.githubusercontent.com/yugecin/opsu-dance/master/version " ;
2015-01-16 19:42:54 +01:00
2015-01-22 06:44:45 +01:00
/** The beatmap directory. */
2014-06-30 04:17:04 +02:00
private static File beatmapDir ;
2015-01-22 06:44:45 +01:00
/** The OSZ archive directory. */
2014-07-06 07:58:44 +02:00
private static File oszDir ;
2015-01-22 06:44:45 +01:00
/** The screenshot directory (created when needed). */
2014-07-18 06:56:37 +02:00
private static File screenshotDir ;
2015-03-12 01:52:51 +01:00
/** The replay directory (created when needed). */
private static File replayDir ;
2015-04-02 04:10:36 +02:00
/** The replay import directory. */
private static File replayImportDir ;
2015-05-24 05:48:28 +02:00
/** The root skin directory. */
private static File skinRootDir ;
2014-06-30 04:17:04 +02:00
2015-03-03 06:40:51 +01:00
/** Port binding. */
private static int port = 49250 ;
2016-10-14 23:54:06 +02:00
private static boolean noSingleInstance ;
2016-12-20 03:02:01 +01:00
/** The theme song string: {@code filename,title,artist,length(ms)} */
2016-12-23 00:36:31 +01:00
private static String themeString = " theme.mp3,Rainbows,Kevin MacLeod,219350 " ;
2016-12-20 03:02:01 +01:00
/** The theme song timing point string (for computing beats to pulse the logo) . */
2016-12-23 22:53:12 +01:00
private static String themeTimingPoint = " 1080,545.454545454545,4,1,0,100,0,0 " ;
2016-12-20 03:02:01 +01:00
2015-08-27 21:14:04 +02:00
/ * *
* Returns whether the XDG flag in the manifest ( if any ) is set to " true " .
* @return true if XDG directories are enabled , false otherwise
* /
private static boolean checkXDGFlag ( ) {
JarFile jarFile = Utils . getJarFile ( ) ;
if ( jarFile = = null )
return false ;
try {
Manifest manifest = jarFile . getManifest ( ) ;
if ( manifest = = null )
return false ;
Attributes attributes = manifest . getMainAttributes ( ) ;
String value = attributes . getValue ( " Use-XDG " ) ;
return ( value ! = null & & value . equalsIgnoreCase ( " true " ) ) ;
} catch ( IOException e ) {
return false ;
}
}
2015-02-13 21:03:17 +01:00
/ * *
* Returns the directory based on the XDG base directory specification for
2015-08-27 21:14:04 +02:00
* Unix - like operating systems , only if the " XDG " flag is enabled .
2015-02-13 21:03:17 +01:00
* @param env the environment variable to check ( XDG_ * _ * )
* @param fallback the fallback directory relative to ~ home
* @return the XDG base directory , or the working directory if unavailable
* /
private static File getXDGBaseDir ( String env , String fallback ) {
2016-12-23 01:54:05 +01:00
File workingDir = Utils . isJarRunning ( ) ?
Utils . getRunningDirectory ( ) . getParentFile ( ) : Utils . getWorkingDirectory ( ) ;
2015-08-27 21:14:04 +02:00
if ( ! USE_XDG )
2016-12-23 01:54:05 +01:00
return workingDir ;
2015-02-13 21:03:17 +01:00
String OS = System . getProperty ( " os.name " ) . toLowerCase ( ) ;
if ( OS . indexOf ( " nix " ) > = 0 | | OS . indexOf ( " nux " ) > = 0 | | OS . indexOf ( " aix " ) > 0 ) {
String rootPath = System . getenv ( env ) ;
if ( rootPath = = null ) {
String home = System . getProperty ( " user.home " ) ;
if ( home = = null )
return new File ( " ./ " ) ;
rootPath = String . format ( " %s/%s " , home , fallback ) ;
}
File dir = new File ( rootPath , " opsu " ) ;
2015-07-11 17:51:52 +02:00
if ( ! dir . isDirectory ( ) & & ! dir . mkdir ( ) )
2017-01-21 01:16:27 +01:00
ErrorHandler . error ( String . format ( " Failed to create configuration folder at '%s/opsu'. " , rootPath ) , new Exception ( " empty " ) ) . preventReport ( ) . show ( ) ;
2015-02-13 21:03:17 +01:00
return dir ;
} else
2016-12-23 01:54:05 +01:00
return workingDir ;
2015-02-13 21:03:17 +01:00
}
2015-11-01 03:20:08 +01:00
/ * *
* Returns the osu ! installation directory .
* @return the directory , or null if not found
* /
private static File getOsuInstallationDirectory ( ) {
if ( ! System . getProperty ( " os.name " ) . startsWith ( " Win " ) )
return null ; // only works on Windows
// registry location
final WinReg . HKEY rootKey = WinReg . HKEY_CLASSES_ROOT ;
final String regKey = " osu \\ DefaultIcon " ;
final String regValue = null ; // default value
final String regPathPattern = " \" (.+) \\ \\ [^ \\ /]+ \\ .exe \" " ;
String value ;
try {
value = Advapi32Util . registryGetStringValue ( rootKey , regKey , regValue ) ;
} catch ( Win32Exception e ) {
return null ; // key/value not found
}
Pattern pattern = Pattern . compile ( regPathPattern ) ;
Matcher m = pattern . matcher ( value ) ;
if ( ! m . find ( ) )
return null ;
File dir = new File ( m . group ( 1 ) ) ;
return ( dir . isDirectory ( ) ) ? dir : null ;
}
2015-01-22 06:44:45 +01:00
/** Game options. */
public enum GameOption {
2015-06-12 22:04:20 +02:00
// internal options (not displayed in-game)
BEATMAP_DIRECTORY ( " BeatmapDirectory " ) {
@Override
public String write ( ) { return getBeatmapDir ( ) . getAbsolutePath ( ) ; }
@Override
public void read ( String s ) { beatmapDir = new File ( s ) ; }
} ,
OSZ_DIRECTORY ( " OSZDirectory " ) {
@Override
public String write ( ) { return getOSZDir ( ) . getAbsolutePath ( ) ; }
@Override
public void read ( String s ) { oszDir = new File ( s ) ; }
} ,
SCREENSHOT_DIRECTORY ( " ScreenshotDirectory " ) {
@Override
public String write ( ) { return getScreenshotDir ( ) . getAbsolutePath ( ) ; }
@Override
public void read ( String s ) { screenshotDir = new File ( s ) ; }
} ,
REPLAY_DIRECTORY ( " ReplayDirectory " ) {
@Override
public String write ( ) { return getReplayDir ( ) . getAbsolutePath ( ) ; }
@Override
public void read ( String s ) { replayDir = new File ( s ) ; }
} ,
2015-06-30 03:18:28 +02:00
REPLAY_IMPORT_DIRECTORY ( " ReplayImportDirectory " ) {
@Override
public String write ( ) { return getReplayImportDir ( ) . getAbsolutePath ( ) ; }
@Override
public void read ( String s ) { replayImportDir = new File ( s ) ; }
} ,
2015-06-12 22:04:20 +02:00
SKIN_DIRECTORY ( " SkinDirectory " ) {
@Override
public String write ( ) { return getSkinRootDir ( ) . getAbsolutePath ( ) ; }
@Override
public void read ( String s ) { skinRootDir = new File ( s ) ; }
} ,
THEME_SONG ( " ThemeSong " ) {
@Override
public String write ( ) { return themeString ; }
@Override
2016-12-23 00:36:31 +01:00
public void read ( String s ) {
String oldThemeString = themeString ;
themeString = s ;
Beatmap beatmap = getThemeBeatmap ( ) ;
if ( beatmap = = null ) {
themeString = oldThemeString ;
Log . warn ( String . format ( " The theme song string [%s] is malformed. " , s ) ) ;
} else if ( ! beatmap . audioFilename . isFile ( ) ) {
themeString = oldThemeString ;
Log . warn ( String . format ( " Cannot find theme song [%s]. " , beatmap . audioFilename . getAbsolutePath ( ) ) ) ;
}
}
} ,
THEME_SONG_TIMINGPOINT ( " ThemeSongTiming " ) {
@Override
public String write ( ) { return themeTimingPoint ; }
@Override
public void read ( String s ) {
try {
new TimingPoint ( s ) ;
themeTimingPoint = s ;
} catch ( Exception e ) {
Log . warn ( String . format ( " The theme song timing point [%s] is malformed. " , s ) ) ;
}
}
2015-06-12 22:04:20 +02:00
} ,
PORT ( " Port " ) {
@Override
public String write ( ) { return Integer . toString ( port ) ; }
@Override
public void read ( String s ) {
int i = Integer . parseInt ( s ) ;
2015-08-13 05:00:58 +02:00
if ( i > 0 & & i < = 65535 )
2015-06-12 22:04:20 +02:00
port = i ;
}
} ,
2016-10-14 23:54:06 +02:00
NOSINGLEINSTANCE ( " NoSingleInstance " ) {
@Override
public String write ( ) { return noSingleInstance + " " ; }
@Override
public void read ( String s ) {
noSingleInstance = ! " false " . equals ( s ) ;
}
} ,
2015-06-12 22:04:20 +02:00
// in-game options
2017-01-21 14:56:34 +01:00
SCREEN_RESOLUTION ( " Screen Resolution " , " ScreenResolution " , " Change the size of the game. " ) {
2015-01-11 19:05:32 +01:00
@Override
2016-10-11 11:37:47 +02:00
public String getValueString ( ) {
return resolutions [ resolutionIdx ] ;
}
2015-01-11 19:05:32 +01:00
@Override
2016-09-27 17:36:10 +02:00
public Object [ ] getListItems ( ) {
2016-10-11 11:37:47 +02:00
return resolutions ;
2016-09-27 17:36:10 +02:00
}
@Override
public void clickListItem ( int index ) {
2016-10-11 11:37:47 +02:00
resolutionIdx = index ;
2017-01-21 14:23:34 +01:00
setDisplayMode ( DisplayContainer . instance ) ;
2015-06-12 22:04:20 +02:00
}
@Override
public void read ( String s ) {
try {
2016-10-11 11:37:47 +02:00
resolutionIdx = Integer . parseInt ( s ) ;
} catch ( NumberFormatException ignored ) { }
}
@Override
public String write ( ) {
return resolutionIdx + " " ;
2015-01-11 19:05:32 +01:00
}
} ,
2016-10-11 11:51:29 +02:00
ALLOW_LARGER_RESOLUTIONS ( " Allow large resolutions " , " AllowLargeRes " , " Allow resolutions larger than the native resolution " , false ) ,
FULLSCREEN ( " Fullscreen Mode " , " Fullscreen " , " Restart to apply changes. " , false ) ,
2017-01-21 14:56:34 +01:00
SKIN ( " Skin " , " Skin " , " Change how the game looks. " ) {
2015-05-24 05:48:28 +02:00
@Override
public String getValueString ( ) { return skinName ; }
@Override
2016-09-27 17:36:10 +02:00
public Object [ ] getListItems ( ) {
return skinDirs ;
}
@Override
public void clickListItem ( int index ) {
skinName = skinDirs [ index ] ;
2017-01-21 15:18:04 +01:00
reloadSkin ( ) ;
2015-05-24 05:48:28 +02:00
}
2015-06-12 22:04:20 +02:00
@Override
public void read ( String s ) { skinName = s ; }
2015-05-24 05:48:28 +02:00
} ,
2017-02-05 23:36:39 +01:00
TARGET_UPS ( " target UPS " , " targetUPS " , " Higher values result in less input lag and smoother cursor trail, but may cause high CPU usage. " , 480 , 20 , 1000 ) {
2015-01-11 19:05:32 +01:00
@Override
2015-01-15 07:52:16 +01:00
public String getValueString ( ) {
2017-02-05 23:36:39 +01:00
return String . format ( " %dups " , val ) ;
2015-01-15 07:52:16 +01:00
}
2015-01-11 19:05:32 +01:00
@Override
2017-02-05 23:36:39 +01:00
public void setValue ( int value ) {
super . setValue ( value ) ;
displayContainer . setUPS ( value ) ;
}
} ,
TARGET_FPS ( " FPS limit " , " FPSlimit " , " Higher values may cause high CPU usage. A value higher than the UPS has no effect. " ) {
@Override
public String getValueString ( ) {
return String . format ( " %dfps " , getTargetFPS ( ) ) ;
}
private CachedVariable < Object [ ] > $_getListItems = new CachedVariable < > ( new CachedVariable . Getter < Object [ ] > ( ) {
@Override
public Object [ ] get ( ) {
String [ ] list = new String [ targetFPS . length ] ;
for ( int i = 0 ; i < targetFPS . length ; i + + ) {
list [ i ] = String . format ( " %dfps " , targetFPS [ i ] ) ;
}
return list ;
2016-09-27 17:36:10 +02:00
}
2017-02-05 23:36:39 +01:00
} ) ;
@Override
public Object [ ] getListItems ( ) {
return $_getListItems . get ( ) ;
2016-09-27 17:36:10 +02:00
}
@Override
public void clickListItem ( int index ) {
targetFPSindex = index ;
2017-01-18 22:46:45 +01:00
displayContainer . setFPS ( targetFPS [ index ] ) ;
2015-01-11 19:05:32 +01:00
}
2015-06-12 22:04:20 +02:00
@Override
2017-02-05 23:36:39 +01:00
public String write ( ) {
return Integer . toString ( targetFPS [ targetFPSindex ] ) ;
}
2015-06-12 22:04:20 +02:00
@Override
public void read ( String s ) {
int i = Integer . parseInt ( s ) ;
for ( int j = 0 ; j < targetFPS . length ; j + + ) {
if ( i = = targetFPS [ j ] ) {
targetFPSindex = j ;
break ;
}
}
}
2015-01-11 19:05:32 +01:00
} ,
2017-02-05 23:36:39 +01:00
SHOW_FPS ( " Show FPS Counters " , " FpsCounter " , " Show FPS and UPS counters in the bottom-right hand corner. " , true ) ,
USE_FPS_DELTAS ( " Use deltas for FPS counters " , " FpsCounterDeltas " , " Show time between updates instead of updates per second. " , false ) {
@Override
public boolean showCondition ( ) {
return SHOW_FPS . bool ;
}
} ,
2015-06-12 22:12:21 +02:00
SHOW_UNICODE ( " Prefer Non-English Metadata " , " ShowUnicode " , " Where available, song titles will be shown in their native language. " , false ) {
@Override
2017-01-19 16:03:53 +01:00
public void click ( ) {
2017-01-29 11:28:24 +01:00
super . click ( ) ;
2015-06-12 22:12:21 +02:00
if ( bool ) {
try {
2015-08-21 03:40:07 +02:00
Fonts . LARGE . loadGlyphs ( ) ;
Fonts . MEDIUM . loadGlyphs ( ) ;
Fonts . DEFAULT . loadGlyphs ( ) ;
2015-06-12 22:12:21 +02:00
} catch ( SlickException e ) {
Log . warn ( " Failed to load glyphs. " , e ) ;
}
}
}
} ,
SCREENSHOT_FORMAT ( " Screenshot Format " , " ScreenshotFormat " , " Press F12 to take a screenshot. " ) {
@Override
2017-01-29 00:20:56 +01:00
public String getValueString ( ) { return screenshotFormat [ screenshotFormatIndex ] ; }
2015-06-12 22:12:21 +02:00
@Override
2016-09-27 17:36:10 +02:00
public Object [ ] getListItems ( ) {
return screenshotFormat ;
}
@Override
public void clickListItem ( int index ) {
screenshotFormatIndex = index ;
}
2015-06-12 22:12:21 +02:00
@Override
public String write ( ) { return Integer . toString ( screenshotFormatIndex ) ; }
@Override
public void read ( String s ) {
int i = Integer . parseInt ( s ) ;
if ( i > = 0 & & i < screenshotFormat . length )
screenshotFormatIndex = i ;
}
} ,
2017-01-29 18:18:25 +01:00
CURSOR_SIZE ( " Size " , " CursorSize " , " Change the cursor scale. " , 100 , 50 , 200 ) {
2015-06-14 19:30:33 +02:00
@Override
public String getValueString ( ) { return String . format ( " %.2fx " , val / 100f ) ; }
@Override
public String write ( ) { return String . format ( Locale . US , " %.2f " , val / 100f ) ; }
@Override
public void read ( String s ) {
int i = ( int ) ( Float . parseFloat ( s ) * 100f ) ;
if ( i > = 50 & & i < = 200 )
val = i ;
}
} ,
2017-01-21 13:11:34 +01:00
NEW_CURSOR ( " Enable New Cursor " , " NewCursor " , " Use the new cursor style (may cause higher CPU usage). " , true ) ,
2015-06-12 22:12:21 +02:00
DYNAMIC_BACKGROUND ( " Enable Dynamic Backgrounds " , " DynamicBackground " , " The song background will be used as the main menu background. " , true ) ,
LOAD_VERBOSE ( " Show Detailed Loading Progress " , " LoadVerbose " , " Display more specific loading information in the splash screen. " , false ) ,
2016-12-18 21:03:19 +01:00
COLOR_MAIN_MENU_LOGO ( " Use cursor color as main menu logo tint " , " ColorMainMenuLogo " , " Colorful main menu logo " , false ) ,
2017-01-29 18:18:25 +01:00
MASTER_VOLUME ( " Master " , " VolumeUniversal " , " Global volume level. " , 35 , 0 , 100 ) {
2015-01-20 20:52:02 +01:00
@Override
2016-12-11 16:32:21 +01:00
public void setValue ( int value ) {
super . setValue ( value ) ;
SoundStore . get ( ) . setMusicVolume ( getMasterVolume ( ) * getMusicVolume ( ) ) ;
2015-01-20 20:52:02 +01:00
}
} ,
2017-01-29 18:18:25 +01:00
MUSIC_VOLUME ( " Music " , " VolumeMusic " , " Volume of music. " , 80 , 0 , 100 ) {
2015-01-11 19:05:32 +01:00
@Override
2016-12-11 16:32:21 +01:00
public void setValue ( int value ) {
super . setValue ( value ) ;
SoundStore . get ( ) . setMusicVolume ( getMasterVolume ( ) * getMusicVolume ( ) ) ;
2015-01-11 19:05:32 +01:00
}
2015-06-12 22:04:20 +02:00
} ,
2017-01-29 18:18:25 +01:00
SAMPLE_VOLUME_OVERRIDE ( " Sample override " , " BMSampleOverride " , " Override beatmap hitsound volume " , 100 , 0 , 100 ) {
2016-10-01 10:46:28 +02:00
@Override
public String getValueString ( ) {
if ( val = = 0 ) {
return " Disabled " ;
}
return super . getValueString ( ) ;
}
} ,
2017-01-29 18:18:25 +01:00
EFFECT_VOLUME ( " Effects " , " VolumeEffect " , " Volume of menu and game sounds. " , 70 , 0 , 100 ) ,
HITSOUND_VOLUME ( " Hit Sounds " , " VolumeHitSound " , " Volume of hit sounds. " , 30 , 0 , 100 ) ,
2015-06-12 22:04:20 +02:00
MUSIC_OFFSET ( " Music Offset " , " Offset " , " Adjust this value if hit objects are out of sync. " , - 75 , - 500 , 500 ) {
2015-01-11 19:05:32 +01:00
@Override
2015-03-03 06:40:51 +01:00
public String getValueString ( ) { return String . format ( " %dms " , val ) ; }
2015-01-11 19:05:32 +01:00
} ,
2015-06-12 22:12:21 +02:00
DISABLE_SOUNDS ( " Disable All Sound Effects " , " DisableSound " , " May resolve Linux sound driver issues. Requires a restart. " ,
2015-07-11 17:51:52 +02:00
( System . getProperty ( " os.name " ) . toLowerCase ( ) . contains ( " linux " ) ) ) ,
2015-06-12 22:12:21 +02:00
KEY_LEFT ( " Left Game Key " , " keyOsuLeft " , " Select this option to input a key. " ) {
2015-01-11 19:05:32 +01:00
@Override
2015-06-12 22:12:21 +02:00
public String getValueString ( ) { return Keyboard . getKeyName ( getGameKeyLeft ( ) ) ; }
2015-01-11 19:05:32 +01:00
@Override
2015-06-12 22:12:21 +02:00
public String write ( ) { return Keyboard . getKeyName ( getGameKeyLeft ( ) ) ; }
2015-06-12 22:04:20 +02:00
@Override
2015-06-12 22:12:21 +02:00
public void read ( String s ) { setGameKeyLeft ( Keyboard . getKeyIndex ( s ) ) ; }
} ,
KEY_RIGHT ( " Right Game Key " , " keyOsuRight " , " Select this option to input a key. " ) {
@Override
public String getValueString ( ) { return Keyboard . getKeyName ( getGameKeyRight ( ) ) ; }
2015-06-12 22:04:20 +02:00
@Override
2015-06-12 22:12:21 +02:00
public String write ( ) { return Keyboard . getKeyName ( getGameKeyRight ( ) ) ; }
2015-01-11 19:05:32 +01:00
@Override
2015-06-12 22:12:21 +02:00
public void read ( String s ) { setGameKeyRight ( Keyboard . getKeyIndex ( s ) ) ; }
2015-01-11 19:05:32 +01:00
} ,
2016-09-30 12:18:34 +02:00
DISABLE_MOUSE_WHEEL ( " Disable mouse wheel in play mode " , " MouseDisableWheel " , " During play, you can use the mouse wheel to adjust the volume and pause the game. This will disable that functionality. " , false ) ,
DISABLE_MOUSE_BUTTONS ( " Disable mouse buttons in play mode " , " MouseDisableButtons " , " This option will disable all mouse buttons. Specifically for people who use their keyboard to click. " , false ) ,
2015-09-11 17:43:19 +02:00
DISABLE_CURSOR ( " Disable Cursor " , " DisableCursor " , " Hide the cursor sprite. " , false ) ,
2015-06-13 01:25:19 +02:00
BACKGROUND_DIM ( " Background Dim " , " DimLevel " , " Percentage to dim the background image during gameplay. " , 50 , 0 , 100 ) ,
2016-12-11 14:07:23 +01:00
DANCE_REMOVE_BG ( " Use black background instead of image " , " RemoveBG " , " Hello darkness my old friend " , true ) ,
2015-06-12 22:04:20 +02:00
FORCE_DEFAULT_PLAYFIELD ( " Force Default Playfield " , " ForceDefaultPlayfield " , " Override the song background with the default playfield background. " , false ) ,
IGNORE_BEATMAP_SKINS ( " Ignore All Beatmap Skins " , " IgnoreBeatmapSkins " , " Never use skin element overrides provided by beatmaps. " , false ) ,
2015-09-16 17:19:23 +02:00
SNAKING_SLIDERS ( " Snaking sliders " , " SnakingSliders " , " Sliders gradually snake out from their starting point. " , true ) ,
2017-01-27 10:54:35 +01:00
SHRINKING_SLIDERS ( " Shrinking sliders " , " ShrinkingSliders " , " Sliders shrinks when sliderball passes (aka knorkesliders) " , true ) ,
2016-10-15 00:09:52 +02:00
FALLBACK_SLIDERS ( " Fallback sliders " , " FallbackSliders " , " Enable this if sliders won't render " , false ) ,
2017-01-27 10:54:35 +01:00
MERGING_SLIDERS ( " Merging sliders " , " MergingSliders " , " Merge sliders (aka knorkesliders) " , true ) {
2016-12-04 17:35:55 +01:00
@Override
public boolean showCondition ( ) {
return ! FALLBACK_SLIDERS . bool ;
}
} ,
2016-12-25 16:57:09 +01:00
MERGING_SLIDERS_MIRROR_POOL ( " Merging sliders mirror pool " , " MergingSliderMirrorPool " , " Amount of mirrors to calculate for merging sliders (impacts performance) " , 2 , 1 , 5 ) {
@Override
public String getValueString ( ) {
return String . valueOf ( val ) ;
}
@Override
public boolean showCondition ( ) {
return MERGING_SLIDERS . showCondition ( ) & & MERGING_SLIDERS . getBooleanValue ( ) ;
}
} ,
2017-01-29 18:18:25 +01:00
DRAW_SLIDER_ENDCIRCLES ( " Draw endcircles " , " DrawSliderEndCircles " , " Old slider style " , false ) ,
2015-06-12 22:12:21 +02:00
SHOW_HIT_LIGHTING ( " Show Hit Lighting " , " HitLighting " , " Adds an effect behind hit explosions. " , true ) ,
2016-12-10 17:33:34 +01:00
SHOW_HIT_ANIMATIONS ( " Show Hit Animations " , " HitAnimations " , " Fade out circles and curves. " , true ) ,
2016-12-12 08:37:21 +01:00
SHOW_REVERSEARROW_ANIMATIONS ( " Show reverse arrow animations " , " ReverseArrowAnimations " , " Fade out reverse arrows after passing. " , true ) ,
2015-06-12 22:12:21 +02:00
SHOW_COMBO_BURSTS ( " Show Combo Bursts " , " ComboBurst " , " A character image is displayed at combo milestones. " , true ) ,
SHOW_PERFECT_HIT ( " Show Perfect Hits " , " PerfectHit " , " Whether to show perfect hit result bursts (300s, slider ticks). " , true ) ,
SHOW_FOLLOW_POINTS ( " Show Follow Points " , " FollowPoints " , " Whether to show follow points between hit objects. " , true ) ,
SHOW_HIT_ERROR_BAR ( " Show Hit Error Bar " , " ScoreMeter " , " Shows precisely how accurate you were with each hit. " , false ) ,
2016-11-13 04:17:28 +01:00
MAP_START_DELAY ( " Map start delay " , " StartDelay " , " Have a fix amount of time to prepare your play/record " , 20 , 1 , 50 ) {
@Override
public String getValueString ( ) {
2016-12-11 13:31:54 +01:00
return ( val * 100 ) + " ms " ;
2016-11-13 04:17:28 +01:00
}
} ,
2016-12-04 22:53:47 +01:00
MAP_END_DELAY ( " Map end delay " , " EndDelay " , " Have a fix amount of time at the and of the map for a smooth finish " , 50 , 1 , 150 ) {
2016-11-13 04:17:28 +01:00
@Override
public String getValueString ( ) {
2016-12-11 13:31:54 +01:00
return ( val * 100 ) + " ms " ;
2016-11-13 04:17:28 +01:00
}
} ,
2016-11-20 13:32:12 +01:00
EPILEPSY_WARNING ( " Epilepsy warning image " , " EpiWarn " , " Show a little warning for flashing colours in the beginning " , 0 , 0 , 20 ) {
2016-11-17 00:12:44 +01:00
@Override
public String getValueString ( ) {
if ( val = = 0 ) {
return " Disabled " ;
}
2016-12-11 13:31:54 +01:00
return ( val * 100 ) + " ms " ;
2016-11-17 00:12:44 +01:00
}
} ,
2015-06-12 22:12:21 +02:00
LOAD_HD_IMAGES ( " Load HD Images " , " LoadHDImages " , String . format ( " Loads HD (%s) images when available. Increases memory usage and loading times. " , GameImage . HD_SUFFIX ) , true ) ,
2017-01-29 19:49:57 +01:00
FIXED_CS ( " Fixed CS " , " FixedCS " , " Determines the size of circles and sliders. " , 0 , 0 , 100 ) {
2015-01-11 19:05:32 +01:00
@Override
2015-03-03 06:40:51 +01:00
public String getValueString ( ) { return ( val = = 0 ) ? " Disabled " : String . format ( " %.1f " , val / 10f ) ; }
2015-06-12 22:04:20 +02:00
@Override
public String write ( ) { return String . format ( Locale . US , " %.1f " , val / 10f ) ; }
@Override
public void read ( String s ) {
int i = ( int ) ( Float . parseFloat ( s ) * 10f ) ;
if ( i > = 0 & & i < = 100 )
val = i ;
}
2015-01-11 19:05:32 +01:00
} ,
2017-01-29 19:49:57 +01:00
FIXED_HP ( " Fixed HP " , " FixedHP " , " Determines the rate at which health decreases. " , 0 , 0 , 100 ) {
2015-01-11 19:05:32 +01:00
@Override
2015-03-03 06:40:51 +01:00
public String getValueString ( ) { return ( val = = 0 ) ? " Disabled " : String . format ( " %.1f " , val / 10f ) ; }
2015-06-12 22:04:20 +02:00
@Override
public String write ( ) { return String . format ( Locale . US , " %.1f " , val / 10f ) ; }
@Override
public void read ( String s ) {
int i = ( int ) ( Float . parseFloat ( s ) * 10f ) ;
if ( i > = 0 & & i < = 100 )
val = i ;
}
2015-01-11 19:05:32 +01:00
} ,
2017-01-29 19:49:57 +01:00
FIXED_AR ( " Fixed AR " , " FixedAR " , " Determines how long hit circles stay on the screen. " , 0 , 0 , 100 ) {
2015-01-11 19:05:32 +01:00
@Override
2015-03-03 06:40:51 +01:00
public String getValueString ( ) { return ( val = = 0 ) ? " Disabled " : String . format ( " %.1f " , val / 10f ) ; }
2015-06-12 22:04:20 +02:00
@Override
public String write ( ) { return String . format ( Locale . US , " %.1f " , val / 10f ) ; }
@Override
public void read ( String s ) {
int i = ( int ) ( Float . parseFloat ( s ) * 10f ) ;
if ( i > = 0 & & i < = 100 )
val = i ;
}
2015-01-11 19:05:32 +01:00
} ,
2017-01-29 19:49:57 +01:00
FIXED_OD ( " Fixed OD " , " FixedOD " , " Determines the time window for hit results. " , 0 , 0 , 100 ) {
2015-01-11 19:05:32 +01:00
@Override
2015-03-03 06:40:51 +01:00
public String getValueString ( ) { return ( val = = 0 ) ? " Disabled " : String . format ( " %.1f " , val / 10f ) ; }
2015-06-12 22:04:20 +02:00
@Override
public String write ( ) { return String . format ( Locale . US , " %.1f " , val / 10f ) ; }
@Override
public void read ( String s ) {
int i = ( int ) ( Float . parseFloat ( s ) * 10f ) ;
if ( i > = 0 & & i < = 100 )
val = i ;
}
2015-01-11 19:05:32 +01:00
} ,
2015-06-12 22:04:20 +02:00
CHECKPOINT ( " Track Checkpoint " , " Checkpoint " , " Press Ctrl+L while playing to load a checkpoint, and Ctrl+S to set one. " , 0 , 0 , 3599 ) {
2015-01-11 19:05:32 +01:00
@Override
public String getValueString ( ) {
2015-03-03 06:40:51 +01:00
return ( val = = 0 ) ? " Disabled " : String . format ( " %02d:%02d " ,
TimeUnit . SECONDS . toMinutes ( val ) ,
val - TimeUnit . MINUTES . toSeconds ( TimeUnit . SECONDS . toMinutes ( val ) ) ) ;
2015-01-11 19:05:32 +01:00
}
} ,
2015-07-03 05:16:14 +02:00
ENABLE_THEME_SONG ( " Enable Theme Song " , " MenuMusic " , " Whether to play the theme song upon starting opsu! " , true ) ,
2015-07-08 02:03:54 +02:00
REPLAY_SEEKING ( " Replay Seeking " , " ReplaySeeking " , " Enable a seeking bar on the left side of the screen during replays. " , false ) ,
2015-08-21 17:25:52 +02:00
DISABLE_UPDATER ( " Disable Automatic Updates " , " DisableUpdater " , " Disable automatic checking for updates upon starting opsu!. " , false ) ,
2016-09-27 17:55:41 +02:00
ENABLE_WATCH_SERVICE ( " Enable Watch Service " , " WatchService " , " Watch the beatmap directory for changes. Requires a restart. " , false ) ,
2017-01-29 18:18:25 +01:00
DANCE_MOVER ( " Algorithm " , " Mover " , " Algorithm that decides how to move from note to note " ) {
2016-09-27 22:02:49 +02:00
@Override
public Object [ ] getListItems ( ) {
2016-11-20 17:24:58 +01:00
return Dancer . moverFactories ;
2016-09-27 22:02:49 +02:00
}
@Override
public void clickListItem ( int index ) {
2016-12-12 19:48:03 +01:00
if ( Game . isInGame & & Dancer . moverFactories [ index ] instanceof PolyMoverFactory ) {
// TODO remove this when #79 is fixed
2017-01-21 23:52:19 +01:00
EventBus . instance . post ( new BarNotificationEvent ( " This mover is disabled in the storyboard right now " ) ) ;
2016-12-12 19:48:03 +01:00
return ;
}
2016-11-20 17:24:58 +01:00
Dancer . instance . setMoverFactoryIndex ( index ) ;
2016-09-27 22:02:49 +02:00
}
@Override
public String getValueString ( ) {
2016-11-20 17:24:58 +01:00
return Dancer . moverFactories [ Dancer . instance . getMoverFactoryIndex ( ) ] . toString ( ) ;
2016-09-27 22:02:49 +02:00
}
@Override
public String write ( ) {
2016-11-20 17:24:58 +01:00
return String . valueOf ( Dancer . instance . getMoverFactoryIndex ( ) ) ;
2016-09-27 22:02:49 +02:00
}
@Override
public void read ( String s ) {
2016-11-19 02:33:15 +01:00
int i = Integer . parseInt ( s ) ;
2016-11-20 17:24:58 +01:00
Dancer . instance . setMoverFactoryIndex ( i ) ;
2016-09-27 22:02:49 +02:00
}
} ,
2017-01-01 23:47:11 +01:00
DANCE_EXGON_DELAY ( " ExGon delay " , " ExGonDelay " , " Delay between moves for the ExGon mover " , 25 , 2 , 750 ) {
@Override
public String getValueString ( ) {
return String . valueOf ( val ) ;
}
@Override
public boolean showCondition ( ) {
return Dancer . moverFactories [ Dancer . instance . getMoverFactoryIndex ( ) ] instanceof ExgonMoverFactory ;
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_QUAD_BEZ_AGGRESSIVENESS ( " Bezier aggressiveness " , " QuadBezAgr " , " AKA initial D factor " , 50 , 0 , 200 ) {
2016-12-04 13:54:26 +01:00
@Override
public String getValueString ( ) {
2016-12-11 13:31:54 +01:00
return String . valueOf ( val ) ;
2016-12-04 14:31:35 +01:00
}
2016-12-04 13:40:41 +01:00
@Override
public boolean showCondition ( ) {
return Dancer . moverFactories [ Dancer . instance . getMoverFactoryIndex ( ) ] instanceof QuadraticBezierMoverFactory ;
}
2016-12-04 13:37:00 +01:00
} ,
2017-01-29 18:18:25 +01:00
DANCE_QUAD_BEZ_SLIDER_AGGRESSIVENESS_FACTOR ( " Exit aggressiveness " , " CubBezSliderExitAgr " , " AKA initial D factor for sliderexits " , 4 , 1 , 6 ) {
2016-12-04 15:32:38 +01:00
@Override
public String getValueString ( ) {
2016-12-11 13:31:54 +01:00
return String . valueOf ( val ) ;
2016-12-04 15:32:38 +01:00
}
@Override
public boolean showCondition ( ) {
2016-12-04 16:11:15 +01:00
return DANCE_QUAD_BEZ_AGGRESSIVENESS . showCondition ( )
& & Dancer . sliderMoverController instanceof DefaultSliderMoverController ;
2016-12-04 15:32:38 +01:00
}
} ,
DANCE_QUAD_BEZ_USE_CUBIC_ON_SLIDERS ( " Use cubic bezier before sliders " , " QuadBezCubicSliders " , " Slider entry looks better using this " , true ) {
@Override
public boolean showCondition ( ) {
2016-12-04 16:11:15 +01:00
return DANCE_QUAD_BEZ_SLIDER_AGGRESSIVENESS_FACTOR . showCondition ( ) ;
2016-12-04 15:32:38 +01:00
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_QUAD_BEZ_CUBIC_AGGRESSIVENESS_FACTOR ( " Entry aggressiveness " , " CubBezSliderEntryAgr " , " AKA initial D factor for sliderentries " , 4 , 1 , 6 ) {
2016-12-04 15:32:38 +01:00
@Override
public String getValueString ( ) {
2016-12-11 13:31:54 +01:00
return String . valueOf ( val ) ;
2016-12-04 15:32:38 +01:00
}
@Override
public boolean showCondition ( ) {
2016-12-04 16:11:15 +01:00
return DANCE_QUAD_BEZ_USE_CUBIC_ON_SLIDERS . showCondition ( )
& & DANCE_QUAD_BEZ_USE_CUBIC_ON_SLIDERS . getBooleanValue ( ) ;
2016-12-04 15:32:38 +01:00
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_MOVER_DIRECTION ( " Direction " , " MoverDirection " , " The direction the mover goes " ) {
2016-09-30 21:45:19 +02:00
@Override
public String getValueString ( ) {
return Dancer . moverDirection . toString ( ) ;
}
@Override
public Object [ ] getListItems ( ) {
return MoverDirection . values ( ) ;
}
@Override
public void clickListItem ( int index ) {
Dancer . moverDirection = MoverDirection . values ( ) [ index ] ;
}
@Override
public String write ( ) {
return " " + Dancer . moverDirection . nr ;
}
@Override
public void read ( String s ) {
Dancer . moverDirection = MoverDirection . values ( ) [ Integer . parseInt ( s ) ] ;
}
} ,
2016-11-12 15:41:20 +01:00
DANCE_SLIDER_MOVER_TYPE ( " Slider mover " , " SliderMover " , " How to move in sliders " ) {
@Override
public String getValueString ( ) {
return Dancer . sliderMoverController . toString ( ) ;
}
@Override
public Object [ ] getListItems ( ) {
return Dancer . sliderMovers ;
}
@Override
public void clickListItem ( int index ) {
2016-11-12 18:25:25 +01:00
val = index ;
2016-11-12 15:41:20 +01:00
Dancer . sliderMoverController = Dancer . sliderMovers [ index ] ;
}
2016-11-12 18:25:25 +01:00
@Override
public String write ( ) {
return String . valueOf ( val ) ;
}
2016-11-12 15:41:20 +01:00
@Override
public void read ( String s ) {
2016-11-12 18:25:25 +01:00
Dancer . sliderMoverController = Dancer . sliderMovers [ val = Integer . parseInt ( s ) ] ;
2016-11-12 15:41:20 +01:00
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_SPINNER ( " Algorithm " , " Spinner " , " Spinner style " ) {
2016-09-27 22:02:49 +02:00
@Override
public Object [ ] getListItems ( ) {
return Dancer . spinners ;
}
@Override
public void clickListItem ( int index ) {
Dancer . instance . setSpinnerIndex ( index ) ;
}
@Override
public String getValueString ( ) {
return Dancer . spinners [ Dancer . instance . getSpinnerIndex ( ) ] . toString ( ) ;
}
2016-09-27 17:55:41 +02:00
2016-09-27 22:02:49 +02:00
@Override
public String write ( ) {
return Dancer . instance . getSpinnerIndex ( ) + " " ;
}
@Override
public void read ( String s ) {
Dancer . instance . setSpinnerIndex ( Integer . parseInt ( s ) ) ;
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_SPINNER_DELAY ( " Delay " , " SpinnerDelay " , " Fiddle with this if spinner goes too fast. " , 3 , 0 , 20 ) {
2016-10-01 13:44:34 +02:00
@Override
public String getValueString ( ) {
2016-12-11 13:25:04 +01:00
return String . format ( " %dms " , val ) ;
2016-10-01 13:44:34 +02:00
}
2016-09-27 22:02:49 +02:00
} ,
2016-12-11 13:25:04 +01:00
DANCE_LAZY_SLIDERS ( " Lazy sliders " , " LazySliders " , " Don't do short sliders " , false ) ,
DANCE_ONLY_CIRCLE_STACKS ( " Only circle stacks " , " CircleStacks " , " Only do circle movement on stacks " , false ) ,
DANCE_CIRCLE_STREAMS ( " Circle streams " , " CircleStreams " , " Make circles while streaming " , false ) ,
DANCE_MIRROR ( " Mirror collage " , " MirrorCollage " , " Hypnotizing stuff. Toggle this ingame by pressing the M key. " , false ) ,
DANCE_DRAW_APPROACH ( " Draw approach circles " , " DrawApproach " , " Can get a bit busy when using mirror collage " , true ) ,
2017-01-29 18:18:25 +01:00
DANCE_OBJECT_COLOR_OVERRIDE ( " Color " , " ObjColorOverride " , " Override object colors " ) {
2016-09-30 09:17:39 +02:00
@Override
2016-09-30 19:05:53 +02:00
public String getValueString ( ) {
return Dancer . colorOverride . toString ( ) ;
}
@Override
public Object [ ] getListItems ( ) {
return ObjectColorOverrides . values ( ) ;
}
@Override
public void clickListItem ( int index ) {
Dancer . colorOverride = ObjectColorOverrides . values ( ) [ index ] ;
}
@Override
public String write ( ) {
return " " + Dancer . colorOverride . nr ;
}
@Override
public void read ( String s ) {
Dancer . colorOverride = ObjectColorOverrides . values ( ) [ Integer . parseInt ( s ) ] ;
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_OBJECT_COLOR_OVERRIDE_MIRRORED ( " Mirror color " , " ObjColorMirroredOverride " , " Override collage object colors " ) {
2016-09-30 19:05:53 +02:00
@Override
public String getValueString ( ) {
return Dancer . colorMirrorOverride . toString ( ) ;
}
@Override
public Object [ ] getListItems ( ) {
return ObjectColorOverrides . values ( ) ;
}
@Override
public void clickListItem ( int index ) {
Dancer . colorMirrorOverride = ObjectColorOverrides . values ( ) [ index ] ;
}
@Override
public String write ( ) {
return " " + Dancer . colorMirrorOverride . nr ;
}
@Override
public void read ( String s ) {
Dancer . colorMirrorOverride = ObjectColorOverrides . values ( ) [ Integer . parseInt ( s ) ] ;
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_RGB_OBJECT_INC ( " RGB increment " , " RGBInc " , " Amount of hue to shift, used for rainbow object override " , 70 , - 1800 , 1800 ) {
2016-10-01 11:59:03 +02:00
@Override
public String getValueString ( ) {
return String . format ( " %.1f° " , val / 10f ) ;
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_CURSOR_COLOR_OVERRIDE ( " Color " , " CursorColorOverride " , " Override cursor color " ) {
2016-09-30 19:05:53 +02:00
@Override
public String getValueString ( ) {
2016-10-01 11:53:33 +02:00
return Dancer . cursorColorOverride . toString ( ) ;
2016-09-30 19:05:53 +02:00
}
@Override
2016-10-01 11:53:33 +02:00
public Object [ ] getListItems ( ) {
return CursorColorOverrides . values ( ) ;
}
@Override
public void clickListItem ( int index ) {
Dancer . cursorColorOverride = CursorColorOverrides . values ( ) [ index ] ;
}
@Override
public String write ( ) {
return " " + Dancer . cursorColorOverride . nr ;
2016-09-30 09:17:39 +02:00
}
@Override
public void read ( String s ) {
2016-10-01 11:53:33 +02:00
Dancer . cursorColorOverride = CursorColorOverrides . values ( ) [ Integer . parseInt ( s ) ] ;
2016-09-30 09:17:39 +02:00
}
} ,
2017-01-29 18:18:25 +01:00
DANCE_CURSOR_MIRROR_COLOR_OVERRIDE ( " Mirror color " , " CursorMirrorColorOverride " , " Override mirror cursor color " ) {
2016-09-30 19:38:02 +02:00
@Override
2016-10-01 11:53:33 +02:00
public String getValueString ( ) {
return Dancer . cursorColorMirrorOverride . toString ( ) ;
}
@Override
public Object [ ] getListItems ( ) {
return CursorColorOverrides . values ( ) ;
}
@Override
public void clickListItem ( int index ) {
Dancer . cursorColorMirrorOverride = CursorColorOverrides . values ( ) [ index ] ;
}
@Override
public String write ( ) {
return " " + Dancer . cursorColorMirrorOverride . nr ;
}
@Override
public void read ( String s ) {
Dancer . cursorColorMirrorOverride = CursorColorOverrides . values ( ) [ Integer . parseInt ( s ) ] ;
}
} ,
2016-12-11 13:25:04 +01:00
DANCE_CURSOR_ONLY_COLOR_TRAIL ( " Only color cursor trail " , " OnlyColorTrail " , " Don't color the cursor, only the trail " , false ) ,
DANCE_RGB_CURSOR_INC ( " RGB cursor increment " , " RGBCursorInc " , " Amount of hue to shift, used for rainbow cursor override " , 100 , - 2000 , 2000 ) {
2016-10-01 11:53:33 +02:00
@Override
public String getValueString ( ) {
2016-10-01 12:03:07 +02:00
return String . format ( " %.2f° " , val / 1000f ) ;
2016-10-01 11:53:33 +02:00
}
2016-09-30 19:38:02 +02:00
} ,
2017-01-29 18:18:25 +01:00
DANCE_CURSOR_TRAIL_OVERRIDE ( " Trail length " , " CursorTrailOverride " , " Override cursor trail length " , 20 , 20 , 600 ) {
2016-10-01 12:14:39 +02:00
@Override
public String getValueString ( ) {
if ( val = = 20 ) {
return " Disabled " ;
}
return " " + val ;
}
2016-09-30 09:45:36 +02:00
} ,
2016-12-11 13:25:04 +01:00
DANCE_HIDE_OBJECTS ( " Don't draw objects " , " HideObj " , " If you only want to see cursors :) " , false ) ,
DANCE_CIRLCE_IN_SLOW_SLIDERS ( " Do circles in slow sliders " , " CircleInSlider " , " Circle around sliderball in lazy & slow sliders " , false ) ,
DANCE_CIRLCE_IN_LAZY_SLIDERS ( " Do circles in lazy sliders " , " CircleInLazySlider " , " Circle in hitcircle in lazy sliders " , false ) ,
DANCE_HIDE_UI ( " Hide all UI " , " HideUI " , " . " , true ) ,
DANCE_ENABLE_SB ( " Enable storyboard editor " , " EnableStoryBoard " , " Dance storyboard " , false ) ,
2017-01-29 18:18:25 +01:00
PIPPI_ENABLE ( " Enable " , " Pippi " , " Move in circles like dancing pippi (osu! april fools joke 2016) " , false ) ,
PIPPI_RADIUS_PERCENT ( " Radius " , " PippiRad " , " Radius of pippi, percentage of circle radius " , 100 , 0 , 100 ) {
2016-11-20 22:50:49 +01:00
@Override
public String getValueString ( ) {
2016-12-11 14:25:30 +01:00
return val + " % " ;
2016-11-20 22:50:49 +01:00
}
2016-12-13 15:02:37 +01:00
@Override
public void setValue ( int value ) {
super . setValue ( value ) ;
Pippi . setRadiusPercent ( value ) ;
}
2016-11-20 22:50:49 +01:00
} ,
2017-01-29 18:18:25 +01:00
PIPPI_ANGLE_INC_MUL ( " Normal " , " PippiAngIncMul " , " How fast pippi's angle increments " , 10 , - 200 , 200 ) {
2016-09-27 22:33:14 +02:00
@Override
public String getValueString ( ) {
2016-09-30 22:08:45 +02:00
return String . format ( " x%.1f " , val / 10f ) ;
}
} ,
2017-01-29 18:18:25 +01:00
PIPPI_ANGLE_INC_MUL_SLIDER ( " In slider " , " PippiAngIncMulSlider " , " Same as above, but in sliders " , 50 , - 200 , 200 ) {
2016-09-27 22:33:14 +02:00
@Override
public String getValueString ( ) {
2016-09-30 22:08:45 +02:00
return String . format ( " x%.1f " , val / 10f ) ;
}
2016-09-27 22:33:14 +02:00
} ,
2016-12-11 13:25:04 +01:00
PIPPI_SLIDER_FOLLOW_EXPAND ( " Followcircle expand " , " PippiFollowExpand " , " Increase radius in followcircles " , false ) ,
PIPPI_PREVENT_WOBBLY_STREAMS ( " Prevent wobbly streams " , " PippiPreventWobblyStreams " , " Force linear mover while doing streams to prevent wobbly pippi " , true ) ;
2015-01-11 19:05:32 +01:00
2017-01-18 22:46:45 +01:00
public static DisplayContainer displayContainer ;
2016-09-27 22:02:49 +02:00
2015-01-22 06:44:45 +01:00
/** Option name. */
2015-08-21 04:11:55 +02:00
private final String name ;
2015-01-11 19:05:32 +01:00
2015-06-12 22:04:20 +02:00
/** Option name, as displayed in the configuration file. */
2015-08-21 04:11:55 +02:00
private final String displayName ;
2015-06-12 22:04:20 +02:00
2015-01-22 06:44:45 +01:00
/** Option description. */
2015-08-21 04:11:55 +02:00
private final String description ;
2015-01-11 19:05:32 +01:00
2015-03-03 06:40:51 +01:00
/** The boolean value for the option (if applicable). */
protected boolean bool ;
2016-11-20 23:19:13 +01:00
private int defaultVal = 0 ;
2015-03-03 06:40:51 +01:00
/** The integer value for the option (if applicable). */
protected int val ;
/** The upper and lower bounds on the integer value (if applicable). */
private int max , min ;
2015-06-12 22:04:20 +02:00
/** Option types. */
2016-12-11 02:08:35 +01:00
public enum OptionType { BOOLEAN , NUMERIC , OTHER } ;
2015-06-12 22:04:20 +02:00
2015-03-03 06:40:51 +01:00
/** Whether or not this is a numeric option. */
2015-06-12 22:04:20 +02:00
private OptionType type = OptionType . OTHER ;
2015-03-03 06:40:51 +01:00
2017-01-29 11:28:24 +01:00
/ * *
* If this option should not be shown in the optionsmenu because it does
* not match the search string .
* /
private boolean filtered ;
2015-01-11 19:05:32 +01:00
/ * *
2015-06-12 22:04:20 +02:00
* Constructor for internal options ( not displayed in - game ) .
* @param displayName the option name , as displayed in the configuration file
* /
GameOption ( String displayName ) {
2015-08-21 04:11:55 +02:00
this ( null , displayName , null ) ;
2015-06-12 22:04:20 +02:00
}
/ * *
* Constructor for other option types .
2015-01-11 19:05:32 +01:00
* @param name the option name
2015-06-12 22:04:20 +02:00
* @param displayName the option name , as displayed in the configuration file
2015-01-11 19:05:32 +01:00
* @param description the option description
* /
2015-06-12 22:04:20 +02:00
GameOption ( String name , String displayName , String description ) {
2015-01-11 19:05:32 +01:00
this . name = name ;
2015-06-12 22:04:20 +02:00
this . displayName = displayName ;
2015-01-11 19:05:32 +01:00
this . description = description ;
}
2015-03-03 06:40:51 +01:00
/ * *
2015-06-12 22:04:20 +02:00
* Constructor for boolean options .
2015-03-03 06:40:51 +01:00
* @param name the option name
2015-06-12 22:04:20 +02:00
* @param displayName the option name , as displayed in the configuration file
2015-03-03 06:40:51 +01:00
* @param description the option description
* @param value the default boolean value
* /
2015-06-12 22:04:20 +02:00
GameOption ( String name , String displayName , String description , boolean value ) {
this ( name , displayName , description ) ;
2015-03-03 06:40:51 +01:00
this . bool = value ;
2015-06-12 22:04:20 +02:00
this . type = OptionType . BOOLEAN ;
2015-03-03 06:40:51 +01:00
}
/ * *
2015-06-12 22:04:20 +02:00
* Constructor for numeric options .
2015-03-03 06:40:51 +01:00
* @param name the option name
2015-06-12 22:04:20 +02:00
* @param displayName the option name , as displayed in the configuration file
2015-03-03 06:40:51 +01:00
* @param description the option description
* @param value the default integer value
* /
2015-06-12 22:04:20 +02:00
GameOption ( String name , String displayName , String description , int value , int min , int max ) {
this ( name , displayName , description ) ;
2015-03-03 06:40:51 +01:00
this . val = value ;
2016-11-20 23:19:13 +01:00
this . defaultVal = value ;
2015-03-03 06:40:51 +01:00
this . min = min ;
this . max = max ;
2015-06-12 22:04:20 +02:00
this . type = OptionType . NUMERIC ;
2015-03-03 06:40:51 +01:00
}
2016-12-04 13:40:41 +01:00
/ * *
* should the option be shown
* @return true if the option should be shown
* /
public boolean showCondition ( ) {
return true ;
}
2015-01-11 19:05:32 +01:00
/ * *
* Returns the option name .
* @return the name string
* /
public String getName ( ) { return name ; }
2015-06-12 22:04:20 +02:00
/ * *
* Returns the option name , as displayed in the configuration file .
* @return the display name string
* /
public String getDisplayName ( ) { return displayName ; }
2015-01-11 19:05:32 +01:00
/ * *
2015-03-03 06:40:51 +01:00
* Returns the option description .
2015-01-11 19:05:32 +01:00
* @return the description string
* /
public String getDescription ( ) { return description ; }
2015-03-03 06:40:51 +01:00
/ * *
* Returns the boolean value for the option , if applicable .
* @return the boolean value
* /
public boolean getBooleanValue ( ) { return bool ; }
/ * *
* Returns the integer value for the option , if applicable .
* @return the integer value
* /
public int getIntegerValue ( ) { return val ; }
/ * *
* Sets the boolean value for the option .
* @param value the new boolean value
* /
public void setValue ( boolean value ) { this . bool = value ; }
/ * *
* Sets the integer value for the option .
* @param value the new integer value
* /
public void setValue ( int value ) { this . val = value ; }
2015-01-11 19:05:32 +01:00
/ * *
* Returns the value of the option as a string ( via override ) .
2015-03-03 06:40:51 +01:00
* < p >
2015-06-12 22:04:20 +02:00
* By default , this returns " {@code val}% " for numeric options ,
* " Yes " or " No " based on the { @code bool } field for boolean options ,
* and an empty string otherwise .
2015-01-11 19:05:32 +01:00
* @return the value string
* /
2015-03-03 06:40:51 +01:00
public String getValueString ( ) {
2015-06-12 22:04:20 +02:00
if ( type = = OptionType . NUMERIC )
2015-03-03 06:40:51 +01:00
return String . format ( " %d%% " , val ) ;
2015-06-12 22:04:20 +02:00
else if ( type = = OptionType . BOOLEAN )
2015-03-03 06:40:51 +01:00
return ( bool ) ? " Yes " : " No " ;
2015-06-12 22:04:20 +02:00
else
return " " ;
2015-03-03 06:40:51 +01:00
}
2015-01-11 19:05:32 +01:00
/ * *
* Processes a mouse click action ( via override ) .
2015-03-03 06:40:51 +01:00
* < p >
* By default , this inverts the current { @code bool } field .
2015-01-11 19:05:32 +01:00
* /
2017-01-19 16:03:53 +01:00
public void click ( ) { bool = ! bool ; }
2015-01-11 19:05:32 +01:00
2016-09-27 17:36:10 +02:00
/ * *
* Get a list of values to choose from
* @return list with value string or null if no list should be shown
* /
public Object [ ] getListItems ( ) { return null ; }
/ * *
* Fired when an item in the value list has been clicked
* @param index the itemindex which has been clicked
* /
2016-09-27 22:02:49 +02:00
public void clickListItem ( int index ) { }
2016-09-27 17:36:10 +02:00
2015-06-12 22:04:20 +02:00
/ * *
* Returns the string to write to the configuration file ( via override ) .
* < p >
* By default , this returns " {@code val} " for numeric options ,
* " true " or " false " based on the { @code bool } field for boolean options ,
* and { @link # getValueString ( ) } otherwise .
* @return the string to write
* /
public String write ( ) {
if ( type = = OptionType . NUMERIC )
return Integer . toString ( val ) ;
else if ( type = = OptionType . BOOLEAN )
return Boolean . toString ( bool ) ;
else
return getValueString ( ) ;
}
/ * *
* Reads the value of the option from the configuration file ( via override ) .
* < p >
2015-06-13 01:25:19 +02:00
* By default , this sets { @code val } for numeric options only if the
* value is between the min and max bounds , sets { @code bool } for
* boolean options , and does nothing otherwise .
2015-06-12 22:04:20 +02:00
* @param s the value string read from the configuration file
* /
public void read ( String s ) {
2015-06-13 01:25:19 +02:00
if ( type = = OptionType . NUMERIC ) {
int i = Integer . parseInt ( s ) ;
if ( i > = min & & i < = max )
val = i ;
} else if ( type = = OptionType . BOOLEAN )
2015-06-12 22:04:20 +02:00
bool = Boolean . parseBoolean ( s ) ;
}
2016-11-20 23:06:47 +01:00
2016-12-11 02:08:35 +01:00
public OptionType getType ( ) {
return type ;
2016-11-20 23:06:47 +01:00
}
public int getMinValue ( ) {
return min ;
}
public int getMaxValue ( ) {
return max ;
}
2016-11-20 23:19:13 +01:00
public int getDefaultVal ( ) {
return defaultVal ;
}
2017-01-29 11:28:24 +01:00
/ * *
* Update the filtered flag for this option based on the given searchString .
* @param searchString the searched string or null to reset the filtered flag
* @return true if this option does need to be filtered
* /
public boolean filter ( String searchString ) {
if ( searchString = = null | | searchString . length ( ) = = 0 ) {
filtered = false ;
return false ;
}
2017-01-29 16:53:14 +01:00
filtered = ! name . toLowerCase ( ) . contains ( searchString ) & & ! description . toLowerCase ( ) . contains ( searchString ) ;
2017-01-29 11:28:24 +01:00
return filtered ;
}
/ * *
* Check if this option should be filtered ( = not shown ) because it does not
* match the search string .
* @return true if the option shouldn ' t be shown .
* /
public boolean isFiltered ( ) {
return filtered ;
}
2016-11-20 23:06:47 +01:00
}
2014-07-03 00:24:19 +02:00
2015-06-12 22:04:20 +02:00
/** Map of option display names to GameOptions. */
private static HashMap < String , GameOption > optionMap ;
2016-10-11 11:37:47 +02:00
private static String [ ] resolutions = {
null ,
" 800x600 " ,
" 1024x600 " ,
" 1024x768 " ,
" 1280x720 " ,
" 1280x800 " ,
" 1280x960 " ,
" 1280x1024 " ,
" 1366x768 " ,
" 1440x900 " ,
" 1600x900 " ,
" 1600x1200 " ,
" 1680x1050 " ,
" 1920x1080 " ,
" 1920x1200 " ,
" 2560x1440 " ,
" 2560x1600 " ,
" 3840x2160 "
} ;
2014-06-30 04:17:04 +02:00
2016-10-11 11:37:47 +02:00
private static int resolutionIdx ;
2014-06-30 04:17:04 +02:00
2016-09-27 20:52:52 +02:00
public static int width ;
public static int height ;
2015-05-24 05:48:28 +02:00
/** The available skin directories. */
private static String [ ] skinDirs ;
/** The index in the skinDirs array. */
private static int skinDirIndex = 0 ;
/** The name of the skin. */
private static String skinName = " Default " ;
/** The current skin. */
private static Skin skin ;
2015-01-22 06:44:45 +01:00
/** Frame limiters. */
2016-10-01 12:06:05 +02:00
private static final int [ ] targetFPS = { 60 , 120 , 240 , 1000 } ;
2015-01-16 21:44:13 +01:00
2015-01-22 06:44:45 +01:00
/** Index in targetFPS[] array. */
2014-06-30 04:17:04 +02:00
private static int targetFPSindex = 0 ;
2015-01-22 06:44:45 +01:00
/** Screenshot file formats. */
2017-01-29 00:20:56 +01:00
private static String [ ] screenshotFormat = { " PNG " , " JPG " , " BMP " } ;
2014-06-30 04:17:04 +02:00
2015-01-22 06:44:45 +01:00
/** Index in screenshotFormat[] array. */
2014-06-30 04:17:04 +02:00
private static int screenshotFormatIndex = 0 ;
2015-01-22 06:44:45 +01:00
/** Left and right game keys. */
2014-07-18 05:58:37 +02:00
private static int
keyLeft = Keyboard . KEY_NONE ,
keyRight = Keyboard . KEY_NONE ;
2015-01-21 05:56:10 +01:00
// This class should not be instantiated.
private Options ( ) { }
2014-06-30 04:17:04 +02:00
2016-11-20 13:24:37 +01:00
public static String getSkinName ( ) {
return skinName ;
}
2016-10-11 11:47:46 +02:00
public static int getResolutionIdx ( ) {
return resolutionIdx ;
}
2016-10-11 11:51:29 +02:00
public static boolean allowLargeResolutions ( ) {
return GameOption . ALLOW_LARGER_RESOLUTIONS . getBooleanValue ( ) ;
}
2014-06-30 04:17:04 +02:00
/ * *
2014-07-02 01:32:03 +02:00
* Returns the target frame rate .
* @return the target FPS
2014-06-30 04:17:04 +02:00
* /
2014-07-02 01:32:03 +02:00
public static int getTargetFPS ( ) { return targetFPS [ targetFPSindex ] ; }
2014-06-30 04:17:04 +02:00
2017-02-05 23:36:39 +01:00
public static int getTargetUPS ( ) {
return GameOption . TARGET_UPS . val ;
}
2015-03-05 20:40:57 +01:00
/ * *
* Sets the target frame rate to the next available option , and sends a
* bar notification about the action .
* /
2017-01-17 23:44:12 +01:00
public static void setNextFPS ( DisplayContainer displayContainer ) {
2017-01-18 22:46:45 +01:00
GameOption . displayContainer = displayContainer ; // TODO dirty shit
2016-10-01 12:19:26 +02:00
GameOption . TARGET_FPS . clickListItem ( ( targetFPSindex + 1 ) % targetFPS . length ) ;
2017-01-21 23:52:19 +01:00
EventBus . instance . post ( new BarNotificationEvent ( String . format ( " Frame limiter: %s " , GameOption . TARGET_FPS . getValueString ( ) ) ) ) ;
2015-03-05 20:40:57 +01:00
}
2015-01-20 20:52:02 +01:00
/ * *
* Returns the master volume level .
* @return the volume [ 0 , 1 ]
* /
2015-03-03 06:40:51 +01:00
public static float getMasterVolume ( ) { return GameOption . MASTER_VOLUME . getIntegerValue ( ) / 100f ; }
2015-01-20 20:52:02 +01:00
/ * *
* Sets the master volume level ( if within valid range ) .
* @param volume the volume [ 0 , 1 ]
* /
2017-01-17 23:18:12 +01:00
public static void setMasterVolume ( float volume ) {
2015-01-20 20:52:02 +01:00
if ( volume > = 0f & & volume < = 1f ) {
2015-03-03 06:40:51 +01:00
GameOption . MASTER_VOLUME . setValue ( ( int ) ( volume * 100f ) ) ;
2015-03-27 01:45:33 +01:00
MusicController . setVolume ( getMasterVolume ( ) * getMusicVolume ( ) ) ;
2015-01-20 20:52:02 +01:00
}
}
2014-06-30 04:17:04 +02:00
/ * *
* Returns the default music volume .
* @return the volume [ 0 , 1 ]
* /
2015-03-03 06:40:51 +01:00
public static float getMusicVolume ( ) { return GameOption . MUSIC_VOLUME . getIntegerValue ( ) / 100f ; }
2014-06-30 04:17:04 +02:00
2014-07-01 07:14:03 +02:00
/ * *
* Returns the default sound effect volume .
* @return the sound volume [ 0 , 1 ]
* /
2015-03-03 06:40:51 +01:00
public static float getEffectVolume ( ) { return GameOption . EFFECT_VOLUME . getIntegerValue ( ) / 100f ; }
2014-07-01 07:14:03 +02:00
2014-07-03 00:24:19 +02:00
/ * *
2014-07-09 19:36:42 +02:00
* Returns the default hit sound volume .
* @return the hit sound volume [ 0 , 1 ]
2014-07-03 00:24:19 +02:00
* /
2015-03-03 06:40:51 +01:00
public static float getHitSoundVolume ( ) { return GameOption . HITSOUND_VOLUME . getIntegerValue ( ) / 100f ; }
2014-07-03 00:24:19 +02:00
2016-09-30 20:46:38 +02:00
/ * *
* Returns the default hit sound volume .
* @return the hit sound volume [ 0 , 1 ]
* /
2016-10-01 10:46:28 +02:00
public static float getSampleVolumeOverride ( ) { return GameOption . SAMPLE_VOLUME_OVERRIDE . val / 100f ; }
2016-09-30 20:46:38 +02:00
2014-06-30 04:17:04 +02:00
/ * *
* Returns the music offset time .
* @return the offset ( in milliseconds )
* /
2015-03-03 06:40:51 +01:00
public static int getMusicOffset ( ) { return GameOption . MUSIC_OFFSET . getIntegerValue ( ) ; }
2014-06-30 04:17:04 +02:00
/ * *
2014-07-02 01:32:03 +02:00
* Returns the screenshot file format .
* @return the file extension ( " png " , " jpg " , " bmp " )
2014-06-30 04:17:04 +02:00
* /
2017-01-29 00:20:56 +01:00
public static String getScreenshotFormat ( ) { return screenshotFormat [ screenshotFormatIndex ] . toLowerCase ( ) ; }
2014-06-30 04:17:04 +02:00
/ * *
2014-07-11 02:35:07 +02:00
* Sets the container size and makes the window borderless if the container
* size is identical to the screen resolution .
* < p >
* If the configured resolution is larger than the screen size , the smallest
* available resolution will be used .
* /
2017-01-17 23:18:12 +01:00
public static void setDisplayMode ( DisplayContainer container ) {
int screenWidth = container . nativeDisplayMode . getWidth ( ) ;
int screenHeight = container . nativeDisplayMode . getHeight ( ) ;
2015-01-18 21:26:00 +01:00
2016-10-11 11:37:47 +02:00
resolutions [ 0 ] = screenWidth + " x " + screenHeight ;
if ( resolutionIdx < 0 | | resolutionIdx > resolutions . length ) {
resolutionIdx = 0 ;
}
if ( ! resolutions [ resolutionIdx ] . matches ( " ^[0-9]+x[0-9]+$ " ) ) {
resolutionIdx = 0 ;
}
String [ ] res = resolutions [ resolutionIdx ] . split ( " x " ) ;
width = Integer . parseInt ( res [ 0 ] ) ;
height = Integer . parseInt ( res [ 1 ] ) ;
2015-01-15 07:52:16 +01:00
// check for larger-than-screen dimensions
2016-10-11 11:40:16 +02:00
if ( ! GameOption . ALLOW_LARGER_RESOLUTIONS . getBooleanValue ( ) & & ( screenWidth < width | | screenHeight < height ) ) {
2016-10-11 11:37:47 +02:00
width = 800 ;
height = 600 ;
}
2015-01-15 07:52:16 +01:00
2015-01-22 00:56:53 +01:00
try {
2017-01-17 23:18:12 +01:00
container . setDisplayMode ( width , height , isFullscreen ( ) ) ;
} catch ( Exception e ) {
2017-01-18 21:55:06 +01:00
container . eventBus . post ( new BubbleNotificationEvent ( " Failed to change resolution " , BubbleNotificationEvent . COMMONCOLOR_RED ) ) ;
2017-01-17 23:18:12 +01:00
Log . error ( " Failed to set display mode. " , e ) ;
2015-01-22 00:56:53 +01:00
}
2015-01-15 07:52:16 +01:00
2016-10-11 11:22:32 +02:00
if ( ! isFullscreen ( ) ) {
// set borderless window if dimensions match screen size
2016-10-11 11:37:47 +02:00
boolean borderless = ( screenWidth = = width & & screenHeight = = height ) ;
2016-10-11 11:22:32 +02:00
System . setProperty ( " org.lwjgl.opengl.Window.undecorated " , Boolean . toString ( borderless ) ) ;
}
2014-07-11 02:35:07 +02:00
}
2014-06-30 04:17:04 +02:00
2017-01-01 23:47:11 +01:00
public static int getExgonDelay ( ) {
return GameOption . DANCE_EXGON_DELAY . getIntegerValue ( ) ;
}
2016-12-11 13:25:04 +01:00
public static int getQuadBezAggressiveness ( ) {
return GameOption . DANCE_QUAD_BEZ_AGGRESSIVENESS . getIntegerValue ( ) ;
}
public static int getQuadBezSliderAggressiveness ( ) {
return GameOption . DANCE_QUAD_BEZ_SLIDER_AGGRESSIVENESS_FACTOR . getIntegerValue ( ) ;
}
public static boolean isQuadBezCubicEnabled ( ) {
return GameOption . DANCE_QUAD_BEZ_USE_CUBIC_ON_SLIDERS . getBooleanValue ( ) ;
}
public static int getQuadBezSliderEntryAggressiveness ( ) {
return GameOption . DANCE_QUAD_BEZ_CUBIC_AGGRESSIVENESS_FACTOR . getIntegerValue ( ) ;
}
public static int getSpinnerDelay ( ) {
return GameOption . DANCE_SPINNER_DELAY . getIntegerValue ( ) ;
}
public static boolean isLazySliders ( ) {
return GameOption . DANCE_LAZY_SLIDERS . getBooleanValue ( ) ;
}
public static boolean isOnlyCircleStacks ( ) {
return GameOption . DANCE_ONLY_CIRCLE_STACKS . getBooleanValue ( ) ;
}
public static boolean isCircleStreams ( ) {
return GameOption . DANCE_CIRCLE_STREAMS . getBooleanValue ( ) ;
}
public static boolean isMirror ( ) {
return GameOption . DANCE_MIRROR . getBooleanValue ( ) ;
}
public static void setMirror ( boolean mirror ) {
GameOption . DANCE_MIRROR . setValue ( mirror ) ;
}
public static boolean isDrawApproach ( ) {
return GameOption . DANCE_DRAW_APPROACH . getBooleanValue ( ) ;
}
public static int getRGBObjInc ( ) {
return GameOption . DANCE_RGB_OBJECT_INC . getIntegerValue ( ) ;
}
public static boolean isCursorOnlyColorTrail ( ) {
return GameOption . DANCE_CURSOR_ONLY_COLOR_TRAIL . getBooleanValue ( ) ;
}
public static int getRGBCursorInc ( ) {
return GameOption . DANCE_RGB_CURSOR_INC . getIntegerValue ( ) ;
}
public static int getCursorTrailOverride ( ) {
return GameOption . DANCE_CURSOR_TRAIL_OVERRIDE . getIntegerValue ( ) ;
}
public static boolean isHideObjects ( ) {
return GameOption . DANCE_HIDE_OBJECTS . getBooleanValue ( ) ;
}
public static boolean isRemoveBG ( ) {
return GameOption . DANCE_REMOVE_BG . getBooleanValue ( ) ;
}
public static boolean isCircleInSlowSliders ( ) {
return GameOption . DANCE_CIRLCE_IN_SLOW_SLIDERS . getBooleanValue ( ) ;
}
public static boolean isCircleInLazySliders ( ) {
return GameOption . DANCE_CIRLCE_IN_LAZY_SLIDERS . getBooleanValue ( ) ;
}
public static boolean isHideUI ( ) {
return GameOption . DANCE_HIDE_UI . getBooleanValue ( ) ;
}
public static boolean isEnableSB ( ) {
return GameOption . DANCE_ENABLE_SB . getBooleanValue ( ) ;
}
public static boolean isPippiEnabled ( ) {
return GameOption . PIPPI_ENABLE . getBooleanValue ( ) ;
}
public static int getPippiAngIncMultiplier ( ) {
return GameOption . PIPPI_ANGLE_INC_MUL . getIntegerValue ( ) ;
}
public static int getPippiAngIncMultiplierSlider ( ) {
return GameOption . PIPPI_ANGLE_INC_MUL_SLIDER . getIntegerValue ( ) ;
}
public static boolean isPippiFollowcircleExpand ( ) {
return GameOption . PIPPI_SLIDER_FOLLOW_EXPAND . getBooleanValue ( ) ;
}
public static boolean isPippiPreventWobblyStreams ( ) {
return GameOption . PIPPI_PREVENT_WOBBLY_STREAMS . getBooleanValue ( ) ;
}
2016-10-11 11:22:32 +02:00
/ * *
* Returns whether or not fullscreen mode is enabled .
* @return true if enabled
* /
public static boolean isFullscreen ( ) { return GameOption . FULLSCREEN . getBooleanValue ( ) ; }
2014-06-30 04:17:04 +02:00
2014-07-02 01:32:03 +02:00
/ * *
* Returns whether or not the FPS counter display is enabled .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isFPSCounterEnabled ( ) { return GameOption . SHOW_FPS . getBooleanValue ( ) ; }
2014-07-02 01:32:03 +02:00
2017-02-05 23:36:39 +01:00
public static boolean useDeltasForFPSCounter ( ) { return GameOption . USE_FPS_DELTAS . getBooleanValue ( ) ; }
2014-06-30 04:17:04 +02:00
/ * *
* Returns whether or not hit lighting effects are enabled .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isHitLightingEnabled ( ) { return GameOption . SHOW_HIT_LIGHTING . getBooleanValue ( ) ; }
2014-06-30 04:17:04 +02:00
2016-12-10 17:33:34 +01:00
/ * *
* Returns whether or not hit animation effects are enabled .
* @return true if enabled
* /
public static boolean isHitAnimationEnabled ( ) { return GameOption . SHOW_HIT_ANIMATIONS . getBooleanValue ( ) ; }
2016-12-12 08:37:21 +01:00
/ * *
* Returns whether or not hit animation effects are enabled .
* @return true if enabled
* /
public static boolean isReverseArrowAnimationEnabled ( ) { return GameOption . SHOW_REVERSEARROW_ANIMATIONS . getBooleanValue ( ) ; }
2014-06-30 04:17:04 +02:00
/ * *
* Returns whether or not combo burst effects are enabled .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isComboBurstEnabled ( ) { return GameOption . SHOW_COMBO_BURSTS . getBooleanValue ( ) ; }
2014-06-30 04:17:04 +02:00
2014-06-30 18:37:37 +02:00
/ * *
* Returns the port number to bind to .
* @return the port
* /
2014-07-02 01:32:03 +02:00
public static int getPort ( ) { return port ; }
2016-10-14 23:54:06 +02:00
public static boolean noSingleInstance ( ) { return noSingleInstance ; }
2015-06-14 19:30:33 +02:00
/ * *
* Returns the cursor scale .
* @return the scale [ 0 . 5 , 2 ]
* /
public static float getCursorScale ( ) { return GameOption . CURSOR_SIZE . getIntegerValue ( ) / 100f ; }
2014-07-02 01:32:03 +02:00
/ * *
* Returns whether or not the new cursor type is enabled .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isNewCursorEnabled ( ) { return GameOption . NEW_CURSOR . getBooleanValue ( ) ; }
2014-06-30 18:37:37 +02:00
2014-07-02 09:02:11 +02:00
/ * *
* Returns whether or not the main menu background should be the current track image .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isDynamicBackgroundEnabled ( ) { return GameOption . DYNAMIC_BACKGROUND . getBooleanValue ( ) ; }
2014-07-02 09:02:11 +02:00
2014-07-03 00:24:19 +02:00
/ * *
* Returns whether or not to show perfect hit result bursts .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isPerfectHitBurstEnabled ( ) { return GameOption . SHOW_PERFECT_HIT . getBooleanValue ( ) ; }
2014-07-03 00:24:19 +02:00
2015-03-19 04:23:34 +01:00
/ * *
* Returns whether or not to show follow points .
* @return true if enabled
* /
public static boolean isFollowPointEnabled ( ) { return GameOption . SHOW_FOLLOW_POINTS . getBooleanValue ( ) ; }
2014-07-03 00:24:19 +02:00
/ * *
* Returns the background dim level .
* @return the alpha level [ 0 , 1 ]
* /
2015-03-03 06:40:51 +01:00
public static float getBackgroundDim ( ) { return ( 100 - GameOption . BACKGROUND_DIM . getIntegerValue ( ) ) / 100f ; }
2014-07-03 00:24:19 +02:00
2014-07-03 07:05:23 +02:00
/ * *
* Returns whether or not to override the song background with the default playfield background .
* @return true if forced
* /
2015-03-03 06:40:51 +01:00
public static boolean isDefaultPlayfieldForced ( ) { return GameOption . FORCE_DEFAULT_PLAYFIELD . getBooleanValue ( ) ; }
2014-07-03 07:05:23 +02:00
2014-07-04 22:41:52 +02:00
/ * *
* Returns whether or not beatmap skins are ignored .
* @return true if ignored
* /
2015-03-03 06:40:51 +01:00
public static boolean isBeatmapSkinIgnored ( ) { return GameOption . IGNORE_BEATMAP_SKINS . getBooleanValue ( ) ; }
2014-07-04 22:41:52 +02:00
2015-09-16 17:19:23 +02:00
/ * *
* Returns whether or not sliders should snake in or just appear fully at once .
* @return true if sliders should snake in
* /
public static boolean isSliderSnaking ( ) { return GameOption . SNAKING_SLIDERS . getBooleanValue ( ) ; }
2016-10-15 00:09:52 +02:00
public static boolean isFallbackSliders ( ) { return GameOption . FALLBACK_SLIDERS . getBooleanValue ( ) ; }
2016-12-04 17:35:55 +01:00
public static boolean isShrinkingSliders ( ) { return GameOption . SHRINKING_SLIDERS . getBooleanValue ( ) ; }
public static boolean isMergingSliders ( ) { return ! isFallbackSliders ( ) & & GameOption . MERGING_SLIDERS . getBooleanValue ( ) ; }
2016-12-25 16:57:09 +01:00
public static int getMergingSlidersMirrorPool ( ) { return GameOption . MERGING_SLIDERS_MIRROR_POOL . getIntegerValue ( ) ; }
2016-12-24 14:35:36 +01:00
public static boolean isDrawSliderEndCircles ( ) { return GameOption . DRAW_SLIDER_ENDCIRCLES . getBooleanValue ( ) ; }
2016-12-04 17:35:55 +01:00
2014-07-05 07:24:01 +02:00
/ * *
* Returns the fixed circle size override , if any .
2015-03-03 06:40:51 +01:00
* @return the CS value ( 0 , 10 ] , 0f if disabled
2014-07-05 07:24:01 +02:00
* /
2015-03-03 06:40:51 +01:00
public static float getFixedCS ( ) { return GameOption . FIXED_CS . getIntegerValue ( ) / 10f ; }
2014-07-05 07:24:01 +02:00
/ * *
* Returns the fixed HP drain rate override , if any .
2015-03-03 06:40:51 +01:00
* @return the HP value ( 0 , 10 ] , 0f if disabled
2014-07-05 07:24:01 +02:00
* /
2015-03-03 06:40:51 +01:00
public static float getFixedHP ( ) { return GameOption . FIXED_HP . getIntegerValue ( ) / 10f ; }
2014-07-05 07:24:01 +02:00
/ * *
* Returns the fixed approach rate override , if any .
2015-03-03 06:40:51 +01:00
* @return the AR value ( 0 , 10 ] , 0f if disabled
2014-07-05 07:24:01 +02:00
* /
2015-03-03 06:40:51 +01:00
public static float getFixedAR ( ) { return GameOption . FIXED_AR . getIntegerValue ( ) / 10f ; }
2014-07-05 07:24:01 +02:00
/ * *
* Returns the fixed overall difficulty override , if any .
2015-03-03 06:40:51 +01:00
* @return the OD value ( 0 , 10 ] , 0f if disabled
2014-07-05 07:24:01 +02:00
* /
2015-03-03 06:40:51 +01:00
public static float getFixedOD ( ) { return GameOption . FIXED_OD . getIntegerValue ( ) / 10f ; }
2014-07-05 07:24:01 +02:00
2014-07-06 03:00:52 +02:00
/ * *
* Returns whether or not to render loading text in the splash screen .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isLoadVerbose ( ) { return GameOption . LOAD_VERBOSE . getBooleanValue ( ) ; }
2014-07-06 03:00:52 +02:00
2016-12-18 21:03:19 +01:00
/ * *
* Returns whether or not to color the main menu logo .
* @return true if enabled
* /
public static boolean isColorMainMenuLogo ( ) { return GameOption . COLOR_MAIN_MENU_LOGO . getBooleanValue ( ) ; }
2014-07-09 04:17:48 +02:00
/ * *
* Returns the track checkpoint time .
* @return the checkpoint time ( in ms )
* /
2015-03-03 06:40:51 +01:00
public static int getCheckpoint ( ) { return GameOption . CHECKPOINT . getIntegerValue ( ) * 1000 ; }
2014-07-09 04:17:48 +02:00
2014-07-11 04:01:39 +02:00
/ * *
* Returns whether or not all sound effects are disabled .
* @return true if disabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isSoundDisabled ( ) { return GameOption . DISABLE_SOUNDS . getBooleanValue ( ) ; }
2014-07-11 04:01:39 +02:00
2014-08-25 05:48:52 +02:00
/ * *
* Returns whether or not to use non - English metadata where available .
* @return true if Unicode preferred
* /
2015-03-03 06:40:51 +01:00
public static boolean useUnicodeMetadata ( ) { return GameOption . SHOW_UNICODE . getBooleanValue ( ) ; }
2014-08-25 05:48:52 +02:00
2014-12-21 00:17:04 +01:00
/ * *
* Returns whether or not to play the theme song .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isThemeSongEnabled ( ) { return GameOption . ENABLE_THEME_SONG . getBooleanValue ( ) ; }
2014-12-21 00:17:04 +01:00
2015-07-03 05:16:14 +02:00
/ * *
* Returns whether or not replay seeking is enabled .
* @return true if enabled
* /
public static boolean isReplaySeekingEnabled ( ) { return GameOption . REPLAY_SEEKING . getBooleanValue ( ) ; }
2015-07-08 02:03:54 +02:00
/ * *
* Returns whether or not automatic checking for updates is disabled .
* @return true if disabled
* /
public static boolean isUpdaterDisabled ( ) { return GameOption . DISABLE_UPDATER . getBooleanValue ( ) ; }
2015-08-21 17:25:52 +02:00
/ * *
* Returns whether or not the beatmap watch service is enabled .
* @return true if enabled
* /
public static boolean isWatchServiceEnabled ( ) { return GameOption . ENABLE_WATCH_SERVICE . getBooleanValue ( ) ; }
2014-07-09 04:17:48 +02:00
/ * *
* Sets the track checkpoint time , if within bounds .
* @param time the track position ( in ms )
* @return true if within bounds
* /
public static boolean setCheckpoint ( int time ) {
if ( time > = 0 & & time < 3600 ) {
2015-03-03 06:40:51 +01:00
GameOption . CHECKPOINT . setValue ( time ) ;
2014-07-09 04:17:48 +02:00
return true ;
}
return false ;
}
2015-02-15 07:40:01 +01:00
/ * *
* Returns whether or not to show the hit error bar .
* @return true if enabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isHitErrorBarEnabled ( ) { return GameOption . SHOW_HIT_ERROR_BAR . getBooleanValue ( ) ; }
2015-03-14 13:11:33 +01:00
2016-11-13 04:17:28 +01:00
public static int getMapStartDelay ( ) { return GameOption . MAP_START_DELAY . getIntegerValue ( ) * 100 ; }
public static int getMapEndDelay ( ) { return GameOption . MAP_END_DELAY . getIntegerValue ( ) * 100 ; }
2016-11-17 00:12:44 +01:00
public static int getEpilepsyWarningLength ( ) { return GameOption . EPILEPSY_WARNING . getIntegerValue ( ) * 100 ; }
2016-11-13 04:17:28 +01:00
2015-03-14 13:11:33 +01:00
/ * *
2015-03-15 04:28:50 +01:00
* Returns whether or not to load HD ( @2x ) images .
* @return true if HD images are enabled , false if only SD images should be loaded
2015-03-14 13:11:33 +01:00
* /
2015-03-15 04:28:50 +01:00
public static boolean loadHDImages ( ) { return GameOption . LOAD_HD_IMAGES . getBooleanValue ( ) ; }
2015-02-15 07:40:01 +01:00
2015-03-03 04:12:57 +01:00
/ * *
* Returns whether or not the mouse wheel is disabled during gameplay .
* @return true if disabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isMouseWheelDisabled ( ) { return GameOption . DISABLE_MOUSE_WHEEL . getBooleanValue ( ) ; }
2015-03-03 04:12:57 +01:00
/ * *
* Returns whether or not the mouse buttons are disabled during gameplay .
* @return true if disabled
* /
2015-03-03 06:40:51 +01:00
public static boolean isMouseDisabled ( ) { return GameOption . DISABLE_MOUSE_BUTTONS . getBooleanValue ( ) ; }
2015-03-03 04:12:57 +01:00
2015-03-05 20:40:57 +01:00
/ * *
* Toggles the mouse button enabled / disabled state during gameplay and
* sends a bar notification about the action .
* /
public static void toggleMouseDisabled ( ) {
2017-01-19 16:03:53 +01:00
GameOption . DISABLE_MOUSE_BUTTONS . click ( ) ;
2017-01-21 23:52:19 +01:00
EventBus . instance . post ( new BarNotificationEvent ( ( GameOption . DISABLE_MOUSE_BUTTONS . getBooleanValue ( ) ) ?
" Mouse buttons are disabled. " : " Mouse buttons are enabled. " ) ) ;
2015-03-05 20:40:57 +01:00
}
2015-09-11 17:43:19 +02:00
/ * *
* Returns whether or not the cursor sprite should be hidden .
* @return true if disabled
* /
public static boolean isCursorDisabled ( ) { return GameOption . DISABLE_CURSOR . getBooleanValue ( ) ; }
2014-07-18 05:58:37 +02:00
/ * *
* Returns the left game key .
* @return the left key code
* /
public static int getGameKeyLeft ( ) {
if ( keyLeft = = Keyboard . KEY_NONE )
2015-03-07 05:49:59 +01:00
setGameKeyLeft ( Input . KEY_Z ) ;
2014-07-18 05:58:37 +02:00
return keyLeft ;
}
/ * *
* Returns the right game key .
* @return the right key code
* /
public static int getGameKeyRight ( ) {
if ( keyRight = = Keyboard . KEY_NONE )
2015-03-07 05:49:59 +01:00
setGameKeyRight ( Input . KEY_X ) ;
2014-07-18 05:58:37 +02:00
return keyRight ;
}
2015-01-21 05:56:10 +01:00
/ * *
* Sets the left game key .
2015-03-07 20:24:26 +01:00
* This will not be set to the same key as the right game key , nor to any
* reserved keys ( see { @link # isValidGameKey ( int ) } ) .
2015-01-21 05:56:10 +01:00
* @param key the keyboard key
2015-03-07 05:49:59 +01:00
* @return { @code true } if the key was set , { @code false } if it was rejected
* /
public static boolean setGameKeyLeft ( int key ) {
2015-03-07 20:24:26 +01:00
if ( ( key = = keyRight & & key ! = Keyboard . KEY_NONE ) | | ! isValidGameKey ( key ) )
2015-03-07 05:49:59 +01:00
return false ;
keyLeft = key ;
return true ;
}
2015-01-21 05:56:10 +01:00
/ * *
* Sets the right game key .
2015-03-07 20:24:26 +01:00
* This will not be set to the same key as the left game key , nor to any
* reserved keys ( see { @link # isValidGameKey ( int ) } ) .
2015-01-21 05:56:10 +01:00
* @param key the keyboard key
2015-03-07 05:49:59 +01:00
* @return { @code true } if the key was set , { @code false } if it was rejected
* /
public static boolean setGameKeyRight ( int key ) {
2015-03-07 20:24:26 +01:00
if ( ( key = = keyLeft & & key ! = Keyboard . KEY_NONE ) | | ! isValidGameKey ( key ) )
2015-03-07 05:49:59 +01:00
return false ;
keyRight = key ;
return true ;
}
2015-01-21 05:56:10 +01:00
2015-03-07 20:24:26 +01:00
/ * *
* Checks if the given key is a valid game key .
* @param key the keyboard key
* @return { @code true } if valid , { @code false } otherwise
* /
private static boolean isValidGameKey ( int key ) {
return ( key ! = Keyboard . KEY_ESCAPE & & key ! = Keyboard . KEY_SPACE & &
key ! = Keyboard . KEY_UP & & key ! = Keyboard . KEY_DOWN & &
key ! = Keyboard . KEY_F7 & & key ! = Keyboard . KEY_F10 & & key ! = Keyboard . KEY_F12 ) ;
}
2014-06-30 04:17:04 +02:00
/ * *
2014-07-06 07:58:44 +02:00
* Returns the beatmap directory .
2014-06-30 04:17:04 +02:00
* If invalid , this will attempt to search for the directory ,
* and if nothing found , will create one .
2014-07-02 07:53:42 +02:00
* @return the beatmap directory
2014-06-30 04:17:04 +02:00
* /
public static File getBeatmapDir ( ) {
if ( beatmapDir ! = null & & beatmapDir . isDirectory ( ) )
return beatmapDir ;
2015-11-01 03:20:08 +01:00
// use osu! installation directory, if found
File osuDir = getOsuInstallationDirectory ( ) ;
if ( osuDir ! = null ) {
beatmapDir = new File ( osuDir , BEATMAP_DIR . getName ( ) ) ;
2014-07-02 01:32:03 +02:00
if ( beatmapDir . isDirectory ( ) )
2014-06-30 04:17:04 +02:00
return beatmapDir ;
}
2015-07-11 17:51:52 +02:00
2015-11-01 03:20:08 +01:00
// use default directory
beatmapDir = BEATMAP_DIR ;
if ( ! beatmapDir . isDirectory ( ) & & ! beatmapDir . mkdir ( ) )
2017-01-21 01:16:27 +01:00
EventBus . instance . post ( new BubbleNotificationEvent ( String . format ( " Failed to create beatmap directory at '%s'. " , beatmapDir . getAbsolutePath ( ) ) , BubbleNotificationEvent . COMMONCOLOR_RED ) ) ;
2014-06-30 04:17:04 +02:00
return beatmapDir ;
}
2014-07-06 07:58:44 +02:00
/ * *
* Returns the OSZ archive directory .
2014-07-18 06:56:37 +02:00
* If invalid , this will create and return a " SongPacks " directory .
2014-07-06 07:58:44 +02:00
* @return the OSZ archive directory
* /
public static File getOSZDir ( ) {
if ( oszDir ! = null & & oszDir . isDirectory ( ) )
return oszDir ;
2015-02-13 21:03:17 +01:00
oszDir = new File ( DATA_DIR , " SongPacks/ " ) ;
2015-07-11 17:51:52 +02:00
if ( ! oszDir . isDirectory ( ) & & ! oszDir . mkdir ( ) )
2017-01-21 01:16:27 +01:00
EventBus . instance . post ( new BubbleNotificationEvent ( String . format ( " Failed to create song packs directory at '%s'. " , oszDir . getAbsolutePath ( ) ) , BubbleNotificationEvent . COMMONCOLOR_RED ) ) ;
2014-07-06 07:58:44 +02:00
return oszDir ;
}
2015-06-30 02:22:38 +02:00
2015-04-02 04:10:36 +02:00
/ * *
* Returns the replay import directory.
2015-04-05 17:24:05 +02:00
* If invalid , this will create and return a " ReplayImport " directory .
* @return the replay import directory
2015-04-02 04:10:36 +02:00
* /
public static File getReplayImportDir ( ) {
if ( replayImportDir ! = null & & replayImportDir . isDirectory ( ) )
return replayImportDir ;
2014-07-06 07:58:44 +02:00
2015-04-02 04:10:36 +02:00
replayImportDir = new File ( DATA_DIR , " ReplayImport/ " ) ;
2015-07-11 17:51:52 +02:00
if ( ! replayImportDir . isDirectory ( ) & & ! replayImportDir . mkdir ( ) )
2017-01-21 01:16:27 +01:00
EventBus . instance . post ( new BubbleNotificationEvent ( String . format ( " Failed to create replay import directory at '%s'. " , replayImportDir . getAbsolutePath ( ) ) , BubbleNotificationEvent . COMMONCOLOR_RED ) ) ;
2015-04-02 04:10:36 +02:00
return replayImportDir ;
}
2015-06-30 02:22:38 +02:00
2014-07-18 06:56:37 +02:00
/ * *
* Returns the screenshot directory .
* If invalid , this will return a " Screenshot " directory .
* @return the screenshot directory
* /
public static File getScreenshotDir ( ) {
if ( screenshotDir ! = null & & screenshotDir . isDirectory ( ) )
return screenshotDir ;
2015-02-13 21:03:17 +01:00
screenshotDir = new File ( DATA_DIR , " Screenshots/ " ) ;
2014-07-18 06:56:37 +02:00
return screenshotDir ;
}
2015-03-12 01:52:51 +01:00
/ * *
* Returns the replay directory .
* If invalid , this will return a " Replay " directory .
* @return the replay directory
* /
public static File getReplayDir ( ) {
if ( replayDir ! = null & & replayDir . isDirectory ( ) )
return replayDir ;
replayDir = new File ( DATA_DIR , " Replays/ " ) ;
return replayDir ;
}
2014-07-02 07:53:42 +02:00
/ * *
* Returns the current skin directory .
* If invalid , this will create a " Skins " folder in the root directory .
* @return the skin directory
* /
2015-05-24 05:48:28 +02:00
public static File getSkinRootDir ( ) {
if ( skinRootDir ! = null & & skinRootDir . isDirectory ( ) )
return skinRootDir ;
2014-07-02 07:53:42 +02:00
2015-11-01 03:20:08 +01:00
// use osu! installation directory, if found
File osuDir = getOsuInstallationDirectory ( ) ;
if ( osuDir ! = null ) {
skinRootDir = new File ( osuDir , SKIN_ROOT_DIR . getName ( ) ) ;
2015-05-24 05:48:28 +02:00
if ( skinRootDir . isDirectory ( ) )
return skinRootDir ;
}
2015-07-11 17:51:52 +02:00
2015-11-01 03:20:08 +01:00
// use default directory
skinRootDir = SKIN_ROOT_DIR ;
if ( ! skinRootDir . isDirectory ( ) & & ! skinRootDir . mkdir ( ) )
2017-01-21 01:16:27 +01:00
EventBus . instance . post ( new BubbleNotificationEvent ( String . format ( " Failed to create skins directory at '%s'. " , skinRootDir . getAbsolutePath ( ) ) , BubbleNotificationEvent . COMMONCOLOR_RED ) ) ;
2015-05-24 05:48:28 +02:00
return skinRootDir ;
2014-07-02 07:53:42 +02:00
}
2014-07-02 01:32:03 +02:00
2017-01-21 15:18:04 +01:00
public static void reloadSkin ( ) {
loadSkin ( ) ;
SoundController . init ( ) ;
2017-01-21 15:30:07 +01:00
EventBus . instance . post ( new ResolutionOrSkinChangedEvent ( ) ) ;
2017-01-21 15:18:04 +01:00
}
2015-05-24 05:48:28 +02:00
/ * *
* Loads the skin given by the current skin directory .
* If the directory is invalid , the default skin will be loaded .
* /
public static void loadSkin ( ) {
2015-06-03 12:23:23 +02:00
File skinDir = getSkinDir ( ) ;
if ( skinDir = = null ) // invalid skin name
2015-05-24 05:48:28 +02:00
skinName = Skin . DEFAULT_SKIN_NAME ;
// create available skins list
2015-06-03 12:23:23 +02:00
File [ ] dirs = SkinLoader . getSkinDirectories ( getSkinRootDir ( ) ) ;
2015-05-24 05:48:28 +02:00
skinDirs = new String [ dirs . length + 1 ] ;
skinDirs [ 0 ] = Skin . DEFAULT_SKIN_NAME ;
for ( int i = 0 ; i < dirs . length ; i + + )
skinDirs [ i + 1 ] = dirs [ i ] . getName ( ) ;
// set skin and modify resource locations
ResourceLoader . removeAllResourceLocations ( ) ;
if ( skinDir = = null )
skin = new Skin ( null ) ;
else {
// set skin index
for ( int i = 1 ; i < skinDirs . length ; i + + ) {
if ( skinDirs [ i ] . equals ( skinName ) ) {
skinDirIndex = i ;
break ;
}
}
// load the skin
skin = SkinLoader . loadSkin ( skinDir ) ;
ResourceLoader . addResourceLocation ( new FileSystemLocation ( skinDir ) ) ;
}
ResourceLoader . addResourceLocation ( new ClasspathLocation ( ) ) ;
ResourceLoader . addResourceLocation ( new FileSystemLocation ( new File ( " . " ) ) ) ;
ResourceLoader . addResourceLocation ( new FileSystemLocation ( new File ( " ./res/ " ) ) ) ;
}
/ * *
* Returns the current skin .
* @return the skin , or null if no skin is loaded ( see { @link # loadSkin ( ) } )
* /
public static Skin getSkin ( ) { return skin ; }
2015-06-03 12:23:23 +02:00
/ * *
* Returns the current skin directory .
* < p >
* NOTE : This directory will differ from that of the currently loaded skin
* if { @link # loadSkin ( ) } has not been called after a directory change .
* Use { @link Skin # getDirectory ( ) } to get the directory of the currently
* loaded skin .
* @return the skin directory , or null for the default skin
* /
public static File getSkinDir ( ) {
File root = getSkinRootDir ( ) ;
File dir = new File ( root , skinName ) ;
return ( dir . isDirectory ( ) ) ? dir : null ;
}
2014-12-21 03:35:18 +01:00
/ * *
2015-05-17 03:25:19 +02:00
* Returns a dummy Beatmap containing the theme song .
2016-12-23 00:36:31 +01:00
* @return the theme song beatmap , or { @code null } if the theme string is malformed
2014-12-21 03:35:18 +01:00
* /
2015-05-17 03:25:19 +02:00
public static Beatmap getThemeBeatmap ( ) {
2014-12-21 03:35:18 +01:00
String [ ] tokens = themeString . split ( " , " ) ;
2016-12-23 00:36:31 +01:00
if ( tokens . length ! = 4 )
2014-12-21 03:35:18 +01:00
return null ;
2015-05-17 03:25:19 +02:00
Beatmap beatmap = new Beatmap ( null ) ;
beatmap . audioFilename = new File ( tokens [ 0 ] ) ;
beatmap . title = tokens [ 1 ] ;
beatmap . artist = tokens [ 2 ] ;
2014-12-21 03:35:18 +01:00
try {
2015-05-17 03:25:19 +02:00
beatmap . endTime = Integer . parseInt ( tokens [ 3 ] ) ;
2014-12-21 03:35:18 +01:00
} catch ( NumberFormatException e ) {
return null ;
}
2016-12-23 00:36:31 +01:00
try {
beatmap . timingPoints = new ArrayList < > ( 1 ) ;
beatmap . timingPoints . add ( new TimingPoint ( themeTimingPoint ) ) ;
} catch ( Exception e ) {
2014-12-21 03:35:18 +01:00
return null ;
}
2015-05-17 03:25:19 +02:00
return beatmap ;
2014-12-21 03:35:18 +01:00
}
2014-06-30 04:17:04 +02:00
/ * *
* Reads user options from the options file , if it exists .
* /
public static void parseOptions ( ) {
// if no config file, use default settings
2014-07-02 07:53:42 +02:00
if ( ! OPTIONS_FILE . isFile ( ) ) {
2014-06-30 04:17:04 +02:00
saveOptions ( ) ;
return ;
}
2015-06-12 22:04:20 +02:00
// create option map
if ( optionMap = = null ) {
optionMap = new HashMap < String , GameOption > ( ) ;
for ( GameOption option : GameOption . values ( ) )
optionMap . put ( option . getDisplayName ( ) , option ) ;
}
// read file
2014-07-02 07:53:42 +02:00
try ( BufferedReader in = new BufferedReader ( new FileReader ( OPTIONS_FILE ) ) ) {
2014-06-30 04:17:04 +02:00
String line ;
while ( ( line = in . readLine ( ) ) ! = null ) {
line = line . trim ( ) ;
if ( line . length ( ) < 2 | | line . charAt ( 0 ) = = '#' )
continue ;
int index = line . indexOf ( '=' ) ;
if ( index = = - 1 )
continue ;
2015-06-12 22:04:20 +02:00
// read option
String name = line . substring ( 0 , index ) . trim ( ) ;
GameOption option = optionMap . get ( name ) ;
if ( option ! = null ) {
try {
String value = line . substring ( index + 1 ) . trim ( ) ;
option . read ( value ) ;
} catch ( NumberFormatException e ) {
Log . warn ( String . format ( " Format error in options file for line: '%s'. " , line ) , e ) ;
2014-07-18 05:58:37 +02:00
}
2014-06-30 04:17:04 +02:00
}
}
} catch ( IOException e ) {
2017-01-21 01:16:27 +01:00
String err = String . format ( " Failed to read file '%s'. " , OPTIONS_FILE . getAbsolutePath ( ) ) ;
Log . error ( err , e ) ;
EventBus . instance . post ( new BubbleNotificationEvent ( err , BubbleNotificationEvent . COMMONCOLOR_RED ) ) ;
2014-06-30 04:17:04 +02:00
}
}
/ * *
* ( Over ) writes user options to a file .
* /
public static void saveOptions ( ) {
try ( BufferedWriter writer = new BufferedWriter ( new OutputStreamWriter (
new FileOutputStream ( OPTIONS_FILE ) , " utf-8 " ) ) ) {
// header
SimpleDateFormat dateFormat = new SimpleDateFormat ( " EEEE, MMMM dd, yyyy " ) ;
String date = dateFormat . format ( new Date ( ) ) ;
writer . write ( " # opsu! configuration " ) ;
writer . newLine ( ) ;
writer . write ( " # last updated on " ) ;
writer . write ( date ) ;
writer . newLine ( ) ;
writer . newLine ( ) ;
// options
2015-06-12 22:04:20 +02:00
for ( GameOption option : GameOption . values ( ) ) {
writer . write ( option . getDisplayName ( ) ) ;
writer . write ( " = " ) ;
writer . write ( option . write ( ) ) ;
writer . newLine ( ) ;
}
2014-06-30 04:17:04 +02:00
writer . close ( ) ;
} catch ( IOException e ) {
2017-01-21 01:16:27 +01:00
String err = String . format ( " Failed to write to file '%s'. " , OPTIONS_FILE . getAbsolutePath ( ) ) ;
Log . error ( err , e ) ;
EventBus . instance . post ( new BubbleNotificationEvent ( err , BubbleNotificationEvent . COMMONCOLOR_RED ) ) ;
2014-06-30 04:17:04 +02:00
}
}
}