From 636746490e7ff2e5b7c6a71b656c81c21b441bf3 Mon Sep 17 00:00:00 2001 From: John Peterson Date: Sat, 20 Jun 2009 10:58:15 +0000 Subject: [PATCH] MusicMod: Prepare to fix the MusicMod build git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@3505 8ced0084-cf51-0410-be5f-012b33b47a6e --- Externals/MusicMod/Common/Common.vcproj | 342 ++++ Externals/MusicMod/Data/win32/Plainamp.ini | 40 + .../Data/win32/PluginsMusic/in_vgmstream.dll | Bin 0 -> 331776 bytes .../Data/win32/PluginsMusic/out_ds.dll | Bin 0 -> 40960 bytes .../Data/win32/PluginsMusic/out_wave_gpl.dll | Bin 0 -> 14336 bytes Externals/MusicMod/Data/x64/Plainamp.ini | 29 + .../Data/x64/PluginsMusic/in_vgmstream.dll | Bin 0 -> 177152 bytes .../Data/x64/PluginsMusic/out_wave_gpl.dll | Bin 0 -> 55296 bytes Externals/MusicMod/Main/Main.vcproj | 417 +++++ Externals/MusicMod/Main/Src/Frame.cpp | 466 ++++++ Externals/MusicMod/Main/Src/Main.cpp | 342 ++++ Externals/MusicMod/Main/Src/Main.h | 31 + Externals/MusicMod/Player/Lib/fftw3.lib | Bin 0 -> 31296 bytes Externals/MusicMod/Player/Lib/libzlib1.lib | Bin 0 -> 37266 bytes Externals/MusicMod/Player/Player.vcproj | 925 +++++++++++ .../MusicMod/Player/Src/AddDirectory.cpp | 364 ++++ Externals/MusicMod/Player/Src/AddDirectory.h | 26 + Externals/MusicMod/Player/Src/AddFiles.cpp | 181 ++ Externals/MusicMod/Player/Src/AddFiles.h | 26 + Externals/MusicMod/Player/Src/Config.cpp | 687 ++++++++ Externals/MusicMod/Player/Src/Config.h | 239 +++ Externals/MusicMod/Player/Src/Console.cpp | 228 +++ Externals/MusicMod/Player/Src/Console.h | 40 + Externals/MusicMod/Player/Src/DspModule.cpp | 136 ++ Externals/MusicMod/Player/Src/DspModule.h | 67 + Externals/MusicMod/Player/Src/DspPlugin.cpp | 186 +++ Externals/MusicMod/Player/Src/DspPlugin.h | 69 + .../MusicMod/Player/Src/Emabox/Emabox.cpp | 455 +++++ Externals/MusicMod/Player/Src/Emabox/Emabox.h | 141 ++ .../MusicMod/Player/Src/Emabox/EmaboxConfig.h | 32 + Externals/MusicMod/Player/Src/Embed.cpp | 233 +++ Externals/MusicMod/Player/Src/Embed.h | 36 + Externals/MusicMod/Player/Src/Font.cpp | 83 + Externals/MusicMod/Player/Src/Font.h | 34 + Externals/MusicMod/Player/Src/GenPlugin.cpp | 199 +++ Externals/MusicMod/Player/Src/GenPlugin.h | 64 + Externals/MusicMod/Player/Src/Global.h | 151 ++ Externals/MusicMod/Player/Src/GlobalVersion.h | 28 + Externals/MusicMod/Player/Src/Input.cpp | 483 ++++++ Externals/MusicMod/Player/Src/Input.h | 28 + Externals/MusicMod/Player/Src/InputPlugin.cpp | 389 +++++ Externals/MusicMod/Player/Src/InputPlugin.h | 96 ++ Externals/MusicMod/Player/Src/Lock.cpp | 16 + Externals/MusicMod/Player/Src/Lock.h | 111 ++ Externals/MusicMod/Player/Src/Main.cpp | 816 +++++++++ Externals/MusicMod/Player/Src/Main.h | 51 + Externals/MusicMod/Player/Src/Output.cpp | 382 +++++ Externals/MusicMod/Player/Src/Output.h | 27 + .../MusicMod/Player/Src/OutputPlugin.cpp | 316 ++++ Externals/MusicMod/Player/Src/OutputPlugin.h | 87 + Externals/MusicMod/Player/Src/Path.cpp | 243 +++ Externals/MusicMod/Player/Src/Path.h | 31 + Externals/MusicMod/Player/Src/Playback.cpp | 804 +++++++++ Externals/MusicMod/Player/Src/Playback.h | 106 ++ Externals/MusicMod/Player/Src/PlaybackEq.cpp | 223 +++ .../MusicMod/Player/Src/PlaybackOrder.cpp | 205 +++ Externals/MusicMod/Player/Src/Player.cpp | 380 +++++ .../MusicMod/Player/Src/PlayerExport.cpp | 181 ++ Externals/MusicMod/Player/Src/PlayerExport.h | 49 + Externals/MusicMod/Player/Src/Playlist.cpp | 1383 ++++++++++++++++ Externals/MusicMod/Player/Src/Playlist.h | 65 + .../MusicMod/Player/Src/PlaylistControler.cpp | 467 ++++++ .../MusicMod/Player/Src/PlaylistControler.h | 65 + Externals/MusicMod/Player/Src/PlaylistModel.h | 122 ++ .../MusicMod/Player/Src/PlaylistView.cpp | 782 +++++++++ Externals/MusicMod/Player/Src/PlaylistView.h | 27 + Externals/MusicMod/Player/Src/Plugin.cpp | 63 + Externals/MusicMod/Player/Src/Plugin.h | 142 ++ .../MusicMod/Player/Src/PluginManager.cpp | 1143 +++++++++++++ Externals/MusicMod/Player/Src/PluginManager.h | 42 + Externals/MusicMod/Player/Src/Prefs.cpp | 565 +++++++ Externals/MusicMod/Player/Src/Prefs.h | 36 + Externals/MusicMod/Player/Src/Rebar.cpp | 1457 +++++++++++++++++ Externals/MusicMod/Player/Src/Rebar.h | 40 + .../MusicMod/Player/Src/Resources/Buttons.bmp | Bin 0 -> 734 bytes .../Player/Src/Resources/Plainamp.ico | Bin 0 -> 38638 bytes .../MusicMod/Player/Src/Resources/resrc1.aps | Bin 0 -> 73968 bytes .../MusicMod/Player/Src/Resources/resrc1.h | 17 + .../MusicMod/Player/Src/Resources/resrc1.rc | 131 ++ Externals/MusicMod/Player/Src/Status.cpp | 80 + Externals/MusicMod/Player/Src/Status.h | 33 + Externals/MusicMod/Player/Src/Timer.cpp | 126 ++ Externals/MusicMod/Player/Src/Unicode.cpp | 77 + Externals/MusicMod/Player/Src/Unicode.h | 29 + Externals/MusicMod/Player/Src/Util.cpp | 48 + Externals/MusicMod/Player/Src/Util.h | 26 + Externals/MusicMod/Player/Src/VisCache.cpp | 303 ++++ Externals/MusicMod/Player/Src/VisCache.h | 50 + Externals/MusicMod/Player/Src/VisModule.cpp | 358 ++++ Externals/MusicMod/Player/Src/VisModule.h | 71 + Externals/MusicMod/Player/Src/VisPlugin.cpp | 189 +++ Externals/MusicMod/Player/Src/VisPlugin.h | 68 + Externals/MusicMod/Player/Src/Winamp.cpp | 996 +++++++++++ Externals/MusicMod/Player/Src/Winamp.h | 26 + Externals/MusicMod/Player/Src/Winamp/Dsp.h | 47 + Externals/MusicMod/Player/Src/Winamp/Gen.h | 28 + Externals/MusicMod/Player/Src/Winamp/In2.h | 119 ++ Externals/MusicMod/Player/Src/Winamp/Out.h | 69 + Externals/MusicMod/Player/Src/Winamp/Vis.h | 57 + Externals/MusicMod/Player/Src/Winamp/wa_ipc.h | 1103 +++++++++++++ .../MusicMod/Player/Src/Winamp/wa_msgids.h | 297 ++++ Externals/MusicMod/Player/Src/Winmain.cpp | 236 +++ Externals/MusicMod/Player/Src/afxres.h | 823 ++++++++++ Externals/MusicMod/Player/Src/fftw3/fftw3.h | 276 ++++ .../Player/Src/vis_plainbar/vis_plainbar.cpp | 259 +++ Externals/MusicMod/Player/Src/zlib/zconf.h | 332 ++++ Externals/MusicMod/Player/Src/zlib/zlib.h | 1357 +++++++++++++++ Externals/MusicMod/TestPlayer/Src/Player.cpp | 37 + .../MusicMod/TestPlayer/TestPlayer.vcproj | 403 +++++ Source/MusicMod.sln | 498 ++++++ 110 files changed, 26289 insertions(+) create mode 100644 Externals/MusicMod/Common/Common.vcproj create mode 100644 Externals/MusicMod/Data/win32/Plainamp.ini create mode 100644 Externals/MusicMod/Data/win32/PluginsMusic/in_vgmstream.dll create mode 100644 Externals/MusicMod/Data/win32/PluginsMusic/out_ds.dll create mode 100644 Externals/MusicMod/Data/win32/PluginsMusic/out_wave_gpl.dll create mode 100644 Externals/MusicMod/Data/x64/Plainamp.ini create mode 100644 Externals/MusicMod/Data/x64/PluginsMusic/in_vgmstream.dll create mode 100644 Externals/MusicMod/Data/x64/PluginsMusic/out_wave_gpl.dll create mode 100644 Externals/MusicMod/Main/Main.vcproj create mode 100644 Externals/MusicMod/Main/Src/Frame.cpp create mode 100644 Externals/MusicMod/Main/Src/Main.cpp create mode 100644 Externals/MusicMod/Main/Src/Main.h create mode 100644 Externals/MusicMod/Player/Lib/fftw3.lib create mode 100644 Externals/MusicMod/Player/Lib/libzlib1.lib create mode 100644 Externals/MusicMod/Player/Player.vcproj create mode 100644 Externals/MusicMod/Player/Src/AddDirectory.cpp create mode 100644 Externals/MusicMod/Player/Src/AddDirectory.h create mode 100644 Externals/MusicMod/Player/Src/AddFiles.cpp create mode 100644 Externals/MusicMod/Player/Src/AddFiles.h create mode 100644 Externals/MusicMod/Player/Src/Config.cpp create mode 100644 Externals/MusicMod/Player/Src/Config.h create mode 100644 Externals/MusicMod/Player/Src/Console.cpp create mode 100644 Externals/MusicMod/Player/Src/Console.h create mode 100644 Externals/MusicMod/Player/Src/DspModule.cpp create mode 100644 Externals/MusicMod/Player/Src/DspModule.h create mode 100644 Externals/MusicMod/Player/Src/DspPlugin.cpp create mode 100644 Externals/MusicMod/Player/Src/DspPlugin.h create mode 100644 Externals/MusicMod/Player/Src/Emabox/Emabox.cpp create mode 100644 Externals/MusicMod/Player/Src/Emabox/Emabox.h create mode 100644 Externals/MusicMod/Player/Src/Emabox/EmaboxConfig.h create mode 100644 Externals/MusicMod/Player/Src/Embed.cpp create mode 100644 Externals/MusicMod/Player/Src/Embed.h create mode 100644 Externals/MusicMod/Player/Src/Font.cpp create mode 100644 Externals/MusicMod/Player/Src/Font.h create mode 100644 Externals/MusicMod/Player/Src/GenPlugin.cpp create mode 100644 Externals/MusicMod/Player/Src/GenPlugin.h create mode 100644 Externals/MusicMod/Player/Src/Global.h create mode 100644 Externals/MusicMod/Player/Src/GlobalVersion.h create mode 100644 Externals/MusicMod/Player/Src/Input.cpp create mode 100644 Externals/MusicMod/Player/Src/Input.h create mode 100644 Externals/MusicMod/Player/Src/InputPlugin.cpp create mode 100644 Externals/MusicMod/Player/Src/InputPlugin.h create mode 100644 Externals/MusicMod/Player/Src/Lock.cpp create mode 100644 Externals/MusicMod/Player/Src/Lock.h create mode 100644 Externals/MusicMod/Player/Src/Main.cpp create mode 100644 Externals/MusicMod/Player/Src/Main.h create mode 100644 Externals/MusicMod/Player/Src/Output.cpp create mode 100644 Externals/MusicMod/Player/Src/Output.h create mode 100644 Externals/MusicMod/Player/Src/OutputPlugin.cpp create mode 100644 Externals/MusicMod/Player/Src/OutputPlugin.h create mode 100644 Externals/MusicMod/Player/Src/Path.cpp create mode 100644 Externals/MusicMod/Player/Src/Path.h create mode 100644 Externals/MusicMod/Player/Src/Playback.cpp create mode 100644 Externals/MusicMod/Player/Src/Playback.h create mode 100644 Externals/MusicMod/Player/Src/PlaybackEq.cpp create mode 100644 Externals/MusicMod/Player/Src/PlaybackOrder.cpp create mode 100644 Externals/MusicMod/Player/Src/Player.cpp create mode 100644 Externals/MusicMod/Player/Src/PlayerExport.cpp create mode 100644 Externals/MusicMod/Player/Src/PlayerExport.h create mode 100644 Externals/MusicMod/Player/Src/Playlist.cpp create mode 100644 Externals/MusicMod/Player/Src/Playlist.h create mode 100644 Externals/MusicMod/Player/Src/PlaylistControler.cpp create mode 100644 Externals/MusicMod/Player/Src/PlaylistControler.h create mode 100644 Externals/MusicMod/Player/Src/PlaylistModel.h create mode 100644 Externals/MusicMod/Player/Src/PlaylistView.cpp create mode 100644 Externals/MusicMod/Player/Src/PlaylistView.h create mode 100644 Externals/MusicMod/Player/Src/Plugin.cpp create mode 100644 Externals/MusicMod/Player/Src/Plugin.h create mode 100644 Externals/MusicMod/Player/Src/PluginManager.cpp create mode 100644 Externals/MusicMod/Player/Src/PluginManager.h create mode 100644 Externals/MusicMod/Player/Src/Prefs.cpp create mode 100644 Externals/MusicMod/Player/Src/Prefs.h create mode 100644 Externals/MusicMod/Player/Src/Rebar.cpp create mode 100644 Externals/MusicMod/Player/Src/Rebar.h create mode 100644 Externals/MusicMod/Player/Src/Resources/Buttons.bmp create mode 100644 Externals/MusicMod/Player/Src/Resources/Plainamp.ico create mode 100644 Externals/MusicMod/Player/Src/Resources/resrc1.aps create mode 100644 Externals/MusicMod/Player/Src/Resources/resrc1.h create mode 100644 Externals/MusicMod/Player/Src/Resources/resrc1.rc create mode 100644 Externals/MusicMod/Player/Src/Status.cpp create mode 100644 Externals/MusicMod/Player/Src/Status.h create mode 100644 Externals/MusicMod/Player/Src/Timer.cpp create mode 100644 Externals/MusicMod/Player/Src/Unicode.cpp create mode 100644 Externals/MusicMod/Player/Src/Unicode.h create mode 100644 Externals/MusicMod/Player/Src/Util.cpp create mode 100644 Externals/MusicMod/Player/Src/Util.h create mode 100644 Externals/MusicMod/Player/Src/VisCache.cpp create mode 100644 Externals/MusicMod/Player/Src/VisCache.h create mode 100644 Externals/MusicMod/Player/Src/VisModule.cpp create mode 100644 Externals/MusicMod/Player/Src/VisModule.h create mode 100644 Externals/MusicMod/Player/Src/VisPlugin.cpp create mode 100644 Externals/MusicMod/Player/Src/VisPlugin.h create mode 100644 Externals/MusicMod/Player/Src/Winamp.cpp create mode 100644 Externals/MusicMod/Player/Src/Winamp.h create mode 100644 Externals/MusicMod/Player/Src/Winamp/Dsp.h create mode 100644 Externals/MusicMod/Player/Src/Winamp/Gen.h create mode 100644 Externals/MusicMod/Player/Src/Winamp/In2.h create mode 100644 Externals/MusicMod/Player/Src/Winamp/Out.h create mode 100644 Externals/MusicMod/Player/Src/Winamp/Vis.h create mode 100644 Externals/MusicMod/Player/Src/Winamp/wa_ipc.h create mode 100644 Externals/MusicMod/Player/Src/Winamp/wa_msgids.h create mode 100644 Externals/MusicMod/Player/Src/Winmain.cpp create mode 100644 Externals/MusicMod/Player/Src/afxres.h create mode 100644 Externals/MusicMod/Player/Src/fftw3/fftw3.h create mode 100644 Externals/MusicMod/Player/Src/vis_plainbar/vis_plainbar.cpp create mode 100644 Externals/MusicMod/Player/Src/zlib/zconf.h create mode 100644 Externals/MusicMod/Player/Src/zlib/zlib.h create mode 100644 Externals/MusicMod/TestPlayer/Src/Player.cpp create mode 100644 Externals/MusicMod/TestPlayer/TestPlayer.vcproj create mode 100644 Source/MusicMod.sln diff --git a/Externals/MusicMod/Common/Common.vcproj b/Externals/MusicMod/Common/Common.vcproj new file mode 100644 index 0000000000..bfd7927ca2 --- /dev/null +++ b/Externals/MusicMod/Common/Common.vcproj @@ -0,0 +1,342 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Externals/MusicMod/Data/win32/Plainamp.ini b/Externals/MusicMod/Data/win32/Plainamp.ini new file mode 100644 index 0000000000..d82f44952a --- /dev/null +++ b/Externals/MusicMod/Data/win32/Plainamp.ini @@ -0,0 +1,40 @@ +[Plainamp] +OutputPluginActive___out_ds.dll=1 +Volume=255 +Loop=1 +WinPlaceConsole=(1,(62,441,1179,845)) +WinPlaceMain=(1,(274,203,1005,765)) +MinimizeToTray=1 +Panning=0 +CurPresetFixed=-1 +PreventDistortion=1 +Order=3 +PlaylistFollow=1 +PlaylistEntryNumberZeroPadding=1 +CurPlaylistPosition=0 +InfinitePlaylist=0 +ManagerGrid=1 +WinPlaceManager=(1,(500,400,1000,700)) +OrderBand=(0,-2,0,1) +EqBand=(1,-2,0,1) +SeekBand=(2,-2,0,1) +VolBand=(3,-2,0,1) +PanBand=(4,-2,0,1) +ButtonsBand=(5,134,0,1) +VisBand=(6,134,0,1) +InvertPanSlider=0 +CurDir=C:\ +WarnPluginsMissing=1 +[out_wave_gpl] +config=14000000002400000100000000000000FFFFFFFF0100000036 +[out_ds] +cfg_hw_mix=1 +cfg_buf_ms=2000 +cfg_trackhack=0 +cfg_prebuf2=500 +cfg_fade_seek.on=0 +cfg_fade_pause.on=0 +cfg_fadevol=0 +cfg_wait=0 +[Interface] +ShowConsole = False diff --git a/Externals/MusicMod/Data/win32/PluginsMusic/in_vgmstream.dll b/Externals/MusicMod/Data/win32/PluginsMusic/in_vgmstream.dll new file mode 100644 index 0000000000000000000000000000000000000000..f54a9f9e625e99abd495ffe4d883242be3ede0b8 GIT binary patch literal 331776 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ⅅKg33W>5>b`G82ki`Zje)6%+K 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

qFhH53F6U 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%0D`n?YBz5$jDQ4s*S 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 S3_J{cKq^5+(kP~52mk;Y@V|io literal 0 HcmV?d00001 diff --git a/Externals/MusicMod/Data/x64/Plainamp.ini b/Externals/MusicMod/Data/x64/Plainamp.ini new file mode 100644 index 0000000000..b1a908bd6d --- /dev/null +++ b/Externals/MusicMod/Data/x64/Plainamp.ini @@ -0,0 +1,29 @@ +[Plainamp] +OutputPluginActive___out_wave_gpl.dll=1 +Volume=100 +Loop=1 +WinPlaceConsole=(1,(62,441,1179,845)) +WinPlaceMain=(1,(274,203,1005,765)) +MinimizeToTray=1 +Panning=0 +CurPresetFixed=-1 +PreventDistortion=1 +Order=3 +PlaylistFollow=1 +PlaylistEntryNumberZeroPadding=1 +CurPlaylistPosition=0 +InfinitePlaylist=1 +ManagerGrid=1 +WinPlaceManager=(1,(500,400,1000,700)) +OrderBand=(0,-2,0,1) +EqBand=(1,-2,0,1) +SeekBand=(2,-2,0,1) +VolBand=(3,-2,0,1) +PanBand=(4,-2,0,1) +ButtonsBand=(5,134,0,1) +VisBand=(6,134,0,1) +InvertPanSlider=0 +CurDir=C:\ +WarnPluginsMissing=1 +[out_wave_gpl] +config=14000000002400000100000000000000FFFFFFFF0100000036 diff --git a/Externals/MusicMod/Data/x64/PluginsMusic/in_vgmstream.dll b/Externals/MusicMod/Data/x64/PluginsMusic/in_vgmstream.dll new file mode 100644 index 0000000000000000000000000000000000000000..bf2af34f79df624d3f71a082582cb2624bc6f187 GIT binary patch literal 177152 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%0D0h;Xe?E&<0`ED^oI!$x}X&)ARj^(WJBl$ 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 Gg#ZBH;>HjF literal 0 HcmV?d00001 diff --git a/Externals/MusicMod/Data/x64/PluginsMusic/out_wave_gpl.dll b/Externals/MusicMod/Data/x64/PluginsMusic/out_wave_gpl.dll new file mode 100644 index 0000000000000000000000000000000000000000..da0ccab700ab8be3018e49fe122c2a8890f59cb8 GIT binary patch literal 55296 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 literal 0 HcmV?d00001 diff --git a/Externals/MusicMod/Main/Main.vcproj b/Externals/MusicMod/Main/Main.vcproj new file mode 100644 index 0000000000..2f6abdff78 --- /dev/null +++ b/Externals/MusicMod/Main/Main.vcproj @@ -0,0 +1,417 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Externals/MusicMod/Main/Src/Frame.cpp b/Externals/MusicMod/Main/Src/Frame.cpp new file mode 100644 index 0000000000..a29d02c1ed --- /dev/null +++ b/Externals/MusicMod/Main/Src/Frame.cpp @@ -0,0 +1,466 @@ +// Copyright (C) 2003-2008 Dolphin Project. + +// 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, version 2.0. + +// 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 2.0 for more details. + +// A copy of the GPL 2.0 should have been included with the program. +// If not, see http://www.gnu.org/licenses/ + +// Official SVN repository and contact information can be found at +// http://code.google.com/p/dolphin-emu/ + + +////////////////////////////////////////////////////////////////////////////////////////// +// Include +// ŻŻŻŻŻŻŻŻŻŻ +#include // System + +#include "Core.h" // Core + +#include "IniFile.h" // Common +#include "Log.h" + +#include "../../../../Source/Core/DolphinWX/Src/Globals.h" // DolphinWX +#include "../../../../Source/Core/DolphinWX/Src/Frame.h" + +#include "../../../../Source/Core/DolphinWX/resources/toolbar_plugin_dsp.c" // Icons +#include "../../../../Source/Core/DolphinWX/resources/Boomy.h" +#include "../../../../Source/Core/DolphinWX/resources/Vista.h" +#include "../../../../Source/Core/DolphinWX/resources/KDE.h" +#include "../../../../Source/Core/DolphinWX/resources/X-Plastik.h" + +#include "../../Player/Src/PlayerExport.h" // Player +////////////////////////////////// + + + +////////////////////////////////////////////////////////////////////////////////////////// +// Declarations and definitions +// ŻŻŻŻŻŻŻŻŻŻ +namespace MusicMod +{ + bool GlobalMute = false; + bool GlobalPause = false; + bool bShowConsole = false; + int GlobalVolume = 125; + extern bool dllloaded; + + void ShowConsole(); + void Init(); +} +////////////////////////////////// + + + +////////////////////////////////////////////////////////////////////////////////////////// +// Change the brightness of a wxBitmap +// ŻŻŻŻŻŻŻŻŻŻ +wxBitmap SetBrightness(wxBitmap _Bitmap, int _Brightness, bool Gray) +{ + wxImage _Image = _Bitmap.ConvertToImage(); + wxImage _Image2 = _Bitmap.ConvertToImage(); + + wxString Tmp; + + if(_Brightness < 0) _Brightness = 0; // Limits + if(_Brightness > 255) _Brightness = 255; + + _Brightness = 255 - _Brightness; // Make big values brighter + + // Remove the alpha layer first + for(int y = 0; y < _Bitmap.GetWidth(); y++) + { + for(int x = 0; x < _Bitmap.GetHeight(); x++) + _Image.SetAlpha(x, y, 255); + } + + for(int y = 0; y < _Bitmap.GetWidth(); y++) + { + //Tmp += wxString::Format("\n %i: ", y); + + for(int x = 0; x < _Bitmap.GetHeight(); x++) + { + u8 R = _Image.GetRed(x, y); // Get colors + u8 G = _Image.GetGreen(x, y); + u8 B = _Image.GetBlue(x, y); + + //if((x == 5 | x == 6) && y == 15) Tmp += wxString::Format("%03i %03i %03i", R, G, B); + + if(_Brightness > 128) + { + int Bright = _Brightness - 128; + R = R - Bright * (R - 0) / 128; + G = G - Bright * (G - 0) / 128; + B = B - Bright * (B - 0) / 128; + + // 118 - 72 * 118 / 128 = 118 - 66.3 = 52 + // 119 - = 119 - 66.9 = 52 + } + else + { + int Bright = 128 - _Brightness; + R = R - Bright * (R - 255) / 128; + G = G - Bright * (G - 255) / 128; + B = B - Bright * (B - 255) / 128; + } + + //if((x == 5 | x == 6) && y == 15) Tmp += wxString::Format(" %03i %03i %03i | ", R, G, B); + + _Image.SetRGB(x, y, R, G, B); + } + } + // Return the alpha + _Image.SetAlpha(_Image2.GetAlpha(), true); + + // Update the bitmap + if(Gray) + return wxBitmap(_Image.ConvertToGreyscale()); + else + return wxBitmap(_Image); + + //wxMessageBox(Tmp); +} +////////////////////////////////// + + +#ifdef MUSICMOD +void +CFrame::MM_InitBitmaps(int Theme) +{ + // Define the log bitmaps + switch (Theme) + { + case BOOMY: + m_Bitmaps[Toolbar_Log] = wxGetBitmapFromMemory(Toolbar_Log_png); + break; + case VISTA: + m_Bitmaps[Toolbar_Log] = wxGetBitmapFromMemory(Toolbar_Log1_png); + break; + case XPLASTIK: + m_Bitmaps[Toolbar_Log] = wxGetBitmapFromMemory(Toolbar_Log2_png); + break; + case KDE: + m_Bitmaps[Toolbar_Log] = wxGetBitmapFromMemory(Toolbar_Log3_png); + break; + default: PanicAlert("Theme selection went wrong"); + } + + // Create a gray version + m_Bitmaps[Toolbar_PluginDSP_Dis] = wxBitmap(SetBrightness(m_Bitmaps[Toolbar_PluginDSP], 165, true)); + m_Bitmaps[Toolbar_Log_Dis] = wxBitmap(SetBrightness(m_Bitmaps[Toolbar_Log], 165, true)); + + // Update in case the bitmap has been updated + //if (GetToolBar() != NULL) TheToolBar->FindById(Toolbar_Log)->SetNormalBitmap(m_Bitmaps[Toolbar_Log]); +} + + +void +CFrame::MM_PopulateGUI() +{ + wxToolBar* toolBar = TheToolBar; // Shortcut + + toolBar->AddSeparator(); + + MusicMod::Init(); + + // --------------------------------------- + // Draw a rotated music label + // --------------------- + wxBitmap m_RotatedText(30, 15); + wxMemoryDC dc; + dc.SelectObject(m_RotatedText); + wxBrush BackgroundGrayBrush(_T("#ece9d8")); // The right color in windows + + // Set outline and fill colors + dc.SetBackground(BackgroundGrayBrush); + dc.Clear(); + + wxFont m_font(8, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_BOLD); + dc.SetFont(m_font); + dc.SetTextForeground(wxColour(*wxLIGHT_GREY)); + + dc.DrawText(wxT("Music"), 0, 0); + m_RotatedText = wxBitmap(m_RotatedText.ConvertToImage().Rotate90(false)); + + wxStaticBitmap * m_StaticBitmap = new wxStaticBitmap(toolBar, wxID_ANY, m_RotatedText); + + toolBar->AddControl(m_StaticBitmap); + // --------------------------- + + + + mm_ToolMute = toolBar->AddTool(IDM_MUTE, _T("Mute"), m_Bitmaps[Toolbar_PluginDSP], _T("Mute music")); + mm_ToolPlay = toolBar->AddTool(IDM_MUSIC_PLAY, _T("Play"), m_Bitmaps[Toolbar_Play], _T("Play or pause music without pausing the game")); + + // This cause the disabled tool bitmap to become some kind of monochrome version + /* + mm_ToolMute = new wxToolBarToolBase(toolBar, IDM_MUTE, _T("Mute"), m_Bitmaps[Toolbar_PluginDSP], + m_Bitmaps[Toolbar_PluginDSP], wxITEM_CHECK, 0, _T("Mute music")); + toolBar->AddTool(mm_ToolMute); + + mm_ToolPlay = new wxToolBarToolBase(toolBar, IDM_MUSIC_PLAY, _T("Play"), m_Bitmaps[Toolbar_Play], + m_Bitmaps[Toolbar_Play], wxITEM_NORMAL, 0, _T("Play or pause music without pausing the game")); + toolBar->AddTool(mm_ToolPlay); + */ + + + // --------------------- + /* Lots of code to get a label for the slider, in 2.9.0 AddControl accepts a label so then + this code can be simplified a lot */ + // --------- + wxPanel * mm_SliderPanel = new wxPanel(toolBar, IDS_VOLUME_PANEL, wxDefaultPosition, wxDefaultSize); + mm_Slider = new wxSlider(mm_SliderPanel, IDS_VOLUME, 125, 0, 255, wxDefaultPosition, wxDefaultSize); + //mm_Slider->SetToolTip("Change the music volume"); + mm_Slider->SetValue(MusicMod::GlobalVolume); + + wxStaticText * mm_SliderText = new wxStaticText(mm_SliderPanel, IDS_VOLUME_LABEL, _T("Volume"), wxDefaultPosition, wxDefaultSize); + wxBoxSizer * mm_VolSizer = new wxBoxSizer(wxVERTICAL); + mm_VolSizer->Add(mm_Slider, 0, wxEXPAND | wxALL, 0); + mm_VolSizer->Add(mm_SliderText, 0, wxCENTER | wxALL, 0); + + mm_SliderPanel->SetSizer(mm_VolSizer); + mm_SliderPanel->SetSize(mm_VolSizer->GetMinSize().GetWidth(), mm_VolSizer->GetMinSize().GetHeight()); + + toolBar->AddControl((wxControl*)mm_SliderPanel); + // --------- + + mm_ToolLog = toolBar->AddTool(IDT_LOG, _T("Log"), m_Bitmaps[Toolbar_Log], + wxT("Show or hide log. Enable the log window and restart Dolphin to show the DLL status.")); +} + + +////////////////////////////////////////////////////////////////////////////////////////// +// Update GUI +// ŻŻŻŻŻŻŻŻŻŻ +void +CFrame::MM_UpdateGUI() +{ + if(MusicMod::GlobalMute) + { + mm_ToolMute->SetLabel("Unmute"); + mm_ToolMute->SetNormalBitmap(m_Bitmaps[Toolbar_PluginDSP_Dis]); + } + else + { + mm_ToolMute->SetLabel("Mute"); + mm_ToolMute->SetNormalBitmap(m_Bitmaps[Toolbar_PluginDSP]); + } + + if(MusicMod::GlobalPause) + { + mm_ToolPlay->SetLabel("Play"); + mm_ToolPlay->SetNormalBitmap(m_Bitmaps[Toolbar_Play]); + } + else + { + mm_ToolPlay->SetLabel("Pause"); + mm_ToolPlay->SetNormalBitmap(m_Bitmaps[Toolbar_Pause]); + } + + if(MusicMod::bShowConsole) + { + mm_ToolLog->SetNormalBitmap(m_Bitmaps[Toolbar_Log]); + } + else + { + mm_ToolLog->SetNormalBitmap(m_Bitmaps[Toolbar_Log_Dis]); + } +} +////////////////////////////////// + + + +// ======================================================================================= +// Play and stop music +// --------------------------------------------------------------------------------------- +void +CFrame::MM_OnPlay() +{ + //INFO_LOG(AUDIO,"\nCFrame::OnPlayMusicMod > Begin\n"); + + // Save the volume + MusicMod::GlobalVolume = mm_Slider->GetValue(); + + IniFile file; + file.Load("Plainamp.ini"); + file.Set("Plainamp", "Volume", MusicMod::GlobalVolume); + file.Save("Plainamp.ini"); + + if (Core::GetState() != Core::CORE_UNINITIALIZED) + { + if (Core::GetState() == Core::CORE_RUN) + { + //INFO_LOG(AUDIO,"CFrame::OnPlayMusicMod > Pause\n"); + if(!MusicMod::GlobalPause) // we may has set this elsewhere + { + MusicMod::GlobalPause = true; + if (MusicMod::dllloaded) + { + Player_Pause(); + } + } + } + else + { + //INFO_LOG(AUDIO,"CFrame::OnPlayMusicMod > Play\n"); + if(MusicMod::GlobalPause) // we may has set this elsewhere + { + MusicMod::GlobalPause = false; + if (MusicMod::dllloaded) + { + Player_Unpause(); + } + } + } + } +} + +void +CFrame::MM_OnStop() +{ + Player_Stop(); + MusicMod::GlobalPause = false; +} +// ======================================================================================= + + +// ======================================================================================= +// Mute music +// --------------------------------------------------------------------------------------- +void +CFrame::MM_OnMute(wxCommandEvent& WXUNUSED (event)) +{ + //INFO_LOG(AUDIO,"CFrame::OnMute > Begin\n"); + //MessageBox(0, "", "", 0); + + if(!MusicMod::GlobalMute) + { + if(MusicMod::dllloaded) // avoid crash + { + Player_Mute(MusicMod::GlobalVolume); + } + + MusicMod::GlobalMute = true; + UpdateGUI(); + } + else + { + if(MusicMod::dllloaded) // avoid crash + { + Player_Mute(MusicMod::GlobalVolume); + } + MusicMod::GlobalMute = false; + UpdateGUI(); + } +} +// ======================================================================================= + + +// ======================================================================================= +// Pause music +// --------------------------------------------------------------------------------------- +void +CFrame::MM_OnPause(wxCommandEvent& WXUNUSED (event)) +{ + INFO_LOG(AUDIO,"CFrame::OnPause > Begin\n"); + //MessageBox(0, "", "", 0); + + if(!MusicMod::GlobalPause) + { + if(MusicMod::dllloaded) // avoid crash + { + Player_Pause(); + } + MusicMod::GlobalPause = true; + UpdateGUI(); + } + else + { + if(MusicMod::dllloaded) // avoid crash + { + Player_Pause(); + } + MusicMod::GlobalPause = false; + UpdateGUI(); + } +} + + + + +// ======================================================================================= +// Change volume +// --------------------------------------------------------------------------------------- +void CFrame::MM_OnVolume(wxScrollEvent& event) +{ + //INFO_LOG(AUDIO,"CFrame::OnVolume > Begin <%i>\n", event.GetPosition()); + //MessageBox(0, "", "", 0); + + //if(event.GetEventType() == wxEVT_SCROLL_PAGEUP || event.GetEventType() == wxEVT_SCROLL_PAGEDOWN) + // return; + + if(MusicMod::dllloaded) // avoid crash + { + + Player_Volume(event.GetPosition()); + MusicMod::GlobalVolume = event.GetPosition(); + + MusicMod::GlobalMute = false; // Unmute to + mm_ToolMute->Toggle(false); + + + if(event.GetEventType() == wxEVT_SCROLL_CHANGED) + { + // Only update this on release, to avoid major flickering when changing volume + UpdateGUI(); + + /* Use this to avoid that the focus get stuck on the slider when the main + window has been replaced */ + this->SetFocus();} + } +} +//======================================================================================= + + + +// ======================================================================================= +// Show log +// --------------------------------------------------------------------------------------- +void CFrame::MM_OnLog(wxCommandEvent& event) +{ + //INFO_LOG(AUDIO,"CFrame::OnLog > Begin\n"); + //MessageBox(0, "", "", 0); + + if(!MusicMod::dllloaded) return; // Avoid crash + + MusicMod::bShowConsole = !MusicMod::bShowConsole; + + if(MusicMod::bShowConsole) + /* What we do here is run StartConsoleWin() in Common directly after each + other first in the exe then in the DLL, sometimes this would give me a rampant memory + usage increase until the exe crashed at 700 MB memory usage or something like that. + For that reason I'm trying to sleep for a moment between them here. */ + { MusicMod::ShowConsole(); Sleep(100); Player_Console(true); } + else + { + #if defined (_WIN32) + Console::Close(); Player_Console(false); + #endif + } + + IniFile file; + file.Load("Plainamp.ini"); + file.Set("Interface", "ShowConsole", MusicMod::bShowConsole); + file.Save("Plainamp.ini"); + + UpdateGUI(); +} +//======================================================================================= + +#endif // MUSICMOD diff --git a/Externals/MusicMod/Main/Src/Main.cpp b/Externals/MusicMod/Main/Src/Main.cpp new file mode 100644 index 0000000000..89d95367a2 --- /dev/null +++ b/Externals/MusicMod/Main/Src/Main.cpp @@ -0,0 +1,342 @@ +// Copyright (C) 2003-2008 Dolphin Project. + +// 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, version 2.0. + +// 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 2.0 for more details. + +// A copy of the GPL 2.0 should have been included with the program. +// If not, see http://www.gnu.org/licenses/ + +// Official SVN repository and contact information can be found at +// http://code.google.com/p/dolphin-emu/ + + +////////////////////////////////////////////////////////////////////////////////////////// +// Include +// ŻŻŻŻŻŻŻŻŻŻ +#include +#include +#include +#include + +#include "Common.h" // Common +#include "IniFile.h" +#include "Log.h" + +#include "PowerPC/PowerPc.h" // Core + +#include "../../../../Source/Core/DiscIO/Src/FileSystemGCWii.h" // This file has #include "Filesystem.h" +#include "../../../../Source/Core/DiscIO/Src/VolumeCreator.h" + +#include "../../Player/Src/PlayerExport.h" // Local player +////////////////////////////////// + + + +////////////////////////////////////////////////////////////////////////////////////////// +// Declarations and definitions +// ŻŻŻŻŻŻŻŻŻŻ +namespace MusicMod +{ + +struct MyFilesStructure +{ + std::string path; + int offset; // Is int enough, how high does offset go? +}; + +std::vector MyFiles; +void StructSort (std::vector &MyFiles); + + +// Playback +std::string CurrentFile; +std::string CurrentPlayFile; +std::string CurrentPlayFilePath; +std::string unique_gameid; + +std::string MusicPath; + +DiscIO::CFileSystemGCWii* my_pFileSystem; + +int WritingFile = false; +bool dllloaded = false; + +extern bool bShowConsole; // Externally define +extern int GlobalVolume; +////////////////////////////////// + + +// ======================================================================================= +// Supported music files +// --------------------------------------------------------------------------------------- +bool CheckFileEnding(std::string FileName) +{ + if ( + (FileName.find(".adp") != std::string::npos) // 1080 Avalanche, Crash Bandicoot etc + || (FileName.find(".afc") != std::string::npos) // Zelda WW + || (FileName.find(".ast") != std::string::npos) // Zelda TP, Mario Kart + || (FileName.find(".dsp") != std::string::npos) // Metroid Prime + || (FileName.find(".hps") != std::string::npos) // SSB Melee + ) + return true; + + return false; +} +// ======================================================================================= + + +// ======================================================================================= +// A function to sort the filelist table after offset +// ------------------------ +void StructSort (std::vector &MyFiles) +{ + MyFilesStructure temp; + + //INFO_LOG(AUDIO,"StructSort > Begin\n"); + + for(int i = 0; i < MyFiles.size() - 1; i++) + { + for (int j = i + 1; j < MyFiles.size(); j++) + { + if (MyFiles[ i ].offset > MyFiles[ j ].offset) //comparing cost + { + temp = MyFiles[ i ]; // Swapping entire struct + MyFiles[ i ] = MyFiles[ j ]; + MyFiles[ j ] = temp; + } + } + } + + + for (long i=1; i<(long)MyFiles.size(); ++i) + { + std::cout << i << " " << MyFiles[i].path.c_str() << "#" << MyFiles[i].offset << "\n"; + } + + //INFO_LOG(AUDIO,"StructSort > Done\n"); +} +// ============================ + + +// ======================================================================================= +/* Run these things once */ +// ------------------------ +void ShowConsole() +{ +// Console::Open(100, 2000, "MusicMod", true); // Give room for 2000 rows +} + +void Init() +{ + // These things below will not need to be updated after a new game is started + if (dllloaded) return; + + // --------------------------------------- + // Load config + // --------------------- + IniFile file; + file.Load("Plainamp.ini"); + file.Get("Interface", "ShowConsole", &MusicMod::bShowConsole, false); + file.Get("Plainamp", "Volume", &MusicMod::GlobalVolume, 125); + // ------- + + // --------------------------------------- + // Make a debugging window + // --------------------- + if(MusicMod::bShowConsole) ShowConsole(); + + // Write version + #ifdef _M_X64 + INFO_LOG(AUDIO,"64 bit version\n"); + #else + INFO_LOG(AUDIO,"32 bit version\n"); + #endif + // ----------- + + // Set volume + Player_Volume(MusicMod::GlobalVolume); + + // Show DLL status + Player_Main(MusicMod::bShowConsole); + //play_file("c:\\demo36_02.ast"); + //INFO_LOG(AUDIO,"DLL loaded\n"); + + dllloaded = true; // Do this once +} +// ============================ + + +// ======================================================================================= +/* This will read the GC file system. */ +// ------------------------ +void Main(std::string FileName) +{ + // + DiscIO::IVolume* pVolume = DiscIO::CreateVolumeFromFilename(FileName.c_str()); + + // + my_pFileSystem = new DiscIO::CFileSystemGCWii(pVolume); + + /* We have to sort the files according to offset so that out scan in Blob.cpp works. + Because StructSort() only works for MyFilesStructure I copy the offset and filenames + to a new vetor first. */ + MyFiles.resize(my_pFileSystem->m_FileInfoVector.size()); // Set size + for (size_t i = 0; i < my_pFileSystem->m_FileInfoVector.size(); i++) + { + MyFiles.at(i).offset = my_pFileSystem->m_FileInfoVector.at(i).m_Offset; + MyFiles.at(i).path = my_pFileSystem->m_FileInfoVector.at(i).m_FullPath; + } + + // Sort the files by offset + StructSort(MyFiles); + + // --------------------------------------------------------------------------------------- + // Make Music directory + // ------------------------- + LPSECURITY_ATTRIBUTES attr; + attr = NULL; + MusicPath = "Music\\"; + INFO_LOG(AUDIO,"Created a Music directory\n"); + CreateDirectory(MusicPath.c_str(), attr); + // ---------------------------------------------------------------------------------------- +} + + +// ======================================================================================= +// Check if we should play this file +// --------------------------------------------------------------------------------------- +void CheckFile(std::string File, int FileNumber) +{ + // Do nothing if we found the same file again + if (CurrentFile == File) return; + + //INFO_LOG(AUDIO,">>>> (%i)Current read %s <%u = %ux%i> \n", i, CurrentFiles[i].path.c_str(), offset, CurrentFiles[i].offset, size); + + // Check if it's a music file + if (CheckFileEnding(File.c_str())) + { + /* Don't restart the playback if we find the same music file again. If the game is playing + a streaming music file it may read from it once in a while, after it has read other + files in between, if did not do this check we would restart the playback in those cases */ + if (CurrentPlayFile == File) return; + + // Notify the user + INFO_LOG(AUDIO,"\n >>> (%i/%i) Match %s\n\n", FileNumber, MyFiles.size(), File.c_str()); + + // Save the matched file + CurrentPlayFile = File; + + // --------------------------------------------------------------------------------------- + // We will now save the file to the PC hard drive + // ------------------ + // Get the filename + std::size_t pointer = File.find_last_of("\\"); + std::string fragment = File.substr (0, (pointer-0)); + int compare = File.length() - fragment.length(); // Get the length of the filename + fragment = File.substr ((pointer+1), compare); // Now we have the filename + + // Create the target file path + std::string FilePath = (MusicPath + fragment); + + WritingFile = true; // Avoid detecting the file we are writing + INFO_LOG(AUDIO,"Writing <%s> to <%s>\n", File.c_str(), FilePath.c_str()); + my_pFileSystem->ExportFile(File.c_str(), FilePath.c_str()); + WritingFile = false; + + // --------------------------------------------------------------------------------------- + // Play the file we found + // ------------------ + if(dllloaded) + { + Player_Play((char*)FilePath.c_str()); // retype it from const char* to char* + } else { + INFO_LOG(AUDIO,"Warning > Music DLL is not loaded"); + } + + // --------------------------------------------------------------------------------------- + // Remove the last file, if any + // ------------------ + if(CurrentPlayFilePath.length() > 0) + { + if(!remove(CurrentPlayFilePath.c_str())) + { + INFO_LOG(AUDIO,"The program failed to remove <%s>\n", CurrentPlayFilePath.c_str()); + } else { + INFO_LOG(AUDIO,"The program removed <%s>\n", CurrentPlayFilePath.c_str()); + } + } + + // --------------------------------------------------------------------------------------- + // Save the current playing file + // ------------------ + CurrentPlayFilePath = FilePath; // Save the filename so we can remove it later + } + + // Tell the user about the files we ignored + INFO_LOG(AUDIO,"(%i/%i) Ignored %s\n", FileNumber, MyFiles.size(), File.c_str()); + + // Update the current file + CurrentFile = File; +} + + +////////////////////////////////////////////////////////////////////////////////////////// +// Find the current filename for a certain offset on the GC fileystem +// ŻŻŻŻŻŻŻŻŻŻŻŻŻ +void FindFilename(u64 offset, u64 size) +{ + // ======================================================================================= + /* Only do this test: + 1. After boot, not on ISO scan + 2. Not if offset = 0. + 3. Not when WritingFile. We will lead to calls back to here (Read) and it will mess + upp the scanning */ + if(PowerPC::GetState() == PowerPC::CPUState::CPU_RUNNING && offset != 0 && !WritingFile) + { + + + ////////////////////////////////////////////////////////////////////////////////////////// + /* Get the filename. Here we go through all files until we come to the file that has + the matching offset. Before MyFiles has data this loop will go nowhere. We have to + specify (MyFiles.size() - 1) because we will be reading MyFiles.at(i + 1). + MyFiles.size() is the amount of files on the disc, and the last file is + MyFiles.at(MyFiles.size() - 1) */ + // --------------------------------------------------------------------------------------- + for (int i = 0; i < (int)(MyFiles.size() - 1); ++i) + { + // --------------------------------------------------------------------------------------- + /* If we assume that myoffset is the begginning of every file this works. + Suppose there are three files + 1 is 0 to 149 + 2 is 150 to 170 + 3 is 171 to 200 + If the offset is 160 the game is reading file number two, for example + myoffset = 150: (myoffset >= offset) == false + myoffset = 171: (myoffset >= offset) == true, break + + However if the offset is 195 the game is reading the last file and we will get + myoffset = 171: (myoffset >= offset) == false + we therefore need to add the condition (offset > MyFiles[MyFiles.size() - 1].offset) + to. */ + if (MyFiles[i + 1].offset >= offset || offset > MyFiles[MyFiles.size() - 1].offset) + { + // Now we know that the game is reading from MyFiles[i].path + CheckFile(MyFiles[i].path, i); + + // Stop checking + break; + } + } + } // This ends the entire filescan + // ======================================================================================= +} +///////////////////////////////// + + +} // end of namespace diff --git a/Externals/MusicMod/Main/Src/Main.h b/Externals/MusicMod/Main/Src/Main.h new file mode 100644 index 0000000000..92ea473507 --- /dev/null +++ b/Externals/MusicMod/Main/Src/Main.h @@ -0,0 +1,31 @@ +// Copyright (C) 2003-2008 Dolphin Project. + +// 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, version 2.0. + +// 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 2.0 for more details. + +// A copy of the GPL 2.0 should have been included with the program. +// If not, see http://www.gnu.org/licenses/ + +// Official SVN repository and contact information can be found at +// http://code.google.com/p/dolphin-emu/ + + +#include // System: For std + +#include "Common.h" // Common: For u64 + + +namespace MusicMod +{ + +void Main(std::string FileName); +void FindFilename(u64 offset, u64 size); +void CheckFile(std::string File, int FileNumber = 0); + +} \ No newline at end of file diff --git a/Externals/MusicMod/Player/Lib/fftw3.lib b/Externals/MusicMod/Player/Lib/fftw3.lib new file mode 100644 index 0000000000000000000000000000000000000000..0fe0789d9a1bc375cffca840cdb5e769af047f92 GIT binary patch literal 31296 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 Rhi~o&v;`Y#8|I`Z7XX%#*hv5Y literal 0 HcmV?d00001 diff --git a/Externals/MusicMod/Player/Lib/libzlib1.lib b/Externals/MusicMod/Player/Lib/libzlib1.lib new file mode 100644 index 0000000000000000000000000000000000000000..e58100e82252c12b8ec2bbd8ba70c84b7501e4a4 GIT binary patch literal 37266 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 literal 0 HcmV?d00001 diff --git a/Externals/MusicMod/Player/Player.vcproj b/Externals/MusicMod/Player/Player.vcproj new file mode 100644 index 0000000000..4b7df6c15f --- /dev/null +++ b/Externals/MusicMod/Player/Player.vcproj @@ -0,0 +1,925 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Externals/MusicMod/Player/Src/AddDirectory.cpp b/Externals/MusicMod/Player/Src/AddDirectory.cpp new file mode 100644 index 0000000000..17dd6de1d8 --- /dev/null +++ b/Externals/MusicMod/Player/Src/AddDirectory.cpp @@ -0,0 +1,364 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "AddDirectory.h" +#include "Playlist.h" +#include "Main.h" +#include "InputPlugin.h" +#include +#include +#include +#include + +using namespace std; + + + +HWND WindowBrowse = NULL; + + + +void SearchFolder( TCHAR * szPath ); + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +LPITEMIDLIST GetCurrentFolder() +{ +/* + How To Convert a File Path to an ITEMIDLIST + http://support.microsoft.com/default.aspx?scid=kb;en-us;132750 +*/ + LPITEMIDLIST pidl; + LPSHELLFOLDER pDesktopFolder; + TCHAR szPath[ MAX_PATH ]; +#ifndef PA_UNICODE + OLECHAR olePath[ MAX_PATH ]; +#endif + ULONG chEaten; + ULONG dwAttributes; + HRESULT hr; + + // + // Get the path we need to convert. + // + GetCurrentDirectory( MAX_PATH, szPath ); + + // + // Get a pointer to the Desktop's IShellFolder interface. + // + if( SUCCEEDED( SHGetDesktopFolder( &pDesktopFolder ) ) ) + { + + // + // IShellFolder::ParseDisplayName requires the file name be in + // Unicode. + // +#ifndef PA_UNICODE + MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, szPath, -1, olePath, MAX_PATH ); +#endif + // + // Convert the path to an ITEMIDLIST. + // + // hr = pDesktopFolder->lpVtbl->ParseDisplayName( + hr = pDesktopFolder->ParseDisplayName( + ( HWND__ * )pDesktopFolder, + NULL, +#ifndef PA_UNICODE + olePath, +#else + szPath, +#endif + &chEaten, + &pidl, + &dwAttributes + ); + + if( FAILED( hr ) ) + { + // Handle error. + return NULL; + } + + // + // pidl now contains a pointer to an ITEMIDLIST for .\readme.txt. + // This ITEMIDLIST needs to be freed using the IMalloc allocator + // returned from SHGetMalloc(). + // + + // release the desktop folder object + // pDesktopFolder->lpVtbl->Release(); + pDesktopFolder->Release(); + + return pidl; + } + else + { + return NULL; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +BOOL CALLBACK EnumChildProc( HWND hwnd, LPARAM lp ) +{ + TCHAR szClassName[ 8 ] = TEXT( "\0" ); + HWND * hFirstFoundStatic = ( ( HWND * )lp ); + + if( GetClassName( hwnd, szClassName, 7 ) ) + { + if( !_tcscmp( szClassName, TEXT( "Static" ) ) ) + { + if( *hFirstFoundStatic ) + { + // Both found + RECT r1; + GetWindowRect( *hFirstFoundStatic, &r1 ); + + RECT r2; + GetWindowRect( hwnd, &r2 ); + + // First must be taller one + if( r1.bottom - r1.top < r2.bottom - r2.top ) + { + // Swap + RECT r = r1; + HWND h = *hFirstFoundStatic; + + r1 = r2; + *hFirstFoundStatic = hwnd; + + r2 = r; + hwnd = h; + } + + POINT xy2 = { r2.left, r2.top }; + ScreenToClient( WindowBrowse, &xy2 ); + + SetWindowPos( + *hFirstFoundStatic, + NULL, + 0, + 0, + r2.right - r2.left, + r2.bottom - r2.top, + SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOZORDER + ); + + SetWindowPos( + hwnd, + NULL, + xy2.x, + xy2.y + ( r2.bottom - r2.top ) - ( r1.bottom - r1.top ), + r1.right - r1.left, + r1.bottom - r1.top, + SWP_NOOWNERZORDER | SWP_NOZORDER + ); + + + return FALSE; // Stop + } + else + { + // First found + *hFirstFoundStatic = hwnd; + } + } + } + + return TRUE; // Continue +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +int CALLBACK BrowseCallbackProc( HWND hwnd, UINT message, LPARAM lp, LPARAM wp ) +{ + switch( message ) + { + case BFFM_INITIALIZED: + { + WindowBrowse = hwnd; + + // Init with curdir + SendMessage( hwnd, BFFM_SETSELECTION, FALSE, ( LPARAM )GetCurrentFolder() ); + + // Swap static dimensions + HWND hFirstFoundStatic = NULL; + EnumChildWindows( hwnd, EnumChildProc, ( LPARAM )&hFirstFoundStatic ); + + break; + } + + case BFFM_SELCHANGED: + { + TCHAR szPath[ MAX_PATH ] = TEXT( "\0" ); + SHGetPathFromIDList( ( LPITEMIDLIST )lp, szPath ); + SendMessage( hwnd, BFFM_SETSTATUSTEXT, 0, ( LPARAM )szPath ); + + break; + } + + case BFFM_VALIDATEFAILED: + return TRUE; + + } + + return 0; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Shows a Browse-For-Folder dialog and recursively adds supported files +/// to the playlist. Files are sorted by full filaname before being added. +//////////////////////////////////////////////////////////////////////////////// +void AddDirectory() +{ + TCHAR szPath[ MAX_PATH ]; + BROWSEINFO bi = { 0 }; + bi.hwndOwner = WindowMain; + bi.pidlRoot = NULL; // Desktop folder + bi.lpszTitle = TEXT( "Please select a directory:" ); + bi.ulFlags = BIF_VALIDATE | BIF_STATUSTEXT; + bi.lpfn = BrowseCallbackProc; + + LPITEMIDLIST pidl = SHBrowseForFolder( &bi ); + if( !pidl ) return; + + // Get path + SHGetPathFromIDList( pidl, szPath ); + + + // Search + SearchFolder( szPath ); + + + // Stay here + SetCurrentDirectory( szPath ); + + + // Free memory used + IMalloc * imalloc = 0; + if( SUCCEEDED( SHGetMalloc( &imalloc ) ) ) + { + imalloc->Free( pidl ); + imalloc->Release(); + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/* Warning: There is SetCurrentDirectory() here, be aware of it. We don't really + want to use that. */ +//////////////////////////////////////////////////////////////////////////////// +void SearchFolder( TCHAR * szPath ) +{ + // Remove trailing backslash + int iPathLen = ( int )_tcslen( szPath ); + if( iPathLen < 1 ) return; + if( szPath[ iPathLen - 1 ] == TEXT( '\\' ) ) + { + iPathLen--; + } + + // Init working buffer + TCHAR szFullpath[ MAX_PATH ]; + memcpy( szFullpath, szPath, iPathLen * sizeof( TCHAR ) ); + szFullpath[ iPathLen ] = TEXT( '\\' ); + szFullpath[ iPathLen + 1 ] = TEXT( '\0' ); + + // Make pattern + _tcscpy( szFullpath + iPathLen + 1, TEXT( "*" ) ); + + // Find + vector Files; + vector Dirs; + WIN32_FIND_DATA FindFileData; + HANDLE hFind; + hFind = FindFirstFile( szFullpath, &FindFileData ); + if( hFind == INVALID_HANDLE_VALUE ) return; + + do + { + // Skip "." and ".." + if( !_tcscmp( FindFileData.cFileName, TEXT( "." ) ) || + !_tcscmp( FindFileData.cFileName, TEXT( ".." ) ) ) continue; + + // Make full path + _tcscpy( szFullpath + iPathLen + 1, FindFileData.cFileName ); + + // Is directory? + TCHAR * szPartname = new TCHAR[ MAX_PATH ]; + _tcscpy( szPartname, FindFileData.cFileName ); + if( SetCurrentDirectory( szFullpath ) ) + { + // New dir + Dirs.push_back( szPartname ); + continue; + } + + // Search "." + const int iFilenameLen = ( int )_tcslen( FindFileData.cFileName ); + TCHAR * szExt = FindFileData.cFileName + iFilenameLen - 1; + while( ( szExt > FindFileData.cFileName ) && ( *szExt != TEXT( '.' ) ) ) szExt--; + if( *szExt != TEXT( '.' ) ) continue; + szExt++; + + // Check extension + map ::iterator iter = ext_map.find( szExt ); + if( iter == ext_map.end() ) continue; + + // New file + Files.push_back( szPartname ); + } + while( FindNextFile( hFind, &FindFileData ) ); + + FindClose( hFind ); + vector ::iterator iter; + + // Sort and recurse directories + sort( Dirs.begin(), Dirs.end(), TextCompare() ); + iter = Dirs.begin(); + while( iter != Dirs.end() ) + { + TCHAR * szWalk = *iter; + + _tcscpy( szFullpath + iPathLen + 1, szWalk ); + SearchFolder( szFullpath ); + + iter++; + } + + // Sort and add files + sort( Files.begin(), Files.end(), TextCompare() ); + iter = Files.begin(); + while( iter != Files.end() ) + { + TCHAR * szWalk = *iter; + + TCHAR * szKeep = new TCHAR[ MAX_PATH ]; + memcpy( szKeep, szFullpath, ( iPathLen + 1 ) * sizeof( TCHAR ) ); + _tcscpy( szKeep + iPathLen + 1, szWalk ); + playlist->PushBack( szKeep ); + + iter++; + } +} diff --git a/Externals/MusicMod/Player/Src/AddDirectory.h b/Externals/MusicMod/Player/Src/AddDirectory.h new file mode 100644 index 0000000000..8c819cbd63 --- /dev/null +++ b/Externals/MusicMod/Player/Src/AddDirectory.h @@ -0,0 +1,26 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_ADD_DIRECTORY_H +#define PA_ADD_DIRECTORY_H + + + +#include "Global.h" + + + +void AddDirectory(); + + + +#endif // PA_ADD_DIRECTORY_H diff --git a/Externals/MusicMod/Player/Src/AddFiles.cpp b/Externals/MusicMod/Player/Src/AddFiles.cpp new file mode 100644 index 0000000000..d8367894b2 --- /dev/null +++ b/Externals/MusicMod/Player/Src/AddFiles.cpp @@ -0,0 +1,181 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "AddFiles.h" +#include "InputPlugin.h" +#include "Main.h" +#include "Playlist.h" +#include + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +void AddFiles() +{ + int total = 0; + + int iFilterLen = 0; + + InputPlugin * input; + vector ::iterator iter; + + // Get len +// if( input_plugins.empty() ) return; + iter = input_plugins.begin(); + while( iter != input_plugins.end() ) + { + input = *iter; + if( !input ) iter++; + iFilterLen += input->iFiltersLen; + iter++; + } + +// if( !iFilterLen ) return; + + iFilterLen += 40 + 29 + ( int )ext_map.size() * ( 2 + 4 + 1 ) + 7; + + TCHAR * szFilters = new TCHAR[ iFilterLen ]; + TCHAR * walk = szFilters; + + // ..................1.........1....\....\.1.........1........\.1... + memcpy( walk, TEXT( "All files (*.*)\0*.*\0All supported types\0" ), 40 * sizeof( TCHAR ) ); + walk += 40; + + // Add all extensions as ";*.ext" +// if( ext_map.empty() ) return; + map ::iterator iter_ext = ext_map.begin(); + bool bFirst = true; + while( iter_ext != ext_map.end() ) + { + if( !bFirst ) + { + memcpy( walk, TEXT( ";*." ), 3 * sizeof( TCHAR ) ); + walk += 3; + } + else + { + memcpy( walk, TEXT( "*." ), 2 * sizeof( TCHAR ) ); + walk += 2; + bFirst = false; + } + + TCHAR * szExt = iter_ext->first; + int uLen = ( int )_tcslen( szExt ); + memcpy( walk, szExt, uLen * sizeof( TCHAR ) ); + walk += uLen; + + iter_ext++; + } + + // *walk = TEXT( '\0' ); + // walk++; + + // ..................1..........1... + memcpy( walk, TEXT( ";*.m3u\0" ), 7 * sizeof( TCHAR ) ); + walk += 7; + + // ..................1.........1.........1...........1... + memcpy( walk, TEXT( "Playlist files (*.M3U)\0*.m3u\0" ), 29 * sizeof( TCHAR ) ); + walk += 29; + + // Copy filters + iter = input_plugins.begin(); + while( iter != input_plugins.end() ) + { + input = *iter; + if( !input ) iter++; + memcpy( walk, input->szFilters, input->iFiltersLen * sizeof( TCHAR ) ); + walk += input->iFiltersLen; + iter++; + } + + *walk = TEXT( '\0' ); + walk++; + + +//////////////////////////////////////////////////////////////////////////////// + + static TCHAR szFilenames[ 20001 ]; + *szFilenames = TEXT( '\0' ); // Each time! + + OPENFILENAME ofn; + memset( &ofn, 0, sizeof( OPENFILENAME ) ); + ofn.lStructSize = sizeof( OPENFILENAME ); + ofn.hwndOwner = WindowMain; + ofn.hInstance = g_hInstance; + ofn.lpstrFilter = szFilters; // "MPEG Layer 3\0*.mp3\0"; + ofn.lpstrCustomFilter = NULL; + ofn.nMaxCustFilter = 0; + ofn.nFilterIndex = 2; + ofn.lpstrFile = szFilenames; + ofn.nMaxFile = 20000; + ofn.Flags = OFN_EXPLORER | OFN_ALLOWMULTISELECT | OFN_ENABLESIZING | OFN_FILEMUSTEXIST | OFN_HIDEREADONLY; + ofn.nMaxFileTitle = 0, // NULL; + ofn.lpstrInitialDir = NULL; + ofn.lpstrTitle = TEXT( "Add files" ); + + if( !GetOpenFileName( &ofn ) ) return; + + int uDirLen = ( int )_tcslen( szFilenames ); + TCHAR * szDir = szFilenames; + + TCHAR * szFileWalk = szDir + uDirLen + 1; + if( *szFileWalk == TEXT( '\0' ) ) // "\0\0" or just "\0"? + { + // \0\0 -> Single file + if( !_tcsncmp( szDir + uDirLen - 3, TEXT( "m3u" ), 3 ) ) + { + // Playlist file + Playlist::AppendPlaylistFile( szDir ); + } + else + { + // Music file + TCHAR * szKeep = new TCHAR[ uDirLen + 1 ]; + memcpy( szKeep, szDir, uDirLen * sizeof( TCHAR ) ); + szKeep[ uDirLen ] = TEXT( '\0' ); + playlist->PushBack( szKeep ); + } + } + else + { + // \0 -> Several files + int iFileLen; + while( *szFileWalk != TEXT( '\0' ) ) + { + iFileLen = ( int )_tcslen( szFileWalk ); + if( !iFileLen ) return; + + TCHAR * szKeep = new TCHAR[ uDirLen + 1 + iFileLen + 1 ]; + memcpy( szKeep, szDir, uDirLen * sizeof( TCHAR ) ); + szKeep[ uDirLen ] = TEXT( '\\' ); + memcpy( szKeep + uDirLen + 1, szFileWalk, iFileLen * sizeof( TCHAR ) ); + szKeep[ uDirLen + 1 + iFileLen ] = TEXT( '\0' ); + + if( !_tcsncmp( szKeep + uDirLen + 1 + iFileLen - 3, TEXT( "m3u" ), 3 ) ) + { + // Playlist file + Playlist::AppendPlaylistFile( szKeep ); + delete [] szKeep; + } + else + { + // Music file + playlist->PushBack( szKeep ); + } + + szFileWalk += iFileLen + 1; + } + } +} diff --git a/Externals/MusicMod/Player/Src/AddFiles.h b/Externals/MusicMod/Player/Src/AddFiles.h new file mode 100644 index 0000000000..387242ce1a --- /dev/null +++ b/Externals/MusicMod/Player/Src/AddFiles.h @@ -0,0 +1,26 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_ADD_FILES_H +#define PA_ADD_FILES_H + + + +#include "Global.h" + + + +void AddFiles(); + + + +#endif // PA_ADD_FILES_H diff --git a/Externals/MusicMod/Player/Src/Config.cpp b/Externals/MusicMod/Player/Src/Config.cpp new file mode 100644 index 0000000000..d6b3615bf9 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Config.cpp @@ -0,0 +1,687 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + + +//////////////////////////////////////////////////////////////////////////////// +/// Include and declarations, definitions +//////////////////////////////////////////////////////////////////////////////// +#include "Config.h" +#include "Console.h" +#include + +using namespace std; + +map * conf_map = NULL; + +TCHAR * szIniPath = NULL; +const TCHAR * SECTION = TEXT( "Plainamp" ); +// ============================================================================ + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfVar::ConfVar( TCHAR * szKey, ConfMode mode ) +{ + // MessageBox( 0, TEXT( "no const @ ConfVar" ), TEXT( "" ), 0 ); + //INFO_LOG(AUDIO,"ConfVar::ConfVar(TCHAR) > Got <%s>\n", szKey); + + // Init + const int iLen = ( int )_tcslen( szKey ); + m_szKey = new TCHAR[ iLen + 1 ]; + memcpy( m_szKey, szKey, iLen * sizeof( TCHAR ) ); + m_szKey[ iLen ] = TEXT( '\0' ); + + m_bCopyKey = true; + m_Mode = mode; + + m_bRead = false; + + // Register + if( !conf_map ) conf_map = new map; + conf_map->insert( pair( m_szKey, this ) ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfVar::ConfVar( const TCHAR * szKey, ConfMode mode ) +{ + //INFO_LOG(AUDIO,"ConfVar::ConfVar(const TCHAR) > Got <%s>\n", szKey); + + // Init + m_szKey = ( TCHAR * )szKey; + + m_bCopyKey = false; + m_Mode = mode; + + m_bRead = false; + + // Register + if( !conf_map ) conf_map = new map; + conf_map->insert( pair( m_szKey, this ) ); + + //INFO_LOG(AUDIO,"ConfVar::ConfVar(const TCHAR) > Insert <%s>\n", ConfVar::m_szKey); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfVar::~ConfVar() +{ + if( m_bCopyKey && m_szKey ) delete [] m_szKey; +} + + + +// ======================================================================================= +// The line name is collected in ConfVar, then ConfBool gets the boolean +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfBool::ConfBool( bool * pbData, TCHAR * szKey, ConfMode mode, bool bDefault ) : ConfVar( szKey, mode ) +{ + // MessageBox( 0, TEXT( "no const @ ConfBool" ), TEXT( "" ), 0 ); + //INFO_LOG(AUDIO,"ConfBool(TCHAR) > Get <%s>\n", szKey); + + m_pbData = pbData; + m_bDefault = bDefault; + + // *pbData = bDefault; +} +// ======================================================================================= + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfBool::ConfBool( bool * pbData, const TCHAR * szKey, ConfMode mode, bool bDefault ) : ConfVar( szKey, mode ) +{ + //INFO_LOG(AUDIO,"ConfBool(TCHAR) > Get <%s>\n", szKey); + + m_pbData = pbData; + m_bDefault = bDefault; + + // *pbData = bDefault; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfBool::Read() +{ + //INFO_LOG(AUDIO,"ConfBool::Read() > Begin and \n", m_bRead, szIniPath); + + if( m_bRead || !szIniPath ) return; + + *m_pbData = ( GetPrivateProfileInt( SECTION, m_szKey, ( int )m_bDefault, szIniPath ) != 0 ); + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfBool::Write() +{ + WritePrivateProfileString( SECTION, m_szKey, *m_pbData ? TEXT( "1" ) : TEXT( "0" ), szIniPath ); + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfInt::ConfInt( int * piData, TCHAR * szKey, ConfMode mode, int iDefault ) : ConfVar( szKey, mode ) +{ + MessageBox( 0, TEXT( "no const" ), TEXT( "" ), 0 ); + + m_piData = piData; + m_iDefault = iDefault; + + // *piData = iDefault; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfInt::ConfInt( int * piData, const TCHAR * szKey, ConfMode mode, int iDefault ) : ConfVar( szKey, mode ) +{ + m_piData = piData; + m_iDefault = iDefault; + + // *piData = iDefault; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfInt::Read() +{ + if( m_bRead || !szIniPath ) return; + + *m_piData = GetPrivateProfileInt( SECTION, m_szKey, m_iDefault, szIniPath ); + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfInt::Write() +{ + TCHAR szNumber[ 12 ] = TEXT( "" ); + _stprintf( szNumber, TEXT( "%i" ), *m_piData ); + WritePrivateProfileString( SECTION, m_szKey, szNumber, szIniPath ); + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfIntMinMax::ConfIntMinMax( int * piData, TCHAR * szKey, ConfMode mode, int iDefault, int iMin, int iMax ) : ConfInt( piData, szKey, mode, iDefault ) +{ + MessageBox( 0, TEXT( "no const" ), TEXT( "" ), 0 ); + + m_iMin = iMin; + m_iMax = iMax; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfIntMinMax::ConfIntMinMax( int * piData, const TCHAR * szKey, ConfMode mode, int iDefault, int iMin, int iMax ) : ConfInt( piData, szKey, mode, iDefault ) +{ + m_iMin = iMin; + m_iMax = iMax; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfWinPlaceCallback::ConfWinPlaceCallback( WINDOWPLACEMENT * pwpData, TCHAR * szKey, RECT * prDefault, ConfCallback fpCallback ) : ConfVar( szKey, CONF_MODE_INTERNAL ) +{ + MessageBox( 0, TEXT( "no const" ), TEXT( "" ), 0 ); + + m_pwpData = pwpData; + m_prDefault = prDefault; + m_fpCallback = fpCallback; + + pwpData->length = sizeof( WINDOWPLACEMENT ); + pwpData->flags = 0; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfWinPlaceCallback::ConfWinPlaceCallback( WINDOWPLACEMENT * pwpData, const TCHAR * szKey, RECT * prDefault, ConfCallback fpCallback ) : ConfVar( szKey, CONF_MODE_INTERNAL ) +{ + m_pwpData = pwpData; + m_prDefault = prDefault; + m_fpCallback = fpCallback; + + pwpData->length = sizeof( WINDOWPLACEMENT ); + pwpData->flags = 0; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfWinPlaceCallback::Read() +{ + if( m_bRead || !szIniPath ) return; + + POINT ptMinPosition = { -1, -1 }; + POINT ptMaxPosition = { -GetSystemMetrics( SM_CXBORDER ), -GetSystemMetrics( SM_CYBORDER ) }; + + m_pwpData->length = sizeof( WINDOWPLACEMENT ); + m_pwpData->flags = 0; + m_pwpData->ptMinPosition = ptMinPosition; + m_pwpData->ptMaxPosition = ptMaxPosition; + + + TCHAR szOut[ 111 ] = TEXT( "" ); + int iChars = GetPrivateProfileString( SECTION, m_szKey, TEXT( "" ), szOut, 110, szIniPath ); + bool bApplyDefault = true; + if( iChars ) + { + int iFields = _stscanf( + szOut, + TEXT( "(%u,(%i,%i,%i,%i))" ), + &m_pwpData->showCmd, + &m_pwpData->rcNormalPosition.left, + &m_pwpData->rcNormalPosition.top, + &m_pwpData->rcNormalPosition.right, + &m_pwpData->rcNormalPosition.bottom + ); + if( iFields == 5 ) + { + bApplyDefault = false; + } + } + + if( bApplyDefault ) + { + m_pwpData->showCmd = SW_SHOWNORMAL; + m_pwpData->rcNormalPosition = *m_prDefault; + } + + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfWinPlaceCallback::Write() +{ + // Refresh data + if( m_fpCallback ) m_fpCallback( this ); + + TCHAR szData[ 111 ] = TEXT( "" ); + _stprintf( + szData, + TEXT( "(%u,(%i,%i,%i,%i))" ), + m_pwpData->showCmd, + m_pwpData->rcNormalPosition.left, + m_pwpData->rcNormalPosition.top, + m_pwpData->rcNormalPosition.right, + m_pwpData->rcNormalPosition.bottom + ); + WritePrivateProfileString( SECTION, m_szKey, szData, szIniPath ); + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfBandInfoCallback::ConfBandInfoCallback( BandInfo * pbiData, TCHAR * szKey, BandInfo * pbiDefault, ConfCallback fpCallback ) : ConfVar( szKey, CONF_MODE_INTERNAL ) +{ + MessageBox( 0, TEXT( "no const" ), TEXT( "" ), 0 ); + + m_pbiData = pbiData; + m_pbiDefault = pbiDefault; + m_fpCallback = fpCallback; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfBandInfoCallback::ConfBandInfoCallback( BandInfo * pbiData, const TCHAR * szKey, BandInfo * pbiDefault, ConfCallback fpCallback ) : ConfVar( szKey, CONF_MODE_INTERNAL ) +{ + m_pbiData = pbiData; + m_pbiDefault = pbiDefault; + m_fpCallback = fpCallback; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfBandInfoCallback::Read() +{ + if( m_bRead || !szIniPath ) return; + + int iBreak; + int iVisible; + + TCHAR szOut[ 91 ] = TEXT( "" ); + int iChars = GetPrivateProfileString( SECTION, m_szKey, TEXT( "" ), szOut, 90, szIniPath ); + bool bApplyDefault = true; + if( iChars ) + { + int iFields = _stscanf( + szOut, + TEXT( "(%i,%i,%i,%i)" ), + &m_pbiData->m_iIndex, + &m_pbiData->m_iWidth, + &iBreak, + &iVisible + ); + if( iFields == 4 ) + { + m_pbiData->m_bBreak = ( iBreak != 0 ); + m_pbiData->m_bVisible = ( iVisible != 0 ); + + bApplyDefault = false; + } + } + + if( bApplyDefault ) + { + *m_pbiData = *m_pbiDefault; + } + + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfBandInfoCallback::Write() +{ + // Refresh data + if( m_fpCallback ) m_fpCallback( this ); + + TCHAR szData[ 91 ] = TEXT( "" ); + _stprintf( + szData, + TEXT( "(%i,%i,%i,%i)" ), + m_pbiData->m_iIndex, + m_pbiData->m_iWidth, + ( int )m_pbiData->m_bBreak, + ( int )m_pbiData->m_bVisible + ); + WritePrivateProfileString( SECTION, m_szKey, szData, szIniPath ); + m_bRead = true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool ConfBandInfoCallback::Apply( HWND hRebar, int iBandId ) +{ + const int iOldIndex = ( int )SendMessage( hRebar, RB_IDTOINDEX, iBandId, 0 ); + if( iOldIndex == -1 ) return false; + + int & iNewIndex = m_pbiData->m_iIndex; + if( iOldIndex != iNewIndex ) + { + // Move band + if( !SendMessage( hRebar, RB_MOVEBAND, iOldIndex, iNewIndex ) ) + { + return false; + } + } + + REBARBANDINFO rbbi; + rbbi.cbSize = sizeof( REBARBANDINFO ); + rbbi.fMask = RBBIM_STYLE; + + // Get current info + if( !SendMessage( hRebar, RB_GETBANDINFO, iNewIndex, ( LPARAM )&rbbi ) ) + { + return false; + } + + rbbi.fMask = RBBIM_SIZE | RBBIM_STYLE; + rbbi.cx = m_pbiData->m_iWidth; + if( ( rbbi.fStyle & RBBS_BREAK ) == RBBS_BREAK ) + { + if( !m_pbiData->m_bBreak ) rbbi.fStyle -= RBBS_BREAK; + } + else + { + if( m_pbiData->m_bBreak ) rbbi.fStyle |= RBBS_BREAK; + } + + // Update info + if( !SendMessage( hRebar, RB_SETBANDINFO, iNewIndex, ( LPARAM )&rbbi ) ) + { + return false; + } + + // Show/hide band + if( !SendMessage( hRebar, RB_SHOWBAND, iNewIndex, m_pbiData->m_bVisible ? TRUE : FALSE ) ) + { + return false; + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfString::ConfString( TCHAR * szData, TCHAR * szKey, ConfMode mode, TCHAR * szDefault, int iMaxLen ) : ConfVar( szKey, mode ) +{ + MessageBox( 0, TEXT( "no const" ), TEXT( "" ), 0 ); + + m_szData = szData; + m_szDefault = szDefault; + m_iMaxLen = iMaxLen; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfString::ConfString( TCHAR * szData, const TCHAR * szKey, ConfMode mode, TCHAR * szDefault, int iMaxLen ) : ConfVar( szKey, mode ) +{ + m_szData = szData; + m_szDefault = szDefault; + m_iMaxLen = iMaxLen; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfString::Read() +{ + //INFO_LOG(AUDIO, "ConfString::Read() > Begin\n"); + + if( m_bRead || !szIniPath ) return; + + GetPrivateProfileString( SECTION, m_szKey, m_szDefault, m_szData, m_iMaxLen, szIniPath ); + + //INFO_LOG(AUDIO, "ConfString::Read() > GetPrivateProfileString <%s> <%s> <%s>\n", m_szKey, m_szData, szIniPath); + + m_bRead = true; +} + + +// ======================================================================================= +// I don't use this +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfString::Write() +{ + /* + WritePrivateProfileString( SECTION, m_szKey, m_szData, szIniPath ); + m_bRead = true; + */ +} +// ======================================================================================= + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfCurDir::ConfCurDir( TCHAR * szData, TCHAR * szKey ) : ConfString( szData, szKey, + CONF_MODE_INTERNAL, TEXT( "C:\\" ), MAX_PATH ) +{ + +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +ConfCurDir::ConfCurDir( TCHAR * szData, const TCHAR * szKey ) : ConfString( szData, szKey, + CONF_MODE_INTERNAL, TEXT( "C:\\" ), MAX_PATH ) +{ + +} + + +//////////////////////////////////////////////////////////////////////////////// +/* IMPORTANT: This SetCurrentDirectory() has to be disabled or we change the relative paths + for the entire application */ +//////////////////////////////////////////////////////////////////////////////// +void ConfCurDir::Read() +{ + ConfString::Read(); + + // Apply + //SetCurrentDirectory( m_szData ); + + //INFO_LOG(AUDIO,"ConfCurDir::Read > End <%s>\n", m_szData); +} +// ============================================================================== + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ConfCurDir::Write() +{ + // Refresh + GetCurrentDirectory( MAX_PATH, m_szData ); // Note: without trailing slash + + // MessageBox( 0, m_szData, TEXT( "CurDir" ), 0 ); + //INFO_LOG(AUDIO,"ConfCurDir::Read <%s>\n", m_szData); + + ConfString::Write(); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void Conf::Init() +{ + if( szIniPath ) return; + + // ======================================================================================= + // Find the right ini file + // Long filename + + szIniPath = new TCHAR[ _MAX_PATH ]; + + TCHAR szFull[ _MAX_PATH ] = TEXT( "" ); + TCHAR szDrive[ _MAX_DRIVE ] = TEXT( "" ); + TCHAR szDir[ _MAX_DIR ] = TEXT( "" ); + + + // --------------------------------------------------------------------------------------- + // We place the full path in szFull + //GetModuleFileName( hInstance, szFull, _MAX_PATH ); + GetModuleFileName( NULL, szFull, _MAX_PATH ); // No don't use that + // --------------------------------------------------------------------------------------- + + _tsplitpath( szFull, szDrive, szDir, NULL, NULL ); + + + // Convert short to long path + GetLongPathName( szDir, szDir, _MAX_DIR ); + + + // --------------------------------------------------------------------------------------- + // We place information about the file in fd + // --------------------------------------------------------------------------------------- + // Convert short to long file + WIN32_FIND_DATA fd; + HANDLE h = FindFirstFile( szFull, &fd ); + // --------------------------------------------------------------------------------------- + + // --------------------------------------------------------------------------------------- + // Convert File.exe to File.ini + // --------------------------------------------------------------------------------------- + // Search last dot + TCHAR * szSearch = fd.cFileName + _tcslen( fd.cFileName ) - 1; + while( ( *szSearch != TEXT( '.' ) ) && ( szSearch > fd.cFileName ) ) + { + szSearch--; + } + // --------------------------------------------------------------------------------------- + // Replace extension + _tcscpy( szSearch, TEXT( ".ini" ) ); + // --------------------------------------------------------------------------------------- + + // Copy full filename + //_sntprintf( szIniPath, _MAX_PATH, TEXT( "%s%s%s" ), szDrive, szDir, fd.cFileName ); + _sntprintf( szIniPath, _MAX_PATH, TEXT( "%s%s%s" ), szDrive, szDir, TEXT( "Plainamp.ini" ) ); + + INFO_LOG(AUDIO,"DLL > Ini path <%s>\n", szIniPath); + // ======================================================================================= + + + // --------------------------------------------------------------------------------------- + // Read all settings in it + // --------------------------------------------------------------------------------------- + // Read all (original comment) + map::iterator iter = conf_map->begin(); + + // ======================================================================================= + // This will lead us to the Read() function earlier in this file. For example ConfBool::Read() + while( iter != conf_map->end() ) + { + /* By default there was a SetCurrentDirectory() here that would affect the entire process, + exe and everything. It is disabled now. */ + iter->second->Read(); + iter++; + } + // ======================================================================================= + // --------------------------------------------------------------------------------------- + +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void Conf::Write() +{ + if( !szIniPath ) return; + + map::iterator iter = conf_map->begin(); + while( iter != conf_map->end() ) + { + iter->second->Write(); + iter++; + } +} diff --git a/Externals/MusicMod/Player/Src/Config.h b/Externals/MusicMod/Player/Src/Config.h new file mode 100644 index 0000000000..c139c120af --- /dev/null +++ b/Externals/MusicMod/Player/Src/Config.h @@ -0,0 +1,239 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_CONFIG_H +#define PA_CONFIG_H + + + +#include "Global.h" +#include "OutputPlugin.h" + + + +namespace Conf +{ + //void Init( HINSTANCE hInstance ); + void Init( ); + void Write(); +}; + + + +enum ConfMode +{ + CONF_MODE_INTERNAL, // Will not be shown to the user + CONF_MODE_PUBLIC +}; + +class ConfVar; +typedef void ( * ConfCallback )( ConfVar * var ); + +struct BandInfo +{ + int m_iIndex; + int m_iWidth; + bool m_bBreak; + bool m_bVisible; +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Config container +//////////////////////////////////////////////////////////////////////////////// +class ConfVar +{ +public: + ConfVar( TCHAR * szKey, ConfMode mode ); + ConfVar( const TCHAR * szKey, ConfMode mode ); + ~ConfVar(); + +protected: + TCHAR * m_szKey; ///< Unique identifier + ConfMode m_Mode; ///< Mode/visibility + bool m_bRead; ///< Initilization flag + + virtual void Read() = 0; + virtual void Write() = 0; + + // virtual void Backup() = 0; ///< Creates a backup and deletes old backup if it exists + // virtual void Restore() = 0; ///< Restores settings from backup and destroys the backup + +private: + bool m_bCopyKey; ///< Keyname is copy (has to be freed on destruction) + + //friend void Conf::Init( HINSTANCE hInstance ); + friend void Conf::Init( ); + + friend void Conf::Write(); +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Boolean config container +//////////////////////////////////////////////////////////////////////////////// +class ConfBool : public ConfVar +{ +public: + ConfBool( bool * pbData, TCHAR * szKey, ConfMode mode, bool bDefault ); + ConfBool( bool * pbData, const TCHAR * szKey, ConfMode mode, bool bDefault ); + +private: + bool * m_pbData; ///< Target + bool m_bDefault; ///< Default value + + void Read(); + void Write(); + + + friend OutputPlugin::OutputPlugin( TCHAR * szDllpath, bool bKeepLoaded ); +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Integer config container +//////////////////////////////////////////////////////////////////////////////// +class ConfInt : public ConfVar +{ +public: + ConfInt( int * piData, TCHAR * szKey, ConfMode mode, int iDefault ); + ConfInt( int * piData, const TCHAR * szKey, ConfMode mode, int iDefault ); + +protected: + int * m_piData; + int m_iDefault; + + void Read(); + void Write(); +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Integer config container with restricted range +//////////////////////////////////////////////////////////////////////////////// +class ConfIntMinMax : public ConfInt +{ +public: + ConfIntMinMax( int * piData, TCHAR * szKey, ConfMode mode, int iDefault, int iMin, int iMax ); + ConfIntMinMax( int * piData, const TCHAR * szKey, ConfMode mode, int iDefault, int iMin, int iMax ); + +// bool IsValid() { return ( ( *m_piData >= m_iMin ) && ( *m_piData <= m_iMax ) ); } + inline bool IsMin() { return ( *m_piData == m_iMin ); } + inline bool IsMax() { return ( *m_piData == m_iMax ); } + + inline void MakeValidDefault() { if( ( *m_piData < m_iMin ) || ( *m_piData > m_iMax ) ) *m_piData = m_iDefault; } + inline void MakeValidPull() { if( *m_piData < m_iMin ) *m_piData = m_iMin; else if( *m_piData > m_iMax ) *m_piData = m_iMax; } + +private: + int m_iMin; + int m_iMax; + + void Read() { ConfInt::Read(); MakeValidPull(); } +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Window placement config container +/// +/// The callback funtion is called on write() +/// so the data written is up to date. +//////////////////////////////////////////////////////////////////////////////// +class ConfWinPlaceCallback : public ConfVar +{ +public: + ConfWinPlaceCallback( WINDOWPLACEMENT * pwpData, TCHAR * szKey, RECT * prDefault, ConfCallback fpCallback ); + ConfWinPlaceCallback( WINDOWPLACEMENT * pwpData, const TCHAR * szKey, RECT * prDefault, ConfCallback fpCallback ); + + inline void TriggerCallback() { if( m_fpCallback ) m_fpCallback( this ); } + inline void RemoveCallback() { m_fpCallback = NULL; } + +private: + WINDOWPLACEMENT * m_pwpData; + RECT * m_prDefault; + ConfCallback m_fpCallback; + + void Read(); + void Write(); +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Rebar band info config container +/// +/// The callback funtion is called on write() +/// so the data written is up to date. +//////////////////////////////////////////////////////////////////////////////// +class ConfBandInfoCallback : public ConfVar +{ +public: + ConfBandInfoCallback( BandInfo * pbiData, TCHAR * szKey, BandInfo * pbiDefault, ConfCallback fpCallback ); + ConfBandInfoCallback( BandInfo * pbiData, const TCHAR * szKey, BandInfo * pbiDefault, ConfCallback fpCallback ); + + inline void TriggerCallback() { if( m_fpCallback ) m_fpCallback( this ); } + inline void RemoveCallback() { m_fpCallback = NULL; } + + bool Apply( HWND hRebar, int iBandId ); + +private: + BandInfo * m_pbiData; + BandInfo * m_pbiDefault; + ConfCallback m_fpCallback; + + void Read(); + void Write(); +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// String config container +//////////////////////////////////////////////////////////////////////////////// +class ConfString : public ConfVar +{ +public: + ConfString( TCHAR * szData, TCHAR * szKey, ConfMode mode, TCHAR * szDefault, int iMaxLen ); + ConfString( TCHAR * szData, const TCHAR * szKey, ConfMode mode, TCHAR * szDefault, int iMaxLen ); + +protected: + TCHAR * m_szData; + int m_iMaxLen; + TCHAR * m_szDefault; + + void Read(); + void Write(); +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Current directory config container +//////////////////////////////////////////////////////////////////////////////// +class ConfCurDir : public ConfString +{ +public: + ConfCurDir( TCHAR * szData, TCHAR * szKey ); + ConfCurDir( TCHAR * szData, const TCHAR * szKey ); + +private: + void Read(); + void Write(); +}; + + + +#endif // PA_CONFIG_H diff --git a/Externals/MusicMod/Player/Src/Console.cpp b/Externals/MusicMod/Player/Src/Console.cpp new file mode 100644 index 0000000000..08868156d4 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Console.cpp @@ -0,0 +1,228 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Console.h" +#include "Font.h" +#include "Main.h" +#include "Config.h" +#include + + + +HWND WindowConsole = NULL; // extern +int iNext = 0; + +const int iMaxEntries = 10000; + + + +WNDPROC WndprocConsoleBackup = NULL; +LRESULT CALLBACK WndprocConsole( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + + +bool bConsoleVisible; +WINDOWPLACEMENT WinPlaceConsole; + +void WinPlaceConsoleCallback( ConfVar * var ) +{ + if( !IsWindow( WindowConsole ) ) return; + + GetWindowPlacement( WindowConsole, &WinPlaceConsole ); + + // MSDN: If the window identified by the hWnd parameter + // is maximized, the showCmd member is SW_SHOWMAXIMIZED. + // If the window is minimized, showCmd is SW_SHOWMINIMIZED. + // Otherwise, it is SW_SHOWNORMAL. + if( !bConsoleVisible ) + { + WinPlaceConsole.showCmd = SW_HIDE; + } +} + +RECT rConsoleDefault = { 50, 400, 450, 700 }; + +ConfWinPlaceCallback cwpcWinPlaceConsole( + &WinPlaceConsole, + TEXT( "WinPlaceConsole" ), + &rConsoleDefault, + WinPlaceConsoleCallback +); + + + +//////////////////////////////////////////////////////////////////////////////// +/// Creates the console window. +/// Size and visibility is used from config. +/// +/// @return Success flag +//////////////////////////////////////////////////////////////////////////////// +bool Console::Create() +{ + WindowConsole = CreateWindowEx( + WS_EX_TOOLWINDOW | // DWORD dwExStyle + WS_EX_CLIENTEDGE, // + TEXT( "LISTBOX" ), // LPCTSTR lpClassName + TEXT( "Console" ), // LPCTSTR lpWindowName + WS_VSCROLL | // DWORD dwStyle + LBS_DISABLENOSCROLL | // + LBS_EXTENDEDSEL | // + LBS_HASSTRINGS | // + LBS_NOTIFY | // + LBS_NOINTEGRALHEIGHT | // + WS_POPUP | // + WS_OVERLAPPEDWINDOW, // + rConsoleDefault.left, // int x + rConsoleDefault.top, // int y + rConsoleDefault.right - rConsoleDefault.left, // int nWidth + rConsoleDefault.bottom - rConsoleDefault.top, // int nHeight + WindowMain, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + if( !WindowConsole ) return false; + + // A blank line at the bottom will give us more space + SendMessage( WindowConsole, LB_INSERTSTRING, 0, ( LPARAM )TEXT( "" ) ); + + Font::Apply( WindowConsole ); + + bConsoleVisible = ( WinPlaceConsole.showCmd != SW_HIDE ); + SetWindowPlacement( WindowConsole, &WinPlaceConsole ); + + // Exchange window procedure + WndprocConsoleBackup = ( WNDPROC )GetWindowLong( WindowConsole, GWL_WNDPROC ); + if( WndprocConsoleBackup != NULL ) + { + SetWindowLong( WindowConsole, GWL_WNDPROC, ( LONG )WndprocConsole ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Destroys the console window. +/// +/// @return Success flag +//////////////////////////////////////////////////////////////////////////////// +bool Console::Destroy() +{ + if( !WindowConsole ) return false; + DestroyWindow( WindowConsole ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Pops up the console window. +/// +/// @return Success flag +//////////////////////////////////////////////////////////////////////////////// +bool Console::Popup() +{ + if( !WindowConsole ) return false; + if( !IsWindowVisible( WindowConsole ) ) + { + ShowWindow( WindowConsole, SW_SHOW ); + } + + SetActiveWindow( WindowConsole ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Adds a new entry at the end/bottom +/// +/// @param szText Log entry +/// @return Success flag +//////////////////////////////////////////////////////////////////////////////// +bool Console::Append( TCHAR * szText ) +{ + if( !WindowConsole ) return false; + if( iNext > iMaxEntries - 1 ) + { + SendMessage( + WindowConsole, + LB_DELETESTRING, + 0, + 0 + ); + iNext--; + } + + const int uTextLen = ( int )_tcslen( szText ); + TCHAR * szBuffer = new TCHAR[ 11 + uTextLen + 1 ]; + time_t now_time_t = time( NULL ); + struct tm * now_tm = localtime( &now_time_t ); + _tcsftime( szBuffer, 12, TEXT( "%H:%M:%S " ), now_tm ); + memcpy( szBuffer + 11, szText, uTextLen * sizeof( TCHAR ) ); + szBuffer[ 11 + uTextLen ] = TEXT( '\0' ); + + SendMessage( WindowConsole, LB_INSERTSTRING, iNext, ( LPARAM )szBuffer ); + SendMessage( WindowConsole, LB_SETSEL, FALSE, -1 ); + SendMessage( WindowConsole, LB_SETSEL, TRUE, iNext ); + SendMessage( WindowConsole, LB_SETTOPINDEX, iNext, 0 ); + iNext++; + + delete [] szBuffer; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocConsole( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + switch( message ) + { +/* + case WM_CTLCOLORLISTBOX: + if( ( HWND )lp == WindowConsole ) + { + SetBkColor (( HDC )wp, GetSysColor(COLOR_3DFACE)); + return ( LRESULT )GetSysColorBrush(COLOR_3DFACE); + } + break; +*/ + + case WM_SYSCOMMAND: + // Hide instead of closing + if( ( wp & 0xFFF0 ) == SC_CLOSE ) + { + ShowWindow( hwnd, SW_HIDE ); + return 0; + } + break; + + case WM_DESTROY: + cwpcWinPlaceConsole.TriggerCallback(); + cwpcWinPlaceConsole.RemoveCallback(); + break; + + case WM_SHOWWINDOW: + bConsoleVisible = ( wp == TRUE ); + break; + + } + return CallWindowProc( WndprocConsoleBackup, hwnd, message, wp, lp ); +} diff --git a/Externals/MusicMod/Player/Src/Console.h b/Externals/MusicMod/Player/Src/Console.h new file mode 100644 index 0000000000..28386af265 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Console.h @@ -0,0 +1,40 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_CONSOLE_H +#define PA_CONSOLE_H + + + +#include "Global.h" + + + +extern HWND WindowConsole; + + +//////////////////////////////////////////////////////////////////////////////// +/// Logging console window +//////////////////////////////////////////////////////////////////////////////// +namespace Console +{ + bool Create(); + bool Destroy(); + + bool Popup(); + + bool Append( TCHAR * szText ); +} + + + +#endif // PA_CONSOLE_H diff --git a/Externals/MusicMod/Player/Src/DspModule.cpp b/Externals/MusicMod/Player/Src/DspModule.cpp new file mode 100644 index 0000000000..5a49715d1e --- /dev/null +++ b/Externals/MusicMod/Player/Src/DspModule.cpp @@ -0,0 +1,136 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "DspModule.h" +#include "Unicode.h" + + + +DspModule ** active_dsp_mods = NULL; // extern +int active_dsp_count = 0; // extern + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +DspModule::DspModule( char * szName, int iIndex, winampDSPModule * mod, DspPlugin * plugin ) +{ + iArrayIndex = -1; + + iNameLen = ( int )strlen( szName ); + this->szName = new TCHAR[ iNameLen + 1 ]; + ToTchar( this->szName, szName, iNameLen ); + this->szName[ iNameLen ] = TEXT( '\0' ); + + this->iIndex = iIndex; + this->mod = mod; + this->plugin = plugin; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool DspModule::Start( int iIndex ) +{ + if( !mod ) return false; + if( iArrayIndex != -1 ) return false; + if( !mod->Init ) return false; + if( mod->Init( mod ) != 0 ) return false; + +//////////////////////////////////////////////////////////////////////////////// + DspLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + if( !active_dsp_count ) + { + active_dsp_mods = new DspModule * [ 1 ]; + active_dsp_mods[ 0 ] = this; + iArrayIndex = 0; + } + else + { + if( iIndex < 0 ) + iIndex = 0; + else if( iIndex > active_dsp_count ) + iIndex = active_dsp_count; + + DspModule ** new_active_dsp_mods = new DspModule * [ active_dsp_count + 1 ]; + memcpy( new_active_dsp_mods, active_dsp_mods, iIndex * sizeof( DspModule * ) ); + memcpy( new_active_dsp_mods + iIndex + 1, active_dsp_mods + iIndex, ( active_dsp_count - iIndex ) * sizeof( DspModule * ) ); + for( int i = iIndex + 1; i < active_dsp_count + 1; i++ ) + { + new_active_dsp_mods[ i ]->iArrayIndex = i; + } + new_active_dsp_mods[ iIndex ] = this; + iArrayIndex = iIndex; + delete [] active_dsp_mods; + active_dsp_mods = new_active_dsp_mods; + } + active_dsp_count++; + +//////////////////////////////////////////////////////////////////////////////// + DspLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool DspModule::Stop() +{ + if( !mod ) return false; + if( iArrayIndex == -1 ) return false; + if( !mod->Quit ) return true; + +//////////////////////////////////////////////////////////////////////////////// + DspLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + for( int i = iArrayIndex; i < active_dsp_count - 1; i++ ) + { + active_dsp_mods[ i ] = active_dsp_mods[ i + 1 ]; + active_dsp_mods[ i ]->iArrayIndex = i; + } + active_dsp_count--; + +//////////////////////////////////////////////////////////////////////////////// + DspLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + mod->Quit( mod ); + + iArrayIndex = -1; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool DspModule::Config() +{ + if( !mod ) return false; + if( iArrayIndex == -1 ) return false; + if( !mod->Config ) return false; + + mod->Config( mod ); + + return true; +} diff --git a/Externals/MusicMod/Player/Src/DspModule.h b/Externals/MusicMod/Player/Src/DspModule.h new file mode 100644 index 0000000000..5b69b177b1 --- /dev/null +++ b/Externals/MusicMod/Player/Src/DspModule.h @@ -0,0 +1,67 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_DSP_MODULE_H +#define PA_DSP_MODULE_H + + + +#include "Global.h" +#include "DspPlugin.h" +#include "Winamp/Dsp.h" + + + +class DspModule; +class DspPlugin; + +extern DspModule ** active_dsp_mods; +extern int active_dsp_count; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp DSP module wrapper +//////////////////////////////////////////////////////////////////////////////// +class DspModule +{ +public: + inline bool IsActive() { return ( iArrayIndex != -1 ); } + + inline TCHAR * GetName() { return szName; } + inline int GetNameLen() { return iNameLen; } + + DspModule( char * szName, int iIndex, winampDSPModule * mod, DspPlugin * plugin ); +// DspModule( wchar_t * szName, int iIndex, winampVisModule * mod, VisPlugin * plugin ); + + bool Start( int iIndex ); + bool Stop(); + + bool Config(); + +private: + int iArrayIndex; + + TCHAR * szName; + int iNameLen; + + int iIndex; + winampDSPModule * mod; + DspPlugin * plugin; + + + friend int dsp_dosamples( short int * samples, int numsamples, int bps, int nch, int srate ); +}; + + + +#endif // PA_DSP_MODULE_H diff --git a/Externals/MusicMod/Player/Src/DspPlugin.cpp b/Externals/MusicMod/Player/Src/DspPlugin.cpp new file mode 100644 index 0000000000..40988eb8c8 --- /dev/null +++ b/Externals/MusicMod/Player/Src/DspPlugin.cpp @@ -0,0 +1,186 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "DspPlugin.h" +#include "Main.h" +#include "Unicode.h" +#include "Console.h" + + + +vector dsp_plugins; // extern +Lock DspLock = Lock( TEXT( "PLAINAMP_DSP_LOCK" ) ); // extern + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +DspPlugin::DspPlugin( TCHAR * szDllpath, bool bKeepLoaded ) : Plugin( szDllpath ) +{ + header = NULL; + + if( !Load() ) + { + return; + } + + if( !bKeepLoaded ) + { + Unload(); + } + + dsp_plugins.push_back( this ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool DspPlugin::Load() +{ + if( IsLoaded() ) return true; + + // (1) Load DLL + hDLL = LoadLibrary( GetFullpath() ); + if( !hDLL ) return false; + + // (2) Find export + WINAMP_DSP_GETTER winampGetDSPHeader2 = + ( WINAMP_DSP_GETTER )GetProcAddress( hDLL, "winampDSPGetHeader2" ); + if( winampGetDSPHeader2 == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (3) Get header + header = winampGetDSPHeader2(); + if( header == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + +//////////////////////////////////////////////////////////////////////////////// + // Forget old modules or we get them twice + if( !modules.empty() ) + { + modules.clear(); + } +//////////////////////////////////////////////////////////////////////////////// + + if( !szName ) + { + // Note: The prefix is not removed to hide their + // origin at Nullsoft! It just reads easier. + if( !strnicmp( header->description, "nullsoft ", 9 ) ) + { + header->description += 9; + } + iNameLen = ( int )strlen( header->description ); + szName = new TCHAR[ iNameLen + 1 ]; + ToTchar( szName, header->description, iNameLen ); + szName[ iNameLen ] = TEXT( '\0' ); + } + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Loading <%s>, %s" ), GetFilename(), szName ); + Console::Append( szBuffer ); + + // (4) Get modules + winampDSPModule * mod; + int iFound = 0; + while( true ) + { + mod = header->getModule( iFound ); + if( !mod ) break; + + // (4a) Modify module + mod->hDllInstance = hDLL; + mod->hwndParent = WindowMain; + + // (4b) Add module to list + DspModule * dspmod = new DspModule( + mod->description, // char * szName + iFound, // UINT uIndex + mod, // winampDspModule * mod + this // DspPlugin * plugin + ); + modules.push_back( dspmod ); + iFound++; + + _stprintf( szBuffer, TEXT( " %s" ), dspmod->GetName() ); + Console::Append( szBuffer ); + } + + Console::Append( TEXT( " " ) ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool DspPlugin::Unload() +{ + if( !IsLoaded() ) return true; + if( IsActive() ) return false; + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Unloading <%s>" ), GetFilename() ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + printf( ">>>Unloading <%s>\n" , GetFilename() ); + + header = NULL; + + /* + TODO + DspModule * walk; + vector ::iterator iter = modules.begin(); + while( iter != modules.end() ) + { + walk = *iter; + delete [] walk->szName; + delete walk; + + iter++; + } + */ + + FreeLibrary( hDLL ); + hDLL = NULL; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool DspPlugin::IsActive() +{ + vector ::iterator iter = modules.begin(); + while( iter != modules.end() ) + { + if( ( *iter )->IsActive() ) return true; + iter++; + } + return false; +} diff --git a/Externals/MusicMod/Player/Src/DspPlugin.h b/Externals/MusicMod/Player/Src/DspPlugin.h new file mode 100644 index 0000000000..afd299a60c --- /dev/null +++ b/Externals/MusicMod/Player/Src/DspPlugin.h @@ -0,0 +1,69 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_DSP_PLUGIN_H +#define PA_DSP_PLUGIN_H + + + +#include "Global.h" +#include "Plugin.h" +#include "Winamp/Dsp.h" +#include "Lock.h" +#include "DspModule.h" +#include + +using namespace std; + + + +typedef winampDSPHeader * ( * WINAMP_DSP_GETTER )( void ); + + + +class DspModule; +class DspPlugin; + +extern vector dsp_plugins; +extern Lock DspLock; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp DSP plugin wrapper +//////////////////////////////////////////////////////////////////////////////// +class DspPlugin : public Plugin +{ +public: + DspPlugin( TCHAR * szDllpath, bool bKeepLoaded ); + + bool Load(); + bool Unload(); + + TCHAR * GetTypeString() { return TEXT( "DSP" ); } + int GetTypeStringLen() { return 3; } + PluginType GetType() { return PLUGIN_TYPE_DSP; } + + bool IsActive(); + +private: + winampDSPHeader * header; + vector modules; + + + friend class DspModule; + friend void ContextMenuDsp( DspPlugin * dsp, POINT * p ); +}; + + + +#endif // PA_DSP_PLUGIN_H diff --git a/Externals/MusicMod/Player/Src/Emabox/Emabox.cpp b/Externals/MusicMod/Player/Src/Emabox/Emabox.cpp new file mode 100644 index 0000000000..1f1d2bf032 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Emabox/Emabox.cpp @@ -0,0 +1,455 @@ +/*////////////////////////////////////////////////////////////////////////////// +// ExtraMessageBox +// +// Copyright İ 2006 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////*/ + + +/* +TODO +* realign/recenter after height change +* tab stop order when adding buttons +* offer extra callback? +* auto click timer (one button after XXX seconds) +* allow several checkboxes? radio buttons? + +MB_YESNO +MB_YESNOCANCEL +--> MB_YESNOALL +--> MB_YESNOCANCELALL +--> MB_DEFBUTTON5 +--> IDNOALL +--> IDYESALL +*/ + + +#include "Emabox.h" + + + +#define FUNCTION_NORMAL 0 +#define FUNCTION_EXTENDED 1 +#define FUNCTION_INDIRECT 2 + + + +const int SPACE_UNDER_CHECKBOX = 10; +const int SPACE_EXTRA_BOTTOM = 4; + +TCHAR * const szNeverAgain = TEXT( "Do not show again" ); +TCHAR * const szRememberChoice = TEXT( "Remember my choice" ); + +DWORD dwTlsSlot = TLS_OUT_OF_INDEXES; + +#ifdef EMA_AUTOINIT +int bEmaInitDone = 0; +#endif + + + +struct StructEmaBoxData +{ + int * bCheckState; + HHOOK hCBT; /* CBT hook handle */ + WNDPROC WndprocMsgBoxBackup; /* Old wndproc */ + UINT uType; /* Message box type */ + HWND hCheck; /* Checkbox handle */ +}; + +typedef struct StructEmaBoxData EmaBoxData; + + + +void RectScreenToClient( const HWND h, RECT * const r ) +{ + POINT p; + RECT after; + + p.x = r->left; + p.y = r->top; + ScreenToClient( h, &p ); + + after.left = p.x; + after.right = p.x + r->right - r->left; + after.top = p.y; + after.bottom = p.y + r->bottom - r->top; + + memcpy( r, &after, sizeof( RECT ) ); +} + + + +LRESULT CALLBACK WndprocMsgBox( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + /* Find data */ + EmaBoxData * const data = ( EmaBoxData * )TlsGetValue( dwTlsSlot ); + + switch( message ) + { + case WM_COMMAND: + if( HIWORD( wp ) == BN_CLICKED ) + { + if( !data->hCheck || ( ( HWND )lp != data->hCheck ) ) break; + + { + const LRESULT res = SendMessage( ( HWND )lp, BM_GETSTATE, 0, 0 ); + const int bCheckedAfter = ( ( res & BST_CHECKED ) == 0 ); + + /* Update external variable */ + *( data->bCheckState ) = bCheckedAfter ? 1 : 0; + + SendMessage( ( HWND )lp, BM_SETCHECK, ( bCheckedAfter ) ? BST_CHECKED : 0, 0 ); + } + } + break; + + case WM_INITDIALOG: + { + /* Add checkbox */ + if( ( data->uType & MB_CHECKMASC ) != 0 ) + { + int SPACE_OVER_CHECKBOX; + HDC hdc; + RECT rw; /* Window rect */ + RECT rc; /* Client rect */ + HWND hText; /* Message handle */ + RECT rt; /* Message rect */ + int iLabelHeight; + TCHAR * szCheckboxLabel; /* Checkbox label */ + + int iWindowWidthBefore; + int iWindowHeightBefore; + int iClientWidthBefore; + int iClientHeightBefore; + int iNeverAgainWidth; + int iNeverAgainHeight; + + + + /* Get original window dimensions */ + GetWindowRect( hwnd, &rw ); + iWindowWidthBefore = rw.right - rw.left; + iWindowHeightBefore = rw.bottom - rw.top; + + GetClientRect( hwnd, &rc ); + iClientWidthBefore = rc.right - rc.left; + iClientHeightBefore = rc.bottom - rc.top; + + + + { + /* Find handle of the text label */ + HWND hFirstStatic; + HWND hSecondStatic; + + hFirstStatic = FindWindowEx( hwnd, NULL, TEXT( "STATIC" ), NULL ); + if( !hFirstStatic ) break; + + hSecondStatic = FindWindowEx( hwnd, hFirstStatic, TEXT( "STATIC" ), NULL ); + if( !hSecondStatic ) + { + /* Only one static means no icon. */ + /* So hFirstStatic must be the text window. */ + hText = hFirstStatic; + } + else + { + TCHAR szBuf[ 2 ] = TEXT( "" ); + if( !GetWindowText( hSecondStatic, szBuf, 2 ) ) break; + + if( *szBuf != TEXT( '\0' ) ) + { + /* Has text so it must be the label */ + hText = hSecondStatic; + } + else + { + hText = hFirstStatic; + } + } + } + + GetWindowRect( hText, &rt ); + RectScreenToClient( hwnd, &rt ); + + iLabelHeight = rt.bottom - rt.top; + + { + /* Get distance between label and the buttons */ + HWND hAnyButton; + RECT rab; + + hAnyButton = FindWindowEx( hwnd, NULL, TEXT( "BUTTON" ), NULL ); + if( !hAnyButton ) break; + + GetWindowRect( hAnyButton, &rab ); + RectScreenToClient( hwnd, &rab ); + + SPACE_OVER_CHECKBOX = rab.top - rt.bottom; + } + + szCheckboxLabel = ( data->uType & MB_CHECKNEVERAGAIN ) + ? EMA_TEXT_NEVER_AGAIN + : EMA_TEXT_REMEMBER_CHOICE; + + /* Add checkbox */ + data->hCheck = CreateWindow( + TEXT( "BUTTON" ), + szCheckboxLabel, + WS_CHILD | + WS_VISIBLE | + WS_TABSTOP | + BS_VCENTER | + BS_CHECKBOX, + CW_USEDEFAULT, + CW_USEDEFAULT, + CW_USEDEFAULT, + CW_USEDEFAULT, + hwnd, + NULL, + GetModuleHandle( NULL ), + NULL + ); + + + /* Set initial check state */ + SendMessage( data->hCheck, BM_SETCHECK, *( data->bCheckState ) ? BST_CHECKED : 0, 0 ); + + { + /* Apply default font */ + const int cyMenuSize = GetSystemMetrics( SM_CYMENUSIZE ); + const int cxMenuSize = GetSystemMetrics( SM_CXMENUSIZE ); + const HFONT hNewFont = ( HFONT )GetStockObject( DEFAULT_GUI_FONT ); + HFONT hOldFont; + SIZE size; + + SendMessage( data->hCheck, WM_SETFONT, ( WPARAM )hNewFont, ( LPARAM )TRUE ); + + hdc = GetDC( data->hCheck ); + hOldFont = ( HFONT )SelectObject( hdc, GetStockObject( DEFAULT_GUI_FONT ) ); + GetTextExtentPoint32( hdc, szCheckboxLabel, _tcslen( szCheckboxLabel ), &size ); + SelectObject( hdc, hOldFont ); + ReleaseDC( data->hCheck, hdc ); + + iNeverAgainWidth = cxMenuSize + size.cx + 1; + iNeverAgainHeight = ( cyMenuSize > size.cy ) ? cyMenuSize : size.cy; + } + + MoveWindow( + data->hCheck, + ( iClientWidthBefore - ( iNeverAgainWidth ) ) / 2, + rt.top + iLabelHeight + SPACE_OVER_CHECKBOX, + iNeverAgainWidth, + iNeverAgainHeight, + FALSE + ); + + { + /* Move all buttons down (except the checkbox) */ + const int iDistance = iNeverAgainHeight + SPACE_UNDER_CHECKBOX; + HWND hLastButton = NULL; + RECT rb; + for( ; ; ) + { + hLastButton = FindWindowEx( hwnd, hLastButton, TEXT( "BUTTON" ), NULL ); + if( !hLastButton ) break; + if( hLastButton == data->hCheck ) continue; + + GetWindowRect( hLastButton, &rb ); + RectScreenToClient( hwnd, &rb ); + + MoveWindow( hLastButton, rb.left, rb.top + iDistance, rb.right - rb.left, rb.bottom - rb.top, FALSE ); + } + + + /* Enlarge dialog */ + MoveWindow( hwnd, rw.left, rw.top, iWindowWidthBefore, iWindowHeightBefore + iDistance + SPACE_EXTRA_BOTTOM, FALSE ); + } + } + else + { + data->hCheck = NULL; + } + + /* Modify close button */ + switch( data->uType & MB_CLOSEMASK ) + { + case MB_DISABLECLOSE: + { + const HMENU hSysMenu = GetSystemMenu( hwnd, FALSE ); + EnableMenuItem( hSysMenu, SC_CLOSE, MF_BYCOMMAND | MF_GRAYED ); + } + break; + + case MB_NOCLOSE: + { + const LONG style = GetWindowLong( hwnd, GWL_STYLE ); + if( ( style & WS_SYSMENU ) == 0 ) break; + SetWindowLong( hwnd, GWL_STYLE, ( LONG )( style - WS_SYSMENU ) ); + } + break; + + } + } + break; + + } + return CallWindowProc( data->WndprocMsgBoxBackup, hwnd, message, wp, lp ); +} + + + +/* int bFound = 0; */ + +LRESULT CALLBACK HookprocMsgBox( int code, WPARAM wp, LPARAM lp ) +{ + /* Get hook handle */ + EmaBoxData * const data = ( EmaBoxData * )TlsGetValue( dwTlsSlot ); + + if( code == HCBT_CREATEWND ) + { + /* MSDN says WE CANNOT TRUST "CBT_CREATEWND" */ + /* so we use only the window handle */ + /* and get the class name using "GetClassName". (-> Q106079) */ + HWND hwnd = ( HWND )wp; + + /* Check windowclass */ + TCHAR szClass[ 7 ] = TEXT( "" ); + GetClassName( hwnd, szClass, 7 ); + if( !_tcscmp( szClass, TEXT( "#32770" ) ) ) + { +/* + if( bFound ) + { + return CallNextHookEx( hCBT, code, wp, lp ); + } + + bFound = 1; +*/ + /* Exchange window procedure */ + data->WndprocMsgBoxBackup = ( WNDPROC )GetWindowLong( hwnd, GWL_WNDPROC ); + if( data->WndprocMsgBoxBackup != NULL ) + { + SetWindowLong( hwnd, GWL_WNDPROC, ( LONG )WndprocMsgBox ); + } + } + } + return CallNextHookEx( data->hCBT, code, wp, lp ); +} + + + +int ExtraAllTheSame( const HWND hWnd, const LPCTSTR lpText, const LPCTSTR lpCaption, const UINT uType, const WORD wLanguageId, const LPMSGBOXPARAMS lpMsgBoxParams, int * const pbCheckRes, const int iFunction ) +{ + EmaBoxData * data; + HHOOK hCBT; + int res; + +#ifdef EMA_AUTOINIT + if( !bEmaInitDone ) + { + EmaBoxLive(); + bEmaInitDone = 1; + } +#endif + + /* Create thread data */ + data = ( EmaBoxData * )LocalAlloc( NONZEROLPTR, sizeof( EmaBoxData ) ); + TlsSetValue( dwTlsSlot, data ); + data->bCheckState = pbCheckRes; + data->uType = ( iFunction != FUNCTION_INDIRECT ) ? uType : lpMsgBoxParams->dwStyle; + + /* Setup this-thread-only hook */ + hCBT = SetWindowsHookEx( WH_CBT, &HookprocMsgBox, GetModuleHandle( NULL ), GetCurrentThreadId() ); + + switch( iFunction ) + { + case FUNCTION_NORMAL: + res = MessageBox( hWnd, lpText, lpCaption, uType ); + break; + + case FUNCTION_EXTENDED: + res = MessageBoxEx( hWnd, lpText, lpCaption, uType, wLanguageId ); + break; + + case FUNCTION_INDIRECT: + res = MessageBoxIndirect( lpMsgBoxParams ); + break; + + } + + /* Remove hook */ + if( hCBT != NULL ) UnhookWindowsHookEx( hCBT ); + + /* Destroy thread data */ + LocalFree( ( HLOCAL )data ); + + return res; +} + + + +int EmaBox( HWND hWnd, LPCTSTR lpText, LPCTSTR lpCaption, UINT uType, int * pbCheckRes ) +{ + /* Check extra flags */ + if( ( uType & MB_EXTRAMASC ) == 0 ) + { + /* No extra */ + return MessageBox( hWnd, lpText, lpCaption, uType ); + } + + return ExtraAllTheSame( hWnd, lpText, lpCaption, uType, 0, NULL, pbCheckRes, FUNCTION_NORMAL ); +} + + + +int EmaBoxEx( HWND hWnd, LPCTSTR lpText, LPCTSTR lpCaption, UINT uType, WORD wLanguageId, int * pbCheckRes ) +{ + /* Check extra flags */ + if( ( uType & MB_EXTRAMASC ) == 0 ) + { + /* No extra */ + return MessageBoxEx( hWnd, lpText, lpCaption, uType, wLanguageId ); + } + + return ExtraAllTheSame( hWnd, lpText, lpCaption, uType, wLanguageId, NULL, pbCheckRes, FUNCTION_EXTENDED ); +} + + + +int EmaBoxIndirect( const LPMSGBOXPARAMS lpMsgBoxParams, int * pbCheckRes ) +{ + /* Check extra flags */ + if( ( lpMsgBoxParams->dwStyle & MB_EXTRAMASC ) == 0 ) + { + /* No extra */ + return MessageBoxIndirect( lpMsgBoxParams ); + } + + return ExtraAllTheSame( NULL, NULL, NULL, 0, 0, lpMsgBoxParams, pbCheckRes, FUNCTION_INDIRECT ); +} + + + +int EmaBoxLive() +{ + dwTlsSlot = TlsAlloc(); + if( dwTlsSlot == TLS_OUT_OF_INDEXES ) return 0; + + return 1; +} + + + +int EmaBoxDie() +{ + if( dwTlsSlot == TLS_OUT_OF_INDEXES ) return 0; + + TlsFree( dwTlsSlot ); + return 1; +} diff --git a/Externals/MusicMod/Player/Src/Emabox/Emabox.h b/Externals/MusicMod/Player/Src/Emabox/Emabox.h new file mode 100644 index 0000000000..873ec62f0c --- /dev/null +++ b/Externals/MusicMod/Player/Src/Emabox/Emabox.h @@ -0,0 +1,141 @@ +/*////////////////////////////////////////////////////////////////////////////// +// ExtraMessageBox +// +// Copyright İ 2006 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////*/ + + +#ifndef EXTRA_MESSAGE_BOX_H +#define EXTRA_MESSAGE_BOX_H 1 + +#include "EmaboxConfig.h" +#include +#include + +// ======================================================================================= +// Global declarations +#include "../Global.h" +// ======================================================================================= + +/* +== TYPE ============================================================================= +#define MB_TYPEMASK 15 1111 +#define MB_OK 0 0000 +#define MB_OKCANCEL 1 0001 +#define MB_ABORTRETRYIGNORE 2 0010 +#define MB_YESNOCANCEL 3 0011 +#define MB_YESNO 4 0100 +#define MB_RETRYCANCEL 5 0101 +#define MB_CANCELTRYCONTINUE 6 0110 +*/ + +#define MB_YESNOCANCELALL 7 /* 0111 */ +#define MB_YESNOALL 8 /* 1000 */ + +/* +== ICON ============================================================================= +#define MB_ICONMASK 240 11110000 +#define MB_ICONERROR 16 00010000 +#define MB_ICONHAND 16 00010000 +#define MB_ICONSTOP 16 00010000 +#define MB_ICONQUESTION 32 00100000 +#define MB_ICONEXCLAMATION 0x30 00110000 +#define MB_ICONWARNING 0x30 00110000 +#define MB_ICONINFORMATION 64 01000000 +#define MB_ICONASTERISK 64 01000000 +#define MB_USERICON 128 10000000 + +== DEFAULT BUTTON =================================================================== +#define MB_DEFMASK 3840 111100000000 +#define MB_DEFBUTTON1 0 000000000000 +#define MB_DEFBUTTON2 256 000100000000 +#define MB_DEFBUTTON3 512 001000000000 +#define MB_DEFBUTTON4 0x300 001100000000 +*/ + +#define MB_DEFBUTTON5 1024 /* 010000000000 */ +#define MB_DEFBUTTON6 1280 /* 010100000000 */ + +/* +== MODE ============================================================================= +#define MB_MODEMASK 0x00003000 11000000000000 +#define MB_APPLMODAL 0 00000000000000 +#define MB_SYSTEMMODAL 4096 01000000000000 +#define MB_TASKMODAL 0x2000 10000000000000 + +== MISC ============================================================================= +#define MB_MISCMASK 0x0000C000 1100000000000000 +#define MB_HELP 0x4000 0100000000000000 +#define MB_NOFOCUS 0x00008000 1000000000000000 + +== FLAGS ============================================================================ +#define MB_SETFOREGROUND 0x10000 10000000000000000 +#define MB_DEFAULT_DESKTOP_ONLY 0x20000 100000000000000000 +#define MB_TOPMOST 0x40000 1000000000000000000 +#define MB_SERVICE_NOTIFICATION_NT3X 0x00040000 1000000000000000000 +#define MB_SERVICE_NOTIFICATION 0x00040000 1000000000000000000 +#define MB_TOPMOST 0x40000 1000000000000000000 +#define MB_RIGHT 0x80000 10000000000000000000 +#define MB_RTLREADING 0x100000 100000000000000000000 +#define MB_SERVICE_NOTIFICATION 0x00200000 1000000000000000000000 + +== EXTRA FLAGS ====================================================================== +*/ + +#define MB_EXTRAMASC 0xF0000000 /* 11110000000000000000000000000000 */ + +#define MB_CHECKMASC 0xC0000000 /* 11000000000000000000000000000000 */ +#define MB_CHECKNONE 0 /* 00000000000000000000000000000000 */ +#define MB_CHECKNEVERAGAIN 0x40000000 /* 01000000000000000000000000000000 */ +#define MB_CHECKREMEMBERCHOICE 0x80000000 /* 10000000000000000000000000000000 */ + +#define MB_CLOSEMASK 0x30000000 /* 00110000000000000000000000000000 */ +#define MB_NORMALCLOSE 0 /* 00000000000000000000000000000000 */ +#define MB_DISABLECLOSE 0x10000000 /* 00010000000000000000000000000000 */ +#define MB_NOCLOSE 0x20000000 /* 00100000000000000000000000000000 */ + + + +/* Function aliases */ +#define ExtraMessageBoxLive EmaBoxLive +#define ExtraMessageBoxDie EmaBoxDie +#define ExtraMessageBox EmaBox +#define ExtraMessageBoxEx EmaBoxEx +#define ExtraMessageBoxIndirect EmaBoxIndirect + + + +int EmaBoxLive(); + +int EmaBoxDie(); + +int EmaBox( + HWND hWnd, + LPCTSTR lpText, + LPCTSTR lpCaption, + UINT uType, + int * pbCheckRes +); + +int EmaBoxEx( + HWND hWnd, + LPCTSTR lpText, + LPCTSTR lpCaption, + UINT uType, + WORD wLanguageId, + int * pbCheckRes +); + +int EmaBoxIndirect( + const LPMSGBOXPARAMS lpMsgBoxParams, + int * pbCheckRes +); + + + +#endif /* EXTRA_MESSAGE_BOX_H */ diff --git a/Externals/MusicMod/Player/Src/Emabox/EmaboxConfig.h b/Externals/MusicMod/Player/Src/Emabox/EmaboxConfig.h new file mode 100644 index 0000000000..c7730335cd --- /dev/null +++ b/Externals/MusicMod/Player/Src/Emabox/EmaboxConfig.h @@ -0,0 +1,32 @@ +/*////////////////////////////////////////////////////////////////////////////// +// ExtraMessageBox +// +// Copyright İ 2006 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////*/ + + +#ifndef EXTRA_MESSAGE_BOX_CONFIG_H +#define EXTRA_MESSAGE_BOX_CONFIG_H 1 + + + +/* Allow laziness */ +#define EMA_AUTOINIT + +/* Allow overwriting message text */ +#ifndef EMA_TEXT_NEVER_AGAIN +# define EMA_TEXT_NEVER_AGAIN szNeverAgain +#endif + +#ifndef EMA_TEXT_REMEMBER_CHOICE +# define EMA_TEXT_REMEMBER_CHOICE szRememberChoice +#endif + + + +#endif /* EXTRA_MESSAGE_BOX_CONFIG_H */ diff --git a/Externals/MusicMod/Player/Src/Embed.cpp b/Externals/MusicMod/Player/Src/Embed.cpp new file mode 100644 index 0000000000..fdfb456e2e --- /dev/null +++ b/Externals/MusicMod/Player/Src/Embed.cpp @@ -0,0 +1,233 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Embed.h" +#include "Console.h" + + + +#define CLASSNAME_EMBED TEXT( "Winamp Gen" ) +#define TITLE_EMBED TEXT( "Embed target" ) + +#define EMBED_WIDTH 320 +#define EMBED_HEIGHT 240 + + + +const TCHAR * const szEmbedTitle = TITLE_EMBED; +bool bEmbedClassRegistered = false; + + +LRESULT CALLBACK WndprocEmbed( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + + +//////////////////////////////////////////////////////////////////////////////// +/// Creates a new embed window. +/// +/// @param ews Embed window state +/// @return New embed window handle +//////////////////////////////////////////////////////////////////////////////// +HWND Embed::Embed( embedWindowState * ews ) +{ + // Register class + if ( !bEmbedClassRegistered ) + { + WNDCLASS wc = { + 0, // UINT style + WndprocEmbed, // WNDPROC lpfnWndProc + 0, // int cbClsExtra + 0, // int cbWndExtra + g_hInstance, // HINSTANCE hInstance + NULL, // HICON hIcon + LoadCursor( NULL, IDC_ARROW ), // HCURSOR hCursor + ( HBRUSH )COLOR_WINDOW, // HBRUSH hbrBackground + NULL, // LPCTSTR lpszMenuName + CLASSNAME_EMBED // LPCTSTR lpszClassName + }; + + if( !RegisterClass( &wc ) ) return NULL; + + bEmbedClassRegistered = true; + } + + // Create window + HWND WindowEmbed = CreateWindowEx( + WS_EX_WINDOWEDGE | // DWORD dwExStyle + WS_EX_TOOLWINDOW, // + CLASSNAME_EMBED, // LPCTSTR lpClassName + szEmbedTitle, // LPCTSTR lpWindowName + WS_OVERLAPPED | // DWORD dwStyle + WS_CLIPCHILDREN | // + WS_BORDER | // + WS_CAPTION | // + WS_SYSMENU | // + WS_THICKFRAME | // + WS_MINIMIZEBOX | // + WS_MAXIMIZEBOX, // + 10, // int x + 10, // int y + EMBED_WIDTH, // int nWidth + EMBED_HEIGHT, // int nHeight + NULL, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + Console::Append( TEXT( "Embed window born" ) ); + Console::Append( TEXT( " " ) ); + + if( !ews || !ews->me ) return WindowEmbed; + + SetParent( ews->me, WindowEmbed ); + + return WindowEmbed; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +inline bool SameThread( HWND hOther ) +{ + const DWORD dwOtherThreadId = GetWindowThreadProcessId( hOther, NULL ); + const DWORD dwThisThreadId = GetCurrentThreadId(); + return ( dwOtherThreadId == dwThisThreadId ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocEmbed( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ +// static bool bAllowSizeMove = false; + + switch( message ) + { + + case WM_PARENTNOTIFY: + switch( LOWORD( wp ) ) + { + case WM_DESTROY: + { + const HWND hChild = GetWindow( hwnd, GW_CHILD ); + if( !SameThread( hChild ) ) + { + // Vis plugin + DestroyWindow( hwnd ); + } + break; + } + + } + break; + + case WM_SIZE: + { + const HWND hChild = GetWindow( hwnd, GW_CHILD ); + if( !hChild ) break; + MoveWindow( hChild, 0, 0, LOWORD( lp ), HIWORD( lp ), TRUE ); + break; + } + +/* + case WM_ENTERSIZEMOVE: + bAllowSizeMove = true; + break; + + case WM_EXITSIZEMOVE: + bAllowSizeMove = false; + break; + + case WM_WINDOWPOSCHANGING: + { + WINDOWPOS * pos = ( WINDOWPOS * )lp; + + // Update child + if( IsWindow( WindowEmbedChild = GetWindow( WindowEmbed, GW_CHILD ) ) ) + { + RECT r; + GetClientRect( WindowEmbed, &r ); + MoveWindow( WindowEmbedChild, 0, 0, r.right, r.bottom, TRUE ); + } + + if( !bAllowSizeMove ) + { + // Force SWP_NOMOVE + if( ( pos->flags & SWP_NOMOVE ) == 0 ) + { + pos->flags |= SWP_NOMOVE; + } + + // Force SWP_NOSIZE + if( ( pos->flags & SWP_NOSIZE ) == 0 ) + { + pos->flags |= SWP_NOSIZE; + } + + return 0; + } + + break; + } +*/ + case WM_SHOWWINDOW: + { + const HWND hChild = GetWindow( hwnd, GW_CHILD ); + if( wp ) // Shown + { + // Update child size + RECT r; + GetClientRect( hwnd, &r ); + MoveWindow( hChild, 0, 0, r.right, r.bottom, TRUE ); + } + else // Hidden + { + ShowWindow( hChild, SW_HIDE ); + DestroyWindow( hChild ); + } + break; + } + + case WM_SYSCOMMAND: + if( ( wp & 0xFFF0 ) == SC_CLOSE ) + { + const HWND hChild = GetWindow( hwnd, GW_CHILD ); + if( SameThread( hChild ) ) + { + // Not a vis plugin + ShowWindow( hwnd, SW_HIDE ); + return 0; + } + } + break; + + case WM_DESTROY: + { + const HWND hChild = GetWindow( hwnd, GW_CHILD ); + if( hChild && SameThread( hChild ) ) + { + DestroyWindow( hChild ); + } + + Console::Append( TEXT( "Embed window dead" ) ); + Console::Append( TEXT( " " ) ); + break; + } + + } + return DefWindowProc( hwnd, message, wp, lp ); +} diff --git a/Externals/MusicMod/Player/Src/Embed.h b/Externals/MusicMod/Player/Src/Embed.h new file mode 100644 index 0000000000..fc1a7d8611 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Embed.h @@ -0,0 +1,36 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_EMBED_H +#define PA_EMBED_H + + + +#include "Global.h" +#include "Winamp/wa_ipc.h" + + + +//////////////////////////////////////////////////////////////////////////////// +/// Embed window service. +/// Winamp provides embed windows so plugins don't have to take care +/// of window skinning. A plugin let's Winamp create an embed window +/// and uses this new window as parent for its own window. +//////////////////////////////////////////////////////////////////////////////// +namespace Embed +{ + HWND Embed( embedWindowState * ews ); +}; + + + +#endif // PA_EMBED_H diff --git a/Externals/MusicMod/Player/Src/Font.cpp b/Externals/MusicMod/Player/Src/Font.cpp new file mode 100644 index 0000000000..a47b217bb9 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Font.cpp @@ -0,0 +1,83 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Font.h" + + + +HFONT hFont = NULL; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Font::Create() +{ + hFont = CreateFont( + -11, // int nHeight + 0, // int nWidth + 0, // int nEscapement + 0, // int nOrientation + FW_REGULAR, // int fnWeight + FALSE, // DWORD fdwItalic + FALSE, // DWORD fdwUnderline + FALSE, // DWORD fdwStrikeOut + ANSI_CHARSET, // DWORD fdwCharSet + OUT_TT_PRECIS, // DWORD fdwOutputPrecision + CLIP_DEFAULT_PRECIS, // DWORD fdwClipPrecision + ANTIALIASED_QUALITY, // DWORD fdwQuality + FF_DONTCARE | DEFAULT_PITCH, // DWORD fdwPitchAndFamily + TEXT( "Verdana" ) // LPCTSTR lpszFace + ); + + return ( hFont != NULL ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Font::Destroy() +{ + if( !hFont ) return false; + DeleteObject( hFont ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Font::Apply( HWND hwnd ) +{ + if( !hFont ) return false; + SendMessage( + hwnd, + WM_SETFONT, + ( WPARAM )hFont, + FALSE + ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +HFONT Font::Get() +{ + return hFont; +} diff --git a/Externals/MusicMod/Player/Src/Font.h b/Externals/MusicMod/Player/Src/Font.h new file mode 100644 index 0000000000..d32bec237e --- /dev/null +++ b/Externals/MusicMod/Player/Src/Font.h @@ -0,0 +1,34 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_FONT_H +#define PA_FONT_H + + + +#include "Global.h" + + + +namespace Font +{ + bool Create(); + bool Destroy(); + + bool Apply( HWND hwnd ); + HFONT Get(); +}; + + + +#endif // PA_FONT_H + diff --git a/Externals/MusicMod/Player/Src/GenPlugin.cpp b/Externals/MusicMod/Player/Src/GenPlugin.cpp new file mode 100644 index 0000000000..8856229bbe --- /dev/null +++ b/Externals/MusicMod/Player/Src/GenPlugin.cpp @@ -0,0 +1,199 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "GenPlugin.h" +#include "Main.h" +#include "Unicode.h" +#include "Console.h" +#include + + + +vector gen_plugins; // extern + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +GenPlugin::GenPlugin( TCHAR * szDllpath, bool bKeepLoaded ) : Plugin( szDllpath ) +{ + iHookerIndex = -1; + plugin = NULL; + + if( !Load() ) + { + return; + } + + gen_plugins.push_back( this ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool GenPlugin::Load() +{ + if( IsLoaded() ) return true; + + // (1) Load DLL + hDLL = LoadLibrary( GetFullpath() ); + if( !hDLL ) return false; + + // (2) Find export + WINAMP_GEN_GETTER winampGetGeneralPurposePlugin = + ( WINAMP_GEN_GETTER )GetProcAddress( hDLL, "winampGetGeneralPurposePlugin" ); + if( winampGetGeneralPurposePlugin == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (3) Get module + plugin = winampGetGeneralPurposePlugin(); + if( !plugin ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (4) Process module + plugin->hDllInstance = hDLL; + plugin->hwndParent = WindowMain; + + // Note: Some plugins (mainly old ones) set description in init. + // Therefore we init first and copy the name after. + + + // (5) Init + if( plugin->init ) + { + const WNDPROC WndprocBefore = ( WNDPROC )GetWindowLong( WindowMain, GWL_WNDPROC ); + plugin->init(); + const WNDPROC WndprocAfter = ( WNDPROC )GetWindowLong( WindowMain, GWL_WNDPROC ); + + if( WndprocBefore != WndprocAfter ) + { + WndprocBackup = WndprocBefore; + iHookerIndex = iWndprocHookCounter++; + } + } + + + if( !szName ) + { + // Note: The prefix is not removed to hide their + // origin at Nullsoft! It just reads easier. + if( !strnicmp( plugin->description, "nullsoft ", 9 ) ) + { + plugin->description += 9; + } + + // Get rid of " (xxx.dll)" postfix + char * walk = plugin->description + strlen( plugin->description ) - 5; + while( true ) + { + if( ( walk <= plugin->description ) || strnicmp( walk, ".dll)", 5 ) ) break; + while( ( walk > plugin->description ) && ( *walk != '(' ) ) walk--; + if( walk <= plugin->description ) break; + walk--; + if( ( walk <= plugin->description ) || ( *walk != ' ' ) ) break; + *walk = '\0'; + } + + iNameLen = ( int )strlen( plugin->description ); + szName = new TCHAR[ iNameLen + 1 ]; + ToTchar( szName, plugin->description, iNameLen ); + szName[ iNameLen ] = TEXT( '\0' ); + } + + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Loading <%s>, %s" ), GetFilename(), szName ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + + + // Note: Plugins that use a wndproc hook need + // to be unloaded in the inverse loading order. + // This is due to the nature of wndproc hooking. + if( iHookerIndex != -1 ) + { + Console::Append( TEXT( "Wndproc hook added (by plugin)" ) ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool GenPlugin::Unload() +{ + if( !IsLoaded() ) return true; + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Unloading <%s>" ), GetFilename() ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + printf( ">>>Unloading <%s>\n" , GetFilename() ); + + + // Quit + if( plugin ) + { + if( plugin->quit ) plugin->quit(); + plugin = NULL; + } + + + // Remove wndproc hook + if( ( iHookerIndex != -1 ) && ( iHookerIndex == iWndprocHookCounter - 1 ) ) + { + // If we don't restore it the plugins wndproc will + // still be called which is not there anymore -> crash + SetWindowLong( WindowMain, GWL_WNDPROC, ( LONG )WndprocBackup ); + Console::Append( TEXT( "Wndproc hook removed (by host)" ) ); + Console::Append( TEXT( " " ) ); + + iHookerIndex = -1; + iWndprocHookCounter--; + } + + + FreeLibrary( hDLL ); + hDLL = NULL; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool GenPlugin::Config() +{ + if( !IsLoaded() ) return false; + if( !plugin ) return false; + if( !plugin->config ) return false; + + plugin->config(); + + return true; +} diff --git a/Externals/MusicMod/Player/Src/GenPlugin.h b/Externals/MusicMod/Player/Src/GenPlugin.h new file mode 100644 index 0000000000..b32f49536a --- /dev/null +++ b/Externals/MusicMod/Player/Src/GenPlugin.h @@ -0,0 +1,64 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_GEN_PLUGIN_H +#define PA_GEN_PLUGIN_H + + + +#include "Global.h" +#include "Plugin.h" +#include "Winamp/Gen.h" +#include + +using namespace std; + + + +typedef winampGeneralPurposePlugin * ( * WINAMP_GEN_GETTER )( void ); + + + +class GenPlugin; + +extern vector gen_plugins; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp general purpose plugin wrapper +//////////////////////////////////////////////////////////////////////////////// +class GenPlugin : public Plugin +{ +public: + GenPlugin( TCHAR * szDllpath, bool bKeepLoaded ); + + bool Load(); + bool Unload(); + + TCHAR * GetTypeString() { return TEXT( "General" ); } + int GetTypeStringLen() { return 7; } + PluginType GetType() { return PLUGIN_TYPE_GEN; } + + inline bool IsActive() { return IsLoaded(); } + + bool Config(); + + inline bool AllowRuntimeUnload() { return ( iHookerIndex == -1 ) || ( iHookerIndex == iWndprocHookCounter - 1 ); } + +private: + winampGeneralPurposePlugin * plugin; +}; + + + +#endif // PA_GEN_PLUGIN_H diff --git a/Externals/MusicMod/Player/Src/Global.h b/Externals/MusicMod/Player/Src/Global.h new file mode 100644 index 0000000000..684774048d --- /dev/null +++ b/Externals/MusicMod/Player/Src/Global.h @@ -0,0 +1,151 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +////////////////////////////////////////////////////////////////////////////////////////// +// Settings +// ŻŻŻŻŻŻŻŻŻŻ +/* This will build Plainamp without the GUI. Todo: Fix this, I disabled a little to much + to make the regular GUI version build */ +#define NOGUI +//////////////////////////////// + + +////////////////////////////////////////////////////////////////////////////////////////// +// Include +// ŻŻŻŻŻŻŻŻŻŻ +#include "../../../../Source/Core/Common/Src/Log.h" // Global common +///////////////////////// + + +////////////////////////////////////////////////////////////////////////////////////////// +// Declarations and definitions +// ŻŻŻŻŻŻŻŻŻŻ + +// ======================================================================================= +// Because there are undefined in 64 bit it's easy to redefine them in case we use 64 bit +// --------------------- +#ifdef _M_X64 + #define GWL_WNDPROC (-4) + #define GetWindowLong GetWindowLongPtrA // or GetWindowLongPtr +#endif +// ======================================================================================= + +//////////////////////////////////// + + +// ======================================================================================= +// Back to Plainamp code +// --------------------- +#ifndef PA_GLOBAL_H +#define PA_GLOBAL_H + + +// #include "ide_devcpp/Plainamp_Private.h" + + +#ifdef UNICODE +# define PA_UNICODE +#else +# ifdef _UNICODE +# define PA_UNICODE +# endif +#endif + + +// For GetLongPathName +#if _WIN32_WINDOWS < 0x0410 +# undef _WIN32_WINDOWS +# define _WIN32_WINDOWS 0x0410 +#endif + + +#define WIN32_LEAN_AND_MEAN + +/* +#ifndef WINVER +# define WINVER 0x0500 +#else +# if (WINVER < 0x0500) +# undef WINVER +# define WINVER 0x0500 +# endif +#endif +*/ + +#include +#include +#include + + + +#ifndef _WIN32_IE +# define _WIN32_IE 0x0400 +#else +# if (_WIN32_IE < 0x0400) +# undef _WIN32_IE +# define _WIN32_IE 0x0400 +# endif +#endif + +#include + + + + + +extern HINSTANCE g_hInstance; + +extern TCHAR * szHomeDir; +extern int iHomeDirLen; + +extern TCHAR * szPluginDir; +extern int iPluginDirLen; + + + +/* +inline int abs( int x ) +{ + return ( x < 0 ) ? -x : x; +} +*/ + +inline int MIN( int a, int b ) +{ + return ( a < b ) ? a : b; +} + +inline int MAX( int a, int b ) +{ + return ( a > b ) ? a : b; +} + + + + +// Typo help +#define UNIT UINT +#define UINT_PRT UINT_PTR + + + +struct TextCompare +{ + bool operator()( const TCHAR * a, const TCHAR * b ) const + { + return _tcscmp( a, b ) < 0; + } +}; + + +#endif // PA_GLOBAL_H + diff --git a/Externals/MusicMod/Player/Src/GlobalVersion.h b/Externals/MusicMod/Player/Src/GlobalVersion.h new file mode 100644 index 0000000000..9b38065f34 --- /dev/null +++ b/Externals/MusicMod/Player/Src/GlobalVersion.h @@ -0,0 +1,28 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_GLOBAL_VERSION_H +#define PA_GLOBAL_VERSION_H + + + +#define FILE_DESCRIPTION "Plainamp" +#define VER_STRING "0.2.3.1" + + +#define PLAINAMP_TITLE TEXT( FILE_DESCRIPTION ) +#define PLAINAMP_VERSION TEXT( VER_STRING ) +#define PLAINAMP_LONG_TITLE PLAINAMP_TITLE TEXT( " " ) PLAINAMP_VERSION + + + +#endif // PA_GLOBAL_VERSION_H diff --git a/Externals/MusicMod/Player/Src/Input.cpp b/Externals/MusicMod/Player/Src/Input.cpp new file mode 100644 index 0000000000..9054b3d86f --- /dev/null +++ b/Externals/MusicMod/Player/Src/Input.cpp @@ -0,0 +1,483 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Input.h" +#include "Console.h" +#include "Status.h" +#include "Playback.h" +#include "InputPlugin.h" +#include "VisPlugin.h" +#include "DspPlugin.h" +#include "VisCache.h" +#include "Output.h" + +/* +#define FIXED_POINT 16 +#include "kiss_fft/kiss_fftr.h" +*/ + +// ======================================================================================= +//#include "fftw3/fftw3.h" +// ======================================================================================= +#include + + +// #define SKIPPY +#ifndef SKIPPY +# define MAX_DATA_FPS ( 1000 / 12 ) // in_mp3 gives new vis data every 13 ms, so 12 leaves a little space +#else +# define MAX_DATA_FPS ( 1000 / 24 ) // will skip every second frame +#endif + + +#ifdef SKIPPY +bool bLastSkipped = true; +#endif + +int iSpecChannels = 2; +int iWaveChannels = 2; + + +/* +kiss_fft_cfg kiss = { 0 }; +bool bKissInitDone = false; +*/ + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int dsp_isactive() +{ +//////////////////////////////////////////////////////////////////////////////// + DspLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + int res = ( active_dsp_count > 0 ); + +//////////////////////////////////////////////////////////////////////////////// + DspLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + return res; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int dsp_dosamples( short int * samples, int numsamples, int bps, int nch, int srate ) +{ + int num = numsamples; + +//////////////////////////////////////////////////////////////////////////////// + DspLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + for( int i = 0; i < active_dsp_count; i++ ) + { + // Process + num = active_dsp_mods[ i ]->mod->ModifySamples( + active_dsp_mods[ i ]->mod, + samples, + numsamples, + bps, + nch, + srate + ); + } + +//////////////////////////////////////////////////////////////////////////////// + DspLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + return num; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void SAVSAInit( int maxlatency_in_ms, int srate ) +{ +//////////////////////////////////////////////////////////////////////////////// + VisCache::Create(); + VisCache::EnsureLatency( maxlatency_in_ms ); + VisCache::EnsureDataFps( MAX_DATA_FPS ); + VisCache::Clean(); +//////////////////////////////////////////////////////////////////////////////// +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void SAVSADeInit() +{ + // TODO + // Console::Append( TEXT( "SAVSADeInit" ) ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void SAAddPCMData( void * PCMData, int nch, int bps, int timestamp ) +{ + // TODO + // Console::Append( TEXT( "SAAddPCMData" ) ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int SAGetMode() +{ + // TODO + // Console::Append( TEXT( "SAGetMode" ) ); + return 0; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void SAAdd( void * data, int timestamp, int csa ) +{ + // TODO + // Console::Append( TEXT( "SAAdd" ) ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VSAAddPCMData( void * PCMData, int nch, int bps, int timestamp ) +{ + // TODO + // Console::Append( TEXT( "VSAAddPCMData" ) ); + + + #ifdef SKIPPY + if( bLastSkipped ) + bLastSkipped = false; + else + { + // Skip + bLastSkipped = true; + return; + } + #endif + + +//////////////////////////////////////////////////////////////////////////////// + bool bVisLockLeft = false; + VisLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + if( active_vis_count ) + { +//////////////////////////////////////////////////////////////////////////////// + VisLock.Leave(); + bVisLockLeft = true; +//////////////////////////////////////////////////////////////////////////////// + + VisCache::NextFrame(); + VisCache::SetWriteTime( timestamp ); + VisCache::SetReadTime( active_input_plugin->plugin->GetOutputTime() ); + + short * source = ( short * )PCMData; + + // Waveform + static unsigned char wave_left[ 576 ]; + static unsigned char wave_right[ 576 ]; + if( nch < 2 ) + { + // Mono + for( int i = 0; i < 576; i++ ) + { + wave_left[ i ] = ( source[ i ] >> 8 ); + } + + VisCache::PutWaveLeft( wave_left ); + } + else + { + int i; + + // Stereo or more + for( i = 0; i < 576; i++ ) + { + wave_left [ i ] = ( source[ i * nch ] >> 8 ); + wave_right[ i ] = ( source[ i * nch + 1 ] >> 8 ); + } + + VisCache::PutWaveLeft( wave_left ); + VisCache::PutWaveRight( wave_right ); + } + + + // TODO: Much to optimize! + + // ======================================================================================= + /* + // Spectrum + static unsigned char spec_left[ 576 ]; + static unsigned char spec_right[ 576 ]; + static fftw_complex * in = NULL; + static fftw_complex * out = NULL; + + if( !in ) + { + in = ( fftw_complex * )fftw_malloc( 576 * sizeof( fftw_complex ) ); + out = ( fftw_complex * )fftw_malloc( 576 * sizeof( fftw_complex ) ); + } + + static const double factor = 1.0 / 65536.0 / sqrt( 2.0 ); + + + // Put left + int index = 0; + for( int i = 0; i < 576; i++ ) + { + in[ i ][ 0 ] = source[ index += nch ]; + in[ i ][ 1 ] = 0.0; + } + + + // Init FFT + fftw_plan p = fftw_plan_dft_1d( 576, in, out, FFTW_FORWARD, FFTW_ESTIMATE ); + + + // Run left + fftw_execute( p ); + + // Get left + for( int i = 0; i < 576; i++ ) + { + if( i & 1 ) + { + spec_left [ i ] = spec_left [ i - 1 ]; + continue; + } + const double re = out[ i >> 1 ][ 0 ]; + const double im = out[ i >> 1 ][ 1 ]; + const double root = sqrt( re*re + im*im ); + const double final = 160.0 * log10( 1.0 + root * factor ); + spec_left[ i ] = ( final < 255.0 ) ? ( unsigned char )final : 255; + } + VisCache::PutSpecLeft( spec_left ); + + + if( nch > 1 ) + { + // Put right + index = 1; + for( int i = 0; i < 576; i++ ) + { + in[ i ][ 0 ] = source[ index += nch ]; + } + + // Run right + fftw_execute( p ); + + // Get right + for( int i = 0; i < 576; i++ ) + { + if( i & 1 ) + { + spec_right[ i ] = spec_right[ i - 1 ]; + continue; + } + const double re = out[ i >> 1 ][ 0 ]; + const double im = out[ i >> 1 ][ 1 ]; + const double root = sqrt( re*re + im*im ); + const double final = 160.0 * log10( 1.0 + root * factor ); + spec_right[ i ] = ( final < 255.0 ) ? ( unsigned char )final : 255; + } + VisCache::PutSpecRight( spec_right ); + } + // ======================================================================================= + */ + + // Cleanup FFT + // ======================================================================================= + //fftw_destroy_plan( p ); + // ======================================================================================= + + // fftw_free(in); + // fftw_free(out); + } + +//////////////////////////////////////////////////////////////////////////////// + if( bVisLockLeft ) VisLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + for( int i = 0; i < active_vis_count; i++ ) + { + active_vis_mods[ i ]->bAllowRender = true; + } + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int VSAGetMode( int * specNch, int * waveNch ) +{ + iSpecChannels = 0; + iWaveChannels = 0; + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + for( int i = 0; i < active_vis_count; i++ ) + { + const int iSpec = active_vis_mods[ i ]->mod->spectrumNch; + const int iWave = active_vis_mods[ i ]->mod->waveformNch; + + if( iSpec > iSpecChannels ) iSpecChannels = iSpec; + if( iWave > iWaveChannels ) iWaveChannels = iWave; + } + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + *specNch = iSpecChannels; + *waveNch = iWaveChannels; + + return 1; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VSAAdd( void * data, int timestamp ) +{ + #ifdef SKIPPY + if( bLastSkipped ) + bLastSkipped = false; + else + { + // Skip + bLastSkipped = true; + return; + } + #endif + + +//////////////////////////////////////////////////////////////////////////////// + bool bVisLockLeft = false; + VisLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + if( active_vis_count ) + { +//////////////////////////////////////////////////////////////////////////////// + VisLock.Leave(); + bVisLockLeft = true; +//////////////////////////////////////////////////////////////////////////////// + + VisCache::NextFrame(); + VisCache::SetWriteTime( timestamp ); + VisCache::SetReadTime( active_input_plugin->plugin->GetOutputTime() ); + + unsigned char * source = ( unsigned char * )data; + if( iSpecChannels > 0 ) + { + VisCache::PutSpecLeft( source ); + source += 576; + } + if( iSpecChannels > 1 ) + { + VisCache::PutSpecRight( source ); + source += 576; + } + if( iWaveChannels > 0 ) + { + VisCache::PutWaveLeft( source ); + source += 576; + } + if( iWaveChannels > 1 ) + { + VisCache::PutWaveRight( source ); + } + } + +//////////////////////////////////////////////////////////////////////////////// + if( bVisLockLeft ) VisLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + for( int i = 0; i < active_vis_count; i++ ) + { + active_vis_mods[ i ]->bAllowRender = true; + } + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VSASetInfo( int nch, int srate ) +{ + // TODO + +//////////////////////////////////////////////////////////////////////////////// + VisCache::Create(); + VisCache::EnsureDataFps( MAX_DATA_FPS ); + VisCache::Clean(); +//////////////////////////////////////////////////////////////////////////////// +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void SetInfo( int bitrate, int srate, int stereo, int synched ) +{ + // TODO + static int last_valid_srate = 0; + + if( bitrate < 0 ) return; + + if( srate < 0 ) + srate = last_valid_srate; + else + last_valid_srate = srate; + + TCHAR szBuffer[ 5000 ] = TEXT( "" ); + _stprintf( szBuffer, TEXT( " %i kbps, %i kHz" ), bitrate, srate ); + StatusUpdate( szBuffer ); +} diff --git a/Externals/MusicMod/Player/Src/Input.h b/Externals/MusicMod/Player/Src/Input.h new file mode 100644 index 0000000000..f139df901f --- /dev/null +++ b/Externals/MusicMod/Player/Src/Input.h @@ -0,0 +1,28 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Global.h" + + + +int dsp_isactive(); +int dsp_dosamples( short int * samples, int numsamples, int bps, int nch, int srate ); +void SAVSAInit( int maxlatency_in_ms, int srate ); +void SAVSADeInit(); +void SAAddPCMData( void * PCMData, int nch, int bps, int timestamp ); +int SAGetMode(); +void SAAdd(void * data, int timestamp, int csa ); +void VSAAddPCMData( void * PCMData, int nch, int bps, int timestamp ); +int VSAGetMode( int * specNch, int * waveNch ); +void VSAAdd( void * data, int timestamp ); +void VSASetInfo( int nch, int srate ); +void SetInfo( int bitrate, int srate, int stereo, int synched ); diff --git a/Externals/MusicMod/Player/Src/InputPlugin.cpp b/Externals/MusicMod/Player/Src/InputPlugin.cpp new file mode 100644 index 0000000000..4fe945773c --- /dev/null +++ b/Externals/MusicMod/Player/Src/InputPlugin.cpp @@ -0,0 +1,389 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + +// ======================================================================================= +#include "Global.h" +// ======================================================================================= + +#include "InputPlugin.h" +#include "Console.h" +#include "Main.h" +#include "Input.h" +#include "Unicode.h" + + + +map ext_map; // extern +vector input_plugins; // extern +InputPlugin * active_input_plugin = NULL; // extern + + + +// ======================================================================================= +// The InputPlugin class is inherited from the Plugin class +InputPlugin::InputPlugin( TCHAR * szDllpath, bool bKeepLoaded ) : Plugin( szDllpath ) +{ + iHookerIndex = -1; + + szFilters = NULL; + iFiltersLen = 0; + plugin = NULL; + + //INFO_LOG(AUDIO,"\InputPlugin::InputPlugin > Begin\n"); + + if( !Load() ) + { + return; + } + + if( !bKeepLoaded ) + { + Unload(); + } + + input_plugins.push_back( this ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +InputPlugin::~InputPlugin() +{ + if( szFilters ) delete [] szFilters; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +bool InputPlugin::Load() +{ + printf("InputPlugin::Load() was called. IsLoaded: %i\n", IsLoaded()); + + if( IsLoaded() ) return true; + + + // (1) Load DLL + hDLL = LoadLibrary( GetFullpath() ); + if( !hDLL ) return false; + + // (2) Find export + WINAMP_INPUT_GETTER winampGetInModule2 = + ( WINAMP_INPUT_GETTER )GetProcAddress( hDLL, "winampGetInModule2" ); + if( winampGetInModule2 == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (3) Get module + plugin = winampGetInModule2(); + if( plugin == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (4) Process module + plugin->hDllInstance = hDLL; + plugin->hMainWindow = WindowMain; + + plugin->SAVSAInit = SAVSAInit; + plugin->SAVSADeInit = SAVSADeInit; + plugin->SAAddPCMData = SAAddPCMData; + plugin->SAGetMode = SAGetMode; + plugin->SAAdd = SAAdd; + plugin->VSAAddPCMData = VSAAddPCMData; + plugin->VSAGetMode = VSAGetMode; + plugin->VSAAdd = VSAAdd; + plugin->VSASetInfo = VSASetInfo; + + plugin->dsp_dosamples = dsp_dosamples; + plugin->dsp_isactive = dsp_isactive; + + plugin->SetInfo = SetInfo; + + + if( !szName ) + { + // Note: The prefix is not removed to hide their + // origin at Nullsoft! It just reads easier. + if( !strnicmp( plugin->description, "nullsoft ", 9 ) ) + { + plugin->description += 9; + if( !strnicmp( plugin->description, "mpeg(layer1-3/ct aac+/dolby aac) ", 33 ) ) + { + plugin->description += ( 33 - 5 ); + memcpy( plugin->description, "MPEG", 4 * sizeof( char ) ); + } + } + iNameLen = ( int )strlen( plugin->description ); + szName = new TCHAR[ iNameLen + 1 ]; + ToTchar( szName, plugin->description, iNameLen ); + szName[ iNameLen ] = TEXT( '\0' ); + } + + + + // (5) Init + const WNDPROC WndprocBefore = ( WNDPROC )GetWindowLong( WindowMain, GWL_WNDPROC ); + plugin->Init(); + const WNDPROC WndprocAfter = ( WNDPROC )GetWindowLong( WindowMain, GWL_WNDPROC ); + + if( WndprocBefore != WndprocAfter ) + { + WndprocBackup = WndprocBefore; + iHookerIndex = iWndprocHookCounter++; + } + + +#ifdef NOGUI + printf( "Loading <%s>, %s\n" , GetFilename(), szName ); +#else + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Loading <%s>, %s" ), GetFilename(), szName ); + Console::Append( szBuffer ); +#endif + + Integrate(); // This function is just below + + + // Note: Plugins that use a wndproc hook need + // to be unloaded in the inverse loading order. + // This is due to the nature of wndproc hooking. + if( iHookerIndex != -1 ) + { + Console::Append( TEXT( "Wndproc hook added (by plugin)" ) ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +bool InputPlugin::Integrate() +{ + if( !IsLoaded() ) return false; + + // (6) Build filter + + // (6a) First pass: get needed buffer length + int needed = 0; + int len = 0; + bool even = false; + char * walk = plugin->FileExtensions; + while( ( len = ( int )strlen( walk ) ) > 0 ) + { + len++; // For '\0' + if( even ) + { + // Extensions e.g. "mp3;mp2;mp1" + // Worst case "a;b;c" (5) --> "*.a;*.b;*.c" (11) + needed += ( 3 * len ); + } + else + { + // Filter name e.g. "MPEG audio files" + needed += len; + } + even = !even; + walk += len; + } + szFilters = new TCHAR[ needed + 1 ]; + TCHAR * walk_out = szFilters; + + // (6b) Once again with copy + walk = plugin->FileExtensions; + even = false; + while( true ) + { + // Check extensions + char * start_filter = walk; + const int len_filter = ( int )strlen( walk ); + if( len_filter == 0 ) + { + // End reached + break; + } + walk += len_filter + 1; + + // Check filter name + char * start_display = walk; + int len_display = ( int )strlen( walk ); + if( len_display == 0 ) + { + break; + } + walk += ++len_display; + + // Append filter name + ToTchar( walk_out, start_display, len_display ); + + // ======================================================================================= + // Print used filetypes + TCHAR szBuffer[ 5000 ]; + *(walk_out + len_display) = TEXT( '\0' ); + _stprintf( szBuffer, TEXT( " %s" ), walk_out ); + Console::Append( szBuffer ); + //printf( szBuffer, TEXT( " %s\n" ), walk_out ); + // ======================================================================================= + walk_out += len_display; + + // Convert and append extensions + char * walk_filter = start_filter; + char * last_filter = start_filter; + int len; + while( true ) + { + if( ( *walk_filter == ';' ) || ( *walk_filter == '\0' ) ) + { + len = ( walk_filter - last_filter ); + if( len < 1 ) + { + break; + } + + // Add extension to map + TCHAR * szExt = new TCHAR[ len + 1 ]; + ToTchar( szExt, last_filter, len ); + szExt[ len ] = TEXT( '\0' ); + _tcslwr( szExt ); + + ext_map.insert( pair( szExt, this ) ); + + // Append extension as "*.ext[;\0]" + len++; // Also copy ';' and '\0' + memcpy( walk_out, TEXT( "*." ), 2 * sizeof( TCHAR ) ); + walk_out += 2; + ToTchar( walk_out, last_filter, len ); + walk_out += len; + + // Any more extensions? + if( *walk_filter == '\0' ) + { + break; + } + last_filter = walk_filter + 1; + } + walk_filter++; + } + + if( *walk == '\0' ) + { + *walk_out = TEXT( '\0' ); + iFiltersLen = walk_out - szFilters; + + break; + } + } + + Console::Append( TEXT( " " ) ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +bool InputPlugin::DisIntegrate() +{ + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +bool InputPlugin::Unload() +{ + if( !IsLoaded() ) return true; + + DisIntegrate(); + + #ifndef NOGUI + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Unloading '%s'" ), GetFilename() ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + #else + printf( ">>> Unloading '%s'\n" , GetFilename() ); + #endif + + // Quit + if( plugin ) + { + if( plugin->Quit ) plugin->Quit(); + plugin->outMod = NULL; + plugin = NULL; + } + + // Remove wndproc hook + if( ( iHookerIndex != -1 ) && ( iHookerIndex == iWndprocHookCounter - 1 ) ) + { + // If we don't restore it the plugins wndproc will + // still be called which is not there anymore -> crash + SetWindowLong( WindowMain, GWL_WNDPROC, ( LONG )WndprocBackup ); + Console::Append( TEXT( "Wndproc hook removed (by host)" ) ); + Console::Append( TEXT( " " ) ); + + iHookerIndex = -1; + iWndprocHookCounter--; + } + + FreeLibrary( hDLL ); + hDLL = NULL; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +bool InputPlugin::About( HWND hParent ) +{ + if( !plugin ) return false; + if( !plugin->About ) return false; + + plugin->About( hParent ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +bool InputPlugin::Config( HWND hParent ) +{ + if( !plugin ) return false; + if( !plugin->Config ) return false; + + plugin->Config( hParent ); + + // TODO: File extension could have changed (e.g. in_mp3) + // So we have to process ext_map here... + + return true; +} diff --git a/Externals/MusicMod/Player/Src/InputPlugin.h b/Externals/MusicMod/Player/Src/InputPlugin.h new file mode 100644 index 0000000000..2702fbe767 --- /dev/null +++ b/Externals/MusicMod/Player/Src/InputPlugin.h @@ -0,0 +1,96 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_INPUT_PLUGIN_H +#define PA_INPUT_PLUGIN_H + + + +#include "Global.h" +#include "Plugin.h" +#include "Playback.h" +#include "Playlist.h" +#include "Winamp/In2.h" +#include +#include + +using namespace std; + + + +typedef In_Module * ( * WINAMP_INPUT_GETTER )( void ); + + + +class InputPlugin; + +extern map ext_map; +extern vector input_plugins; +extern InputPlugin * active_input_plugin; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp input plugin wrapper +//////////////////////////////////////////////////////////////////////////////// +class InputPlugin : public Plugin +{ +public: + InputPlugin( TCHAR * szDllpath, bool bKeepLoaded ); + ~InputPlugin(); + + bool Load(); + bool Unload(); + + TCHAR * GetTypeString() { return TEXT( "Input" ); } + int GetTypeStringLen() { return 5; } + PluginType GetType() { return PLUGIN_TYPE_INPUT; } + + inline bool IsActive() { return false; } + + bool About( HWND hParent ); + bool Config( HWND hParent ); + + In_Module * plugin; // I moved this from private to public + +private: + TCHAR * szFilters; + int iFiltersLen; + + + bool Integrate(); + bool DisIntegrate(); + + + // TODO + friend bool OpenPlay( TCHAR * szFilename, int iNumber ); + friend bool Playback_PrevOrNext( bool bPrevOrNext ); + friend bool Playback::Play(); + friend bool Playback::Pause(); + friend bool Playback::Stop(); + friend bool Playback::UpdateSeek(); // this one calls some plugin-> members + friend int Playback::PercentToMs( float fPercent ); + friend bool Playback::SeekPercent( float fPercent ); + friend bool SeekRelative( int ms ); + friend void Playback_Volume_Set( int iVol ); + friend bool Playback::Pan::Set( int iPan ); + friend void Playback_Eq_Set( int iPresetIndex ); + + friend void AddFiles(); + friend void VSAAdd( void * data, int timestamp ); + friend void VSAAddPCMData( void * PCMData, int nch, int bps, int timestamp ); + friend int Playlist::GetTitle( int iIndex, char * szAnsiTitle, int iChars ); +}; + + + +#endif // PA_INPUT_PLUGIN_H diff --git a/Externals/MusicMod/Player/Src/Lock.cpp b/Externals/MusicMod/Player/Src/Lock.cpp new file mode 100644 index 0000000000..8590b944d0 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Lock.cpp @@ -0,0 +1,16 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +// #include "Lock.h" + + +// Code moved to for inlining! diff --git a/Externals/MusicMod/Player/Src/Lock.h b/Externals/MusicMod/Player/Src/Lock.h new file mode 100644 index 0000000000..113c93f50e --- /dev/null +++ b/Externals/MusicMod/Player/Src/Lock.h @@ -0,0 +1,111 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_LOCK_H +#define PA_LOCK_H + + + +#include "Global.h" + + + +// INFO: http://www.devarticles.com/c/a/Cplusplus/Multithreading-in-C/3/ + + +// #define LOCK_USES_MUTEX + + + +//////////////////////////////////////////////////////////////////////////////// +/// Lock for thread synchronization +//////////////////////////////////////////////////////////////////////////////// +class Lock +{ +public: + Lock( TCHAR * szName ); + ~Lock(); + + void Enter(); + void Leave(); + +private: +#ifndef LOCK_USES_MUTEX + CRITICAL_SECTION * m_pCrit; +#else + HANDLE hLock; +#endif +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Creates a new named lock. +/// Note: Don't use the same name for several locks +//////////////////////////////////////////////////////////////////////////////// +inline Lock::Lock( TCHAR * szName ) +{ +#ifndef LOCK_USES_MUTEX + m_pCrit = new CRITICAL_SECTION; + InitializeCriticalSection( m_pCrit ); +#else + hLock = CreateMutex( NULL, TRUE, szName ); + ReleaseMutex( hLock ); +#endif +} + + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +inline Lock::~Lock() +{ +#ifndef LOCK_USES_MUTEX + DeleteCriticalSection( m_pCrit ); + delete [] m_pCrit; +#else + CloseHandle( hLock ); +#endif +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Closes lock. +//////////////////////////////////////////////////////////////////////////////// +inline void Lock::Enter() +{ +#ifndef LOCK_USES_MUTEX + EnterCriticalSection( m_pCrit ); +#else + WaitForSingleObject( hLock, INFINITE ); +#endif +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Opens lock. +//////////////////////////////////////////////////////////////////////////////// +inline void Lock::Leave() +{ +#ifndef LOCK_USES_MUTEX + LeaveCriticalSection( m_pCrit ); +#else + ReleaseMutex( hLock ); +#endif +} + + + +#endif // PA_LOCK_H diff --git a/Externals/MusicMod/Player/Src/Main.cpp b/Externals/MusicMod/Player/Src/Main.cpp new file mode 100644 index 0000000000..61fd7428f4 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Main.cpp @@ -0,0 +1,816 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +////////////////////////////////////////////////////////////////////////////////////////// +// Documentation +// ŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻ + +/* ---- The Seekbar ---- + Rebar.cpp handle the progress (playback position) bar. WndprocMain() is called every + second during playback. And several times a second when the cursor is moving over the + main window. If WM_TIMER called WndprocMain it calls Playback::UpdateSeek(); */ + + +/////////////////////////////////// + + +#include "Main.h" +#include "GlobalVersion.h" +#include "Playlist.h" +#include "Console.h" +#include "Status.h" +#include "Rebar.h" +#include "Playback.h" +#include "PluginManager.h" +#include "DspModule.h" +#include "VisModule.h" +#include "InputPlugin.h" +#include "OutputPlugin.h" +#include "InputPlugin.h" +#include "AddDirectory.h" +#include "AddFiles.h" +#include "Winamp.h" +#include "Winamp/wa_ipc.h" +#include "Config.h" +#include + +#define CLASSNAME_MAIN TEXT( "Winamp v1.x" ) +#define MAIN_TITLE PLAINAMP_LONG_TITLE + +#define MAIN_WIDTH 731 +#define MAIN_HEIGHT 562 + + + +HWND WindowMain = NULL; // extern +HMENU main_context_menu = NULL; // extern +HMENU play_context_menu = NULL; +HMENU opts_context_menu = NULL; +HMENU playback_context_menu = NULL; + + + +LRESULT CALLBACK WndprocMain( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + + +WINDOWPLACEMENT WinPlaceMain; + +void WinPlaceMainCallback( ConfVar * var ) +{ + if( !IsWindow( WindowMain ) ) return; + GetWindowPlacement( WindowMain, &WinPlaceMain ); +} + +const int cxScreen = GetSystemMetrics( SM_CXFULLSCREEN ); +const int cyScreen = GetSystemMetrics( SM_CYFULLSCREEN ); + +RECT rMainDefault = { + ( cxScreen - MAIN_WIDTH ) / 2, + ( cyScreen - MAIN_HEIGHT ) / 2, + ( cxScreen - MAIN_WIDTH ) / 2 + MAIN_WIDTH, + ( cyScreen - MAIN_HEIGHT ) / 2 + MAIN_HEIGHT +}; + +ConfWinPlaceCallback cwpcWinPlaceMain( + &WinPlaceMain, + TEXT( "WinPlaceMain" ), + &rMainDefault, + WinPlaceMainCallback +); + + +bool bMinimizeToTray; +ConfBool cbMinimizeToTray( &bMinimizeToTray, TEXT( "MinimizeToTray" ), CONF_MODE_PUBLIC, true ); + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool BuildMainWindow() +{ + + //#ifndef NOGUI + + // ======================================================================================= + /* Disabling this window creation cause continuous "Error setting DirectSound cooperative level" + messages for some reason. So I leave it here for now. */ + + // Register class + WNDCLASS wc = { + 0, // UINT style + WndprocMain, // WNDPROC lpfnWndProc + 0, // int cbClsExtra + 0, // int cbWndExtra + g_hInstance, // HINSTANCE hInstance + LoadIcon( g_hInstance, TEXT( "IDI_ICON1" ) ), // HICON hIcon + LoadCursor( NULL, IDC_ARROW ), // HCURSOR hCursor + ( HBRUSH )COLOR_WINDOW, // HBRUSH hbrBackground + NULL, // LPCTSTR lpszMenuName + CLASSNAME_MAIN // LPCTSTR lpszClassName + }; + + if( !RegisterClass( &wc ) ) return false; + + + // Create WindowMain + WindowMain = CreateWindowEx( + WS_EX_WINDOWEDGE, // DWORD dwExStyle + CLASSNAME_MAIN, // LPCTSTR lpClassName + MAIN_TITLE, // LPCTSTR lpWindowName + WS_OVERLAPPED | // DWORD dwStyle + // WS_VISIBLE | // + WS_CLIPCHILDREN | // + WS_BORDER | // + WS_SYSMENU | // + WS_THICKFRAME | // + WS_MINIMIZEBOX | // + WS_MAXIMIZEBOX, // + rMainDefault.left, // int x + rMainDefault.top, // int y + rMainDefault.right - rMainDefault.left, // int nWidth + rMainDefault.bottom - rMainDefault.top, // int nHeight + NULL, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + // ======================================================================================= + + //#endif + + // ======================================================================================= + #ifdef NOGUI + + // If this is not called a crash occurs + PlaylistView::Create(); + + // We have now done what we wanted so we skip the rest of the file + return true; + + #else + // ======================================================================================= + + + if( !WindowMain ) + { + UnregisterClass( CLASSNAME_MAIN, g_hInstance ); + return false; + } + + + + + // Build context menu + HMENU main_menu = CreateMenu(); + HMENU plainamp_menu = CreatePopupMenu(); + HMENU playback_menu = CreatePopupMenu(); + HMENU playlist_menu = CreatePopupMenu(); + HMENU windows_menu = CreatePopupMenu(); + + + // Plainamp + AppendMenu( plainamp_menu, MF_STRING, WINAMP_OPTIONS_PREFS, TEXT( "Preferences \tCtrl+P" ) ); + AppendMenu( plainamp_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + AppendMenu( plainamp_menu, MF_STRING, WINAMP_HELP_ABOUT, TEXT( "&About\tCtrl+F1" ) ); + AppendMenu( plainamp_menu, MF_STRING, WINAMP_FILE_QUIT, TEXT( "&Exit \tAlt+F4" ) ); + + // Playback + AppendMenu( playback_menu, MF_STRING, WINAMP_BUTTON1, TEXT( "Pre&vious \tZ" ) ); + AppendMenu( playback_menu, MF_STRING, WINAMP_BUTTON2, TEXT( "&Play\tX" ) ); + AppendMenu( playback_menu, MF_STRING, WINAMP_BUTTON3, TEXT( "P&ause\tC" ) ); + AppendMenu( playback_menu, MF_STRING, WINAMP_BUTTON4, TEXT( "&Stop\tV" ) ); + AppendMenu( playback_menu, MF_STRING, WINAMP_BUTTON5, TEXT( "&Next\tB" ) ); + + // Playlist + AppendMenu( playlist_menu, MF_STRING, ID_PE_OPEN, TEXT( "&Open\tCtrl+O" ) ); + AppendMenu( playlist_menu, MF_STRING, ID_PE_SAVEAS, TEXT( "&Save as\tCtrl+S" ) ); + AppendMenu( playlist_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + AppendMenu( playlist_menu, MF_STRING, WINAMP_FILE_PLAY, TEXT( "Add &files\tL" ) ); + AppendMenu( playlist_menu, MF_STRING, WINAMP_FILE_DIR, TEXT( "Add &directory\tShift+L" ) ); + AppendMenu( playlist_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + AppendMenu( playlist_menu, MF_STRING, PLAINAMP_PL_REM_SEL, TEXT( "Remove selected\tDel" ) ); + AppendMenu( playlist_menu, MF_STRING, PLAINAMP_PL_REM_CROP, TEXT( "Remove unselected \tCtrl+Del" ) ); + AppendMenu( playlist_menu, MF_STRING, ID_PE_CLEAR, TEXT( "Remove all\tCtrl+Shift+Del" ) ); + AppendMenu( playlist_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + AppendMenu( playlist_menu, MF_STRING, ID_PE_SELECTALL, TEXT( "Select &all\tCtrl+A" ) ); + AppendMenu( playlist_menu, MF_STRING, ID_PE_NONE, TEXT( "Select &zero\tCtrl+Shift+A" ) ); + AppendMenu( playlist_menu, MF_STRING, ID_PE_INVERT, TEXT( "Select &invert\tCtrl+I" ) ); + + // Windows + AppendMenu( windows_menu, MF_STRING, MENU_MAIN_WINDOWS_CONSOLE, TEXT( "&Console" ) ); + AppendMenu( windows_menu, MF_STRING, MENU_MAIN_WINDOWS_MANAGER, TEXT( "Plugin &Manager" ) ); + + // Main + AppendMenu( main_menu, MF_STRING | MF_POPUP, ( UINT_PTR )plainamp_menu, TEXT( "&Plainamp" ) ); + AppendMenu( main_menu, MF_STRING | MF_POPUP, ( UINT_PTR )playback_menu, TEXT( "Play&back" ) ); + AppendMenu( main_menu, MF_STRING | MF_POPUP, ( UINT_PTR )playlist_menu, TEXT( "Play&list" ) ); + AppendMenu( main_menu, MF_STRING | MF_POPUP, ( UINT_PTR )windows_menu, TEXT( "&Windows" ) ); + + SetMenu( WindowMain, main_menu ); + + //////////////////////////////////////////////////////////////////////////////// + + main_context_menu = CreatePopupMenu(); + AppendMenu( main_context_menu, MF_STRING, WINAMP_HELP_ABOUT, TEXT( "Plainamp" ) ); + + AppendMenu( main_context_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + + play_context_menu = CreatePopupMenu(); + AppendMenu( play_context_menu, MF_STRING, WINAMP_FILE_PLAY, TEXT( "Files \tL" ) ); + AppendMenu( play_context_menu, MF_STRING, WINAMP_FILE_DIR, TEXT( "Folder \tShift+L" ) ); + AppendMenu( main_context_menu, MF_STRING | MF_POPUP, ( UINT_PTR )play_context_menu, TEXT( "Play" ) ); + + AppendMenu( main_context_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + + AppendMenu( main_context_menu, MF_STRING | MF_DISABLED | MF_GRAYED | MF_CHECKED, WINAMP_MAIN_WINDOW, TEXT( "Main Window\tAlt+W" ) ); + AppendMenu( main_context_menu, MF_STRING | MF_DISABLED | MF_GRAYED | MF_CHECKED, WINAMP_OPTIONS_PLEDIT, TEXT( "Playlist Editor\tAlt+E" ) ); + AppendMenu( main_context_menu, MF_STRING | MF_DISABLED | MF_GRAYED, WINAMP_OPTIONS_EQ, TEXT( "Equalizer\tAlt+G" ) ); + AppendMenu( main_context_menu, MF_STRING | MF_DISABLED | MF_GRAYED, WINAMP_OPTIONS_VIDEO, TEXT( "Video\tAlt+V" ) ); + AppendMenu( main_context_menu, MF_STRING, PLAINAMP_TOGGLE_CONSOLE, TEXT( "Console" ) ); + AppendMenu( main_context_menu, MF_STRING, PLAINAMP_TOGGLE_MANAGER, TEXT( "Plugin Manager" ) ); + + + /* + AppendMenu( main_context_menu, MF_STRING | MF_DISABLED | MF_GRAYED, MENU_MAIN_CONTEXT_MANAGER, TEXT( "Plugin Manager" ) ); + AppendMenu( main_context_menu, MF_STRING | MF_DISABLED | MF_GRAYED, MENU_MAIN_CONTEXT_CONSOLE, TEXT( "Console" ) ); + */ + + AppendMenu( main_context_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + + opts_context_menu = CreatePopupMenu(); + AppendMenu( opts_context_menu, MF_STRING, WINAMP_OPTIONS_PREFS, TEXT( "Preferences \tCtrl+P" ) ); + AppendMenu( main_context_menu, MF_STRING | MF_POPUP, ( UINT_PTR )opts_context_menu, TEXT( "Options" ) ); + + playback_context_menu = CreatePopupMenu(); + AppendMenu( playback_context_menu, MF_STRING, WINAMP_BUTTON1, TEXT( "Previous \tZ" ) ); + AppendMenu( playback_context_menu, MF_STRING, WINAMP_BUTTON2, TEXT( "Play\tX" ) ); + AppendMenu( playback_context_menu, MF_STRING, WINAMP_BUTTON3, TEXT( "Pause\tC" ) ); + AppendMenu( playback_context_menu, MF_STRING, WINAMP_BUTTON4, TEXT( "Stop\tV" ) ); + AppendMenu( playback_context_menu, MF_STRING, WINAMP_BUTTON5, TEXT( "Next\tB" ) ); + AppendMenu( main_context_menu, MF_STRING | MF_POPUP, ( UINT_PTR )playback_context_menu, TEXT( "Playback" ) ); + + AppendMenu( main_context_menu, MF_SEPARATOR | MF_DISABLED | MF_GRAYED, ( UINT_PTR )-1, NULL ); + + AppendMenu( main_context_menu, MF_STRING, WINAMP_FILE_QUIT, TEXT( "Exit" ) ); + + + Toolbar::Create(); // This removes all buttons and status bars + //BuildMainStatus(); + + + // ======================================================================================= + // If this is not created a crash occurs + PlaylistView::Create(); + + Playlist::Create(); + + + SetWindowPlacement( WindowMain, &WinPlaceMain ); + + + return true; +#endif +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void About( HWND hParent ) +{ + // For info goto + // http://predef.sourceforge.net/precomp.html + + TCHAR szBuildDetails[ 1000 ] = ""; + +#ifdef __GNUC__ + _stprintf( szBuildDetails, + TEXT( "\n\n\nGNU GCC " __VERSION__ "\n" __DATE__ ) + ); +#else +# ifdef _MSC_VER + _stprintf( + szBuildDetails, + TEXT( "\n\n\nMicrosoft Visual C++ %i.%i\n" __DATE__ ), + _MSC_VER / 100 - 6, + ( _MSC_VER % 100 ) / 10 + ); +# endif +#endif + + TCHAR szBuffer[ 1000 ]; + _stprintf( + szBuffer, + PLAINAMP_LONG_TITLE TEXT( "\n" + "\n" + "Copyright İ 2005 Sebastian Pipping \n" + "\n" + "\n" + "--> http://www.hartwork.org" + "%s" + ), + szBuildDetails + ); + + MessageBox( + hParent, + szBuffer, + TEXT( "About" ), + MB_ICONINFORMATION + ); +} + +#define TRAY_MAIN_ID 13 +#define TRAY_MSG ( WM_USER + 1 ) + + + +NOTIFYICONDATA nid; + +bool AddTrayIcon( HWND hwnd ) +{ + ZeroMemory( &nid, sizeof( NOTIFYICONDATA ) ); + nid.cbSize = sizeof( NOTIFYICONDATA ); + nid.hWnd = hwnd; + nid.uID = TRAY_MAIN_ID; + nid.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP; + nid.uCallbackMessage = TRAY_MSG; + nid.hIcon = LoadIcon( g_hInstance, TEXT( "IDI_ICON1" ) ); + _tcscpy( nid.szTip, TEXT( "Plainamp" ) ); + + return ( Shell_NotifyIcon( NIM_ADD, &nid ) != 0 ); +} + +void RemoveTrayIcon() +{ + Shell_NotifyIcon( NIM_DELETE, &nid ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocMain( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + // Tool windows are hidden on minimize/re-shown on restore + static bool bConsoleTodo = false; + static bool bManagerTodo = false; + static bool bRemoveIcon = false; + + #ifdef NOGUI + //INFO_LOG(AUDIO,"DLL > Main.cpp:WndprocMain() was called. But nothing will be done. \n"); + #else + Console::Append( TEXT( "Main.cpp:WndprocMain was called" ) ); + #endif + + + switch( message ) + { + + #ifdef NOGUI + //printf(" > WndprocMain message: %i\n", message); + #else + + case WM_SETFOCUS: + // To re-"blue" + SetFocus( WindowPlaylist ); + break; + + case WM_CREATE: + // Note: [WindowMain] is not valid yet but [hwnd] is! + Console::Create(); // make the console window + PluginManager::Build(); // make the plugin window + break; + + case WM_NOTIFY: + { + NMHDR * hdr = ( NMHDR * )lp; + + switch( hdr->code ) + { + case LVN_GETDISPINFO: + { + LV_DISPINFO * lpdi = ( LV_DISPINFO * )lp; + playlist->Fill( lpdi->item ); + } + return 0; +/* + case LVN_ODCACHEHINT: + { + LPNMLVCACHEHINT lpCacheHint = (LPNMLVCACHEHINT)lParam; + / + This sample doesn't use this notification, but this is sent when the + ListView is about to ask for a range of items. On this notification, + you should load the specified items into your local cache. It is still + possible to get an LVN_GETDISPINFO for an item that has not been cached, + therefore, your application must take into account the chance of this + occurring. + / + } + return 0; + + case LVN_ODFINDITEM: + { + LPNMLVFINDITEM lpFindItem = (LPNMLVFINDITEM)lParam; + / + This sample doesn't use this notification, but this is sent when the + ListView needs a particular item. Return -1 if the item is not found. + / + } + return 0; +*/ + case NM_CUSTOMDRAW: + { + NMLVCUSTOMDRAW * custom = ( NMLVCUSTOMDRAW * )lp; + + switch( custom->nmcd.dwDrawStage ) + { + case CDDS_PREPAINT: + return CDRF_NOTIFYITEMDRAW; + + case CDDS_ITEMPREPAINT: + return CDRF_NOTIFYSUBITEMDRAW; + + case ( CDDS_SUBITEM | CDDS_ITEMPREPAINT ): + { + // This is the prepaint stage for an item. Here's where we set the + // item's text color. Our return value will tell Windows to draw the + // item itself, but it will use the new color we set here. + // We'll cycle the colors through red, green, and light blue. + + if( custom->nmcd.dwItemSpec == playlist->GetCurIndex() ) + { + custom->clrTextBk = RGB( 225, 225, 225 ); + } + else + { + if( custom->nmcd.dwItemSpec & 1 ) + custom->clrTextBk = RGB( 245, 248, 250 ); + else + custom->clrTextBk = RGB( 255, 255, 255 ); + } + + if( custom->iSubItem == 0 ) + custom->clrText = RGB( 255, 0, 0 ); + else + custom->clrText = RGB( 0, 0, 0 ); + + +/* + if ( (custom->nmcd.dwItemSpec % 3) == 0 ) + crText = RGB(255,0,0); + else if ( (custom->nmcd.dwItemSpec % 3) == 1 ) + crText = RGB(0,255,0); + else + crText = RGB(128,128,255); + + // Store the color back in the NMLVCUSTOMDRAW struct. + custom->clrText = crText; +*/ + // Tell Windows to paint the control itself. + } + /* + custom->clrText = RGB( 190, 190, 190 ); + custom->clrTextBk = RGB( 255, 0, 0 );*/ + return CDRF_DODEFAULT; + + + } + break; + + } + + /* + case RBN_CHILDSIZE: + { + NMREBARCHILDSIZE * chs = ( NMREBARCHILDSIZE * )lp; + const int width_client = chs->rcChild.right - chs->rcChild.left; + int diff = width_client - 120; + if( diff > 0 ) + { + const int width_band = chs->rcBand.right - chs->rcBand.left; + // chs->rcChild.right = chs->rcChild.left + 120; + + DEBUGF( 1000, "CHILDSIZE [%i] [%i]", chs->uBand, width_band ); + + const int client_band_diff = width_band - width_client; + chs->rcBand.right = chs->rcBand.left + 120 + client_band_diff; + // chs->uBand + + + REBARBANDINFO rbbi; + rbbi.cbSize = sizeof( REBARBANDINFO ); + rbbi.fMask = RBBIM_SIZE; + rbbi.cx = 154; //width_band + diff; + LRESULT lResult = SendMessage( + rebar, + RB_SETBANDINFO, + chs->uBand, + ( LPARAM )( REBARBANDINFO * )&rbbi + ); + + } + break; + } + */ + case RBN_HEIGHTCHANGE: + { + const int iRebarHeightBefore = iRebarHeight; + RECT r; + GetWindowRect( WindowRebar, &r ); + iRebarHeight = r.bottom - r.top; + + InvalidateRect( WindowRebar, NULL, TRUE ); + InvalidateRect( WindowPlaylist, NULL, TRUE ); + + RECT client; + GetClientRect( WindowMain, &client ); + PostMessage( + hwnd, + WM_SIZE, + SIZE_RESTORED, + ( client.right - client.left ) << 16 | + ( client.bottom - client.top ) + ); + + break; + } + } + break; + } + + case WM_SYSKEYDOWN: + switch( wp ) // [Alt]+[...] + { + case VK_UP: + case VK_DOWN: + SetFocus( WindowPlaylist ); + SendMessage( WindowPlaylist, message, wp, lp ); + break; + } + break; + + case WM_KEYDOWN: + case WM_KEYUP: + SetFocus( WindowPlaylist ); + SendMessage( WindowPlaylist, message, wp, lp ); + break; + + case WM_COMMAND: + { + const int code = HIWORD( wp ); + switch( code ) + { + case 1: // also == CBN_SELCHANGE + { + if( ( HWND )lp == WindowOrder ) + { + LRESULT res = SendMessage( WindowOrder, CB_GETCURSEL, 0, 0 ); + if( res == CB_ERR ) break; + Playback::Order::SetMode( ( int )res ); + } + else if( ( HWND )lp == WindowEq ) + { + LRESULT res = SendMessage( WindowEq, CB_GETCURSEL, 0, 0 ); + if( res == CB_ERR ) break; + Playback::Eq::SetIndex( ( int )( res - 1 ) ); + } + + return WndprocWinamp( hwnd, message, wp, lp ); + } + + case 0: + { + // Menu + const int id = LOWORD( wp ); + switch( id ) + { + case MENU_MAIN_WINDOWS_CONSOLE: + Console::Popup(); + break; + + case MENU_MAIN_WINDOWS_MANAGER: + PluginManager::Popup(); + break; + + } +/* + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "default 1 id = <%i>" ), id ); + MessageBox( 0, szBuffer, "", 0 ); +*/ + return WndprocWinamp( hwnd, message, wp, lp ); + } + + default: + return WndprocWinamp( hwnd, message, wp, lp ); + } + break; + } + + case WM_GETMINMAXINFO: + { + MINMAXINFO * mmi = ( MINMAXINFO * )lp; + mmi->ptMinTrackSize.x = 400; + mmi->ptMinTrackSize.y = 300; + return 0; + } + + case WM_SIZE: + { + // Resize children + RECT client; + GetClientRect( WindowMain, &client ); + + const int iClientWidth = client.right - client.left; + const int iClientHeight = client.bottom - client.top; + const int iPlaylistHeight = iClientHeight - iRebarHeight - iStatusHeight; + + if( WindowRebar ) + MoveWindow( WindowRebar, 0, 0, iClientWidth, iRebarHeight, TRUE ); + if( WindowPlaylist ) + { + MoveWindow( WindowPlaylist, 0, iRebarHeight, iClientWidth, iPlaylistHeight, TRUE ); + playlist->Resize( WindowMain ); + } + if( WindowStatus ) + MoveWindow( WindowStatus, 0, iRebarHeight + iPlaylistHeight, iClientWidth, iStatusHeight, TRUE ); + break; + } + + case WM_TIMER: + Playback::UpdateSeek(); + break; + + case WM_CONTEXTMENU: + PostMessage( hwnd, WM_COMMAND, WINAMP_MAINMENU, 0 ); + break; + + case WM_CLOSE: + { + // Clean shutdown + + // Stop + Playback::Stop(); + + // Dsp + DspLock.Enter(); + for( int d = active_dsp_count - 1; d >= 0; d-- ) + { + DspLock.Leave(); + active_dsp_mods[ d ]->Stop(); + DspLock.Enter(); + } + DspLock.Leave(); + + + // Vis + VisLock.Enter(); + for( int v = active_vis_count - 1; v >= 0; v-- ) + { + VisLock.Leave(); + active_vis_mods[ v ]->Stop(); + VisLock.Enter(); + } + VisLock.Leave(); + } + break; + + case WM_DESTROY: + { + // ======================================================================================= + // Save playlist + /* + TCHAR * szPlaylistMind = new TCHAR[ iHomeDirLen + 12 + 1 ]; + memcpy( szPlaylistMind, szHomeDir, iHomeDirLen * sizeof( TCHAR ) ); + memcpy( szPlaylistMind + iHomeDirLen, TEXT( "Plainamp.m3u" ), 12 * sizeof( TCHAR ) ); + szPlaylistMind[ iHomeDirLen + 12 ] = TEXT( '\0' ); + + Playlist::ExportPlaylistFile( szPlaylistMind ); + + delete [] szPlaylistMind; + */ + // ======================================================================================= + + cwpcWinPlaceMain.TriggerCallback(); + cwpcWinPlaceMain.RemoveCallback(); + + Console::Destroy(); + PluginManager::Destroy(); + + if( bRemoveIcon ) + { + RemoveTrayIcon(); + bRemoveIcon = false; + } + + PostQuitMessage( 0 ); + return 0; + } + + case WM_ACTIVATEAPP: + { + if( wp != TRUE ) break; + + // Also bring console/manager to front + const bool bConsoleVisible = ( IsWindowVisible( WindowConsole ) == TRUE ); + const bool bManagerVisible = ( IsWindowVisible( WindowManager ) == TRUE ); + const bool bMainTodo = ( bConsoleVisible || bManagerVisible ); + + if( bConsoleVisible ) BringWindowToTop( WindowConsole ); + if( bManagerVisible ) BringWindowToTop( WindowManager ); + if( bMainTodo ) BringWindowToTop( WindowMain ); + + break; + } + + case WM_SYSCOMMAND: + switch( ( wp & 0xFFF0 ) ) + { + case SC_CLOSE: + if( !SendMessage( WindowMain, WM_WA_IPC, 0, IPC_HOOK_OKTOQUIT ) ) + { + return 0; + } + break; + + case SC_MINIMIZE: + { + // Hide console/manager on minimize + bConsoleTodo = ( IsWindowVisible( WindowConsole ) == TRUE ); + if( bConsoleTodo ) ShowWindow( WindowConsole, SW_HIDE ); + + bManagerTodo = ( IsWindowVisible( WindowManager ) == TRUE ); + if( bManagerTodo ) ShowWindow( WindowManager, SW_HIDE ); + + if( bMinimizeToTray ) + { + if( !bRemoveIcon ) + { + bRemoveIcon = AddTrayIcon( hwnd ); + } + + ShowWindow( hwnd, FALSE ); + return 0; + } + + break; + } + + case SC_RESTORE: + { + const LRESULT res = DefWindowProc( hwnd, message, wp, lp ); + + // Unhide console/manager + const bool bMainTodo = ( bConsoleTodo || bManagerTodo ); + if( bConsoleTodo ) ShowWindow( WindowConsole, SW_SHOW ); + if( bManagerTodo ) ShowWindow( WindowManager, SW_SHOW ); + if( bMainTodo ) BringWindowToTop( WindowMain ); + + return res; + } + } + break; + + case TRAY_MSG: + switch( lp ) + { + case WM_RBUTTONDOWN: // TODO: context menu instead + case WM_LBUTTONDOWN: + if( IsWindowVisible( hwnd ) == FALSE ) + { + ShowWindow( hwnd, TRUE ); + } + break; + + case WM_RBUTTONUP: // TODO: context menu instead + case WM_LBUTTONUP: + if( bRemoveIcon ) + { + RemoveTrayIcon(); + bRemoveIcon = false; + } + break; + + } + return 0; + #endif + default: + return WndprocWinamp( hwnd, message, wp, lp ); + } + return DefWindowProc( hwnd, message, wp, lp ); + +} diff --git a/Externals/MusicMod/Player/Src/Main.h b/Externals/MusicMod/Player/Src/Main.h new file mode 100644 index 0000000000..c8a2b4309c --- /dev/null +++ b/Externals/MusicMod/Player/Src/Main.h @@ -0,0 +1,51 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_MAIN_H +#define PA_MAIN_H + + + +#include "Global.h" +#include "Winamp/wa_msgids.h" + + + +// TODO + + + +#define MENU_MAIN_WINDOWS_CONSOLE 41 +#define MENU_MAIN_WINDOWS_MANAGER 42 + +#define MENU_MAIN_CONTEXT_MANAGER WINAMP_MAIN_WINDOW // first window, for gen_dl +#define MENU_MAIN_CONTEXT_CONSOLE WINAMP_OPTIONS_VIDEO // last window, for gen_template + + +#define PLAINAMP_TOGGLE_CONSOLE 50001 +#define PLAINAMP_TOGGLE_MANAGER 50002 + + + + +extern HWND WindowMain; +extern HMENU main_context_menu; + + + +bool BuildMainWindow(); +void About( HWND hParent ); +LRESULT CALLBACK WndprocMain( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + + +#endif // PA_MAIN_H diff --git a/Externals/MusicMod/Player/Src/Output.cpp b/Externals/MusicMod/Player/Src/Output.cpp new file mode 100644 index 0000000000..86bc6f4a08 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Output.cpp @@ -0,0 +1,382 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Output.h" +#include "OutputPlugin.h" +#include "Console.h" +#include + + + +int iNullSampleRate; +int iNullSumBytesPerSample; +int iNullWrittenMillis; +bool bNullPlaying; +int bNullPaused; +DWORD dwNullOpenTimestamp; +DWORD dwNullPauseTimestamp; + +const int NULL_DEFAULT_LATENCY = 1000; + + +int Output_Open( int samplerate, int numchannels, int bitspersamp, int bufferlenms, int prebufferms ); +void Output_Close(); +int Output_Write( char * buf, int len ); +int Output_CanWrite(); +int Output_IsPlaying(); +int Output_Pause( int pause ); +void Output_SetVolume( int volume ); +void Output_SetPan( int pan ); +void Output_Flush( int t ); +int Output_GetOutputTime(); +int Output_GetWrittenTime(); + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +Out_Module output_server = { + OUT_VER, // int version + "Plainamp Output Server", // char * description; + 0x7fffffff, // int id + NULL, // HWND hMainWindow + NULL, // HINSTANCE hDllInstance + NULL, // void (*Config)(HWND hwndParent); + NULL, // void (*About)(HWND hwndParent); + NULL, // void (*Init)(); + NULL, // void (*Quit)(); + Output_Open, // int (*Open)(int samplerate, int numchannels, int bitspersamp, int bufferlenms, int prebufferms); + Output_Close, // void (*Close)(); + Output_Write, // int (*Write)(char *buf, int len); + Output_CanWrite, // int (*CanWrite)(); + Output_IsPlaying, // int (*IsPlaying)(); + Output_Pause, // int (*Pause)(int pause); + Output_SetVolume, // void (*SetVolume)(int volume); + Output_SetPan, // void (*SetPan)(int pan); + Output_Flush, // void (*Flush)(int t); + Output_GetOutputTime, // int (*GetOutputTime)(); + Output_GetWrittenTime, // int (*GetWrittenTime)(); +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Output_Open( int samplerate, int numchannels, int bitspersamp, int bufferlenms, int prebufferms ) +{ + if( active_output_count > 0 ) + { + // Maximum + int res = 0; + int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->Open( samplerate, numchannels, bitspersamp, bufferlenms, prebufferms ); + if( res_temp > res ) res = res_temp; + } + return res; + } + else + { + iNullSampleRate = samplerate; + iNullSumBytesPerSample = numchannels * ( bitspersamp / 8 ); + iNullWrittenMillis = 0; + bNullPlaying = false; + bNullPaused = 0; + dwNullOpenTimestamp = GetTickCount(); + + return NULL_DEFAULT_LATENCY; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void Output_Close() +{ + if( active_output_count > 0 ) + { + for( int i = 0; i < active_output_count; i++ ) + { + active_output_plugins[ i ]->plugin->Close(); + } + } + else + { + bNullPlaying = false; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Output_Write( char * buf, int len ) +{ + if( active_output_count > 0 ) + { + // Maximum + int res = 0; + int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->Write( buf, len ); + if( res_temp > res ) res = res_temp; + } + return res; + } + else + { + if( iNullWrittenMillis == 0 ) + { + dwNullOpenTimestamp = GetTickCount(); + bNullPlaying = true; + } + + iNullWrittenMillis += ( len / iNullSumBytesPerSample ) * 1000 / iNullSampleRate; + return 0; // 0 == Success + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Output_CanWrite() +{ + if( active_output_count > 0 ) + { + // Minimum + int res = 0x7fffffff; + int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->CanWrite(); + if( res_temp < res ) res = res_temp; + } + return res; + } + else + { + const int diff = Output_GetWrittenTime() - Output_GetOutputTime(); + if( diff >= NULL_DEFAULT_LATENCY ) + { + return 0; + } + else + { + return ( NULL_DEFAULT_LATENCY - diff ) * iNullSumBytesPerSample * iNullSampleRate / 1000; + } + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Output_IsPlaying() +{ + if( active_output_count > 0 ) + { + // Maximum + int res = 0; + int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->IsPlaying(); + if( res_temp > res ) res = res_temp; + } + return res; + } + else + { + return ( Output_GetOutputTime() < Output_GetWrittenTime() ) ? 1 : 0; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Output_Pause( int pause ) +{ + if( active_output_count > 0 ) + { + // Maximum + int res = 0; + int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->Pause( pause ); + if( res_temp > res ) res = res_temp; + } + return res; + } + else + { + int res = bNullPaused; + if( pause && !bNullPaused ) + { + // Playback should be paused + dwNullPauseTimestamp = GetTickCount(); + bNullPaused = 1; + } + else if( !pause && bNullPaused ) + { + // Playback should be continued + // Add the gap length to the open timestamp like no gap exists + dwNullOpenTimestamp += ( GetTickCount() - dwNullPauseTimestamp ); + bNullPaused = 0; + } + return res; // Previous state + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void Output_SetVolume( int volume ) +{ + /* There was a call here with the volume value -666 that I could not see the source of, + so I added this check to see that we got a positive volume. */ + if(volume >= 0) + { + // ======================================================================================= + // The volume goes from 0 to 255 + //TCHAR szBuffer[ 5000 ]; + //_stprintf( szBuffer, TEXT( "DLL > Output_SetVolume <%i>" ), volume ); + //Console::Append( szBuffer ); + //Console::Append( TEXT( " " ) ); + //INFO_LOG(AUDIO, "DLL > Output_SetVolume <%i>\n", volume ); + // ======================================================================================= + + for( int i = 0; i < active_output_count; i++ ) + { + active_output_plugins[ i ]->plugin->SetVolume( volume ); + } + } + +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void Output_SetPan( int pan ) +{ + for( int i = 0; i < active_output_count; i++ ) + { + active_output_plugins[ i ]->plugin->SetPan( pan ); + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void Output_Flush( int t ) +{ + if( active_output_count > 0 ) + { + for( int i = 0; i < active_output_count; i++ ) + { + active_output_plugins[ i ]->plugin->Flush( t ); + } + } + else + { + dwNullOpenTimestamp = GetTickCount() - t; + iNullWrittenMillis = t; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Returns the number of milliseconds the song would be at +/// if we had zero latency. This value is never bigger than +/// the one returned by Output_GetWrittenTime(). +//////////////////////////////////////////////////////////////////////////////// +int Output_GetOutputTime() // <= GetWrittenTime() +{ + if( active_output_count > 0 ) + { + // Minimum + int res = 0x7fffffff; + int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->GetOutputTime(); + if( res_temp < res ) res = res_temp; + } + return res; + } + else + { + if( bNullPlaying ) + { + int res; + if( bNullPaused ) + { + res = dwNullPauseTimestamp - dwNullOpenTimestamp; + } + else + { + res = GetTickCount() - dwNullOpenTimestamp; + } + return MIN( res, iNullWrittenMillis ); + } + else + { + return 0; + } + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Returns the number of milliseconds already written. +/// Due to latency this value is always bigger than +/// the value returned by Output_GetOutputTime(). +//////////////////////////////////////////////////////////////////////////////// +int Output_GetWrittenTime() +{ + if( active_output_count > 0 ) + { + // Maximum + int res = 0; + int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->GetWrittenTime(); + if( res_temp > res ) res = res_temp; + } + return res; + } + else + { + return iNullWrittenMillis; + } +} diff --git a/Externals/MusicMod/Player/Src/Output.h b/Externals/MusicMod/Player/Src/Output.h new file mode 100644 index 0000000000..21723f3e73 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Output.h @@ -0,0 +1,27 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_OUTPUT_H +#define PA_OUTPUT_H + + + +#include "Global.h" +#include "Winamp/Out.h" + + + +extern Out_Module output_server; + + + +#endif // PA_OUTPUT_H diff --git a/Externals/MusicMod/Player/Src/OutputPlugin.cpp b/Externals/MusicMod/Player/Src/OutputPlugin.cpp new file mode 100644 index 0000000000..73b01f7be5 --- /dev/null +++ b/Externals/MusicMod/Player/Src/OutputPlugin.cpp @@ -0,0 +1,316 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "OutputPlugin.h" +#include "Main.h" +#include "Unicode.h" +#include "Console.h" +#include "Config.h" +#include "Playback.h" + + + +vector output_plugins; // extern +OutputPlugin ** active_output_plugins = NULL; // extern +int active_output_count = 0; // extern + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +OutputPlugin::OutputPlugin( TCHAR * szDllpath, bool bKeepLoaded ) : Plugin( szDllpath ) +{ + iHookerIndex = -1; + + bActive = false; + iArrayIndex = -1; + plugin = NULL; + + if( !Load() ) + { + return; + } + + //////////////////////////////////////////////////////////////////////////////// + + // Quick hack!!! + TCHAR * szBuffer = new TCHAR[ 500 ]; // NOT LOCAL!!! + _stprintf( szBuffer, TEXT( "OutputPluginActive___%s" ), GetFilename() ); + ConfBool * cbActive = new ConfBool( &bActive, szBuffer, CONF_MODE_INTERNAL, false ); + cbActive->Read(); + printf("OutputPlugin > GetFilename() returned <%s>\n", szBuffer ); + + printf("OutputPlugin > We now have and \n", bActive, bKeepLoaded ); + + if( bActive ) + { + bActive = false; + Start(); + } + else + { + if( !bKeepLoaded ) + { + // Note: out_ds seems to do weird things + // when unloaded here!? + // So out_ds keeps loaded for now. + if( _tcscmp( GetFilename(), TEXT( "out_ds.dll" ) ) ) + { + printf("OutputPlugin > Unload called from OutputPlugin::OutputPlugin\n"); + Unload(); + } + } + } + // Quick hack!!! + //////////////////////////////////////////////////////////////////////////////// + + + output_plugins.push_back( this ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool OutputPlugin::Load() +{ + if( IsLoaded() ) return true; + + // (1) Load DLL + hDLL = LoadLibrary( GetFullpath() ); + if( !hDLL ) return false; + + // (2) Find export + WINAMP_OUTPUT_GETTER winampGetOutModule = + ( WINAMP_OUTPUT_GETTER )GetProcAddress( hDLL, "winampGetOutModule" ); + if( winampGetOutModule == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (3) Get module + plugin = winampGetOutModule(); + if( !plugin ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (4) Process module + plugin->hDllInstance = hDLL; + plugin->hMainWindow = WindowMain; + + if( !szName ) + { + // Note: The prefix is not removed to hide their + // origin at Nullsoft! It just reads easier. + if( !strnicmp( plugin->description, "nullsoft ", 9 ) ) + { + plugin->description += 9; + } + iNameLen = ( int )strlen( plugin->description ); + szName = new TCHAR[ iNameLen + 1 ]; + ToTchar( szName, plugin->description, iNameLen ); + szName[ iNameLen ] = TEXT( '\0' ); + } + + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Loading <%s>, %s" ), GetFilename(), szName ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + printf( ">>>Loading <%s>, %s\n" , GetFilename(), szName ); + + if( plugin->Init ) + { + // We remove the WNDPROC things +#ifdef NOGUI + plugin->Init(); +#else + + // Init + const WNDPROC WndprocBefore = ( WNDPROC )GetWindowLong( WindowMain, GWL_WNDPROC ); + plugin->Init(); + const WNDPROC WndprocAfter = ( WNDPROC )GetWindowLong( WindowMain, GWL_WNDPROC ); + + if( WndprocBefore != WndprocAfter ) + { + WndprocBackup = WndprocBefore; + iHookerIndex = iWndprocHookCounter++; + } + + + // Note: Plugins that use a wndproc hook need + // to be unloaded in the inverse loading order. + // This is due to the nature of wndproc hooking. + if( iHookerIndex != -1 ) + { + Console::Append( TEXT( "Wndproc hook added (by plugin)" ) ); + } +#endif + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool OutputPlugin::Unload() +{ + if( !IsLoaded() ) return true; + if( bActive && Playback::IsPlaying() ) return false; + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Unloading <%s>" ), GetFilename() ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + printf( ">>>Unloading <%s>\n" , GetFilename() ); + + // Quit + if( plugin ) + { + if( plugin->Quit ) plugin->Quit(); + plugin = NULL; + } + + // Remove wndproc hook + if( ( iHookerIndex != -1 ) && ( iHookerIndex == iWndprocHookCounter - 1 ) ) + { + // If we don't restore it the plugins wndproc will + // still be called which is not there anymore -> crash + SetWindowLong( WindowMain, GWL_WNDPROC, ( LONG )WndprocBackup ); + Console::Append( TEXT( "Wndproc hook removed (by host)" ) ); + Console::Append( TEXT( " " ) ); + + iHookerIndex = -1; + iWndprocHookCounter--; + } + + FreeLibrary( hDLL ); + hDLL = NULL; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool OutputPlugin::About( HWND hParent ) +{ + if( !IsLoaded() ) return false; + if( !plugin ) return false; + if( !plugin->About ) return false; + + plugin->About( hParent ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool OutputPlugin::Config( HWND hParent ) +{ + if( !IsLoaded() ) return false; + if( !plugin ) return false; + if( !plugin->Config ) return false; + + plugin->Config( hParent ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool OutputPlugin::Start() +{ + //INFO_LOG(AUDIO, "OutputPlugin::Start() > Begin \n", + // IsLoaded(), bActive, active_output_count ); + + if( !IsLoaded() ) return false; + if( bActive ) return true; + + if( active_output_count ) + { + active_output_plugins = ( OutputPlugin ** )realloc( active_output_plugins, ( active_output_count + 1 ) * sizeof( OutputPlugin * ) ); + active_output_plugins[ active_output_count ] = this; + iArrayIndex = active_output_count; + active_output_count++; + } + else + { + active_output_plugins = ( OutputPlugin ** )malloc( sizeof( OutputPlugin * ) ); + active_output_plugins[ 0 ] = this; + iArrayIndex = 0; + active_output_count = 1; + } + + #ifndef NOGUI + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Output plugin <%s> activated" ), GetFilename() ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + #else + INFO_LOG(AUDIO, "\n >>> Output plugin '%s' activated\n\n" , GetFilename() ); + #endif + + bActive = true; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool OutputPlugin::Stop() +{ + if( !IsLoaded() ) return true; + if( !bActive ) return true; + + const int iMaxIndex = active_output_count - 1; + if( iArrayIndex < iMaxIndex ) + { + active_output_plugins[ iArrayIndex ] = active_output_plugins[ iMaxIndex ]; + active_output_plugins[ iArrayIndex ]->iArrayIndex = iArrayIndex; + } + iArrayIndex = -1; + active_output_count--; + + // TODO Flush? + // TODO Close? + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Output plugin <%s> deactivated" ), GetFilename() ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + + bActive = false; + + return true; +} diff --git a/Externals/MusicMod/Player/Src/OutputPlugin.h b/Externals/MusicMod/Player/Src/OutputPlugin.h new file mode 100644 index 0000000000..6dc4ca0cb9 --- /dev/null +++ b/Externals/MusicMod/Player/Src/OutputPlugin.h @@ -0,0 +1,87 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_OUTPUT_PLUGIN_H +#define PA_OUTPUT_PLUGIN_H + + + +#include "Global.h" +#include "Plugin.h" +#include "Winamp/Out.h" +#include + +using namespace std; + + + +typedef Out_Module * ( * WINAMP_OUTPUT_GETTER )( void ); + + +class OutputPlugin; + +extern vector output_plugins; +extern OutputPlugin ** active_output_plugins; +extern int active_output_count; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp output plugin wrapper +//////////////////////////////////////////////////////////////////////////////// +class OutputPlugin : public Plugin +{ +public: + OutputPlugin( TCHAR * szDllpath, bool bKeepLoaded ); + + bool Load(); + bool Unload(); + + TCHAR * GetTypeString() { return TEXT( "Output" ); } + int GetTypeStringLen() { return 6; } + PluginType GetType() { return PLUGIN_TYPE_OUTPUT; } + + inline bool IsActive() { return bActive; } + + bool About( HWND hParent ); + bool Config( HWND hParent ); + + bool Start(); + bool Stop(); + + Out_Module * plugin; // Moved to public + +private: + bool bActive; + int iArrayIndex; + + + + // TODO + friend bool OpenPlay( TCHAR * szFilename, int iNumber ); + + friend int Output_Open( int samplerate, int numchannels, int bitspersamp, int bufferlenms, int prebufferms ); + friend void Output_Close(); + friend int Output_Write( char * buf, int len ); + friend int Output_CanWrite(); + friend int Output_IsPlaying(); + friend int Output_Pause( int pause ); + friend void Output_SetVolume( int volume ); + friend void Output_SetPan( int pan ); + friend void Output_Flush( int t ); + friend int Output_GetOutputTime(); + friend int Output_GetWrittenTime(); +}; + + + +#endif // PA_OUTPUT_PLUGIN_H diff --git a/Externals/MusicMod/Player/Src/Path.cpp b/Externals/MusicMod/Player/Src/Path.cpp new file mode 100644 index 0000000000..956489bccc --- /dev/null +++ b/Externals/MusicMod/Player/Src/Path.cpp @@ -0,0 +1,243 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Path.h" +#include +#include + +using namespace std; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Applies a virtual root to a filename +/// +/// Example: +/// Rootpath: "C:\111\222\333\" +/// Filename before: "C:\111\444\test.mp3" +/// Filename after: "..\..\444\test.mp3" +//////////////////////////////////////////////////////////////////////////////// +bool ApplyRootToFilename( TCHAR * szRootpath, TCHAR * szFilename ) +{ + // returns modified flag + + int iFilenameLen = ( int )_tcslen( szFilename ); + int iRootLen = ( int )_tcslen( szRootpath ); + + // Too short? + if( ( iRootLen < 2 ) || ( iFilenameLen < 4 ) ) + { + return false; + } + + // Ensure trailing backslash + bool bDelOnRet = false; + TCHAR * szFinalRoot; + TCHAR * szFinalRootBackup; + if( szRootpath[ iRootLen - 1 ] != TEXT( '\\' ) ) + { + szFinalRoot = new TCHAR[ iRootLen + 2 ]; + memcpy( szFinalRoot, szRootpath, iRootLen * sizeof( TCHAR ) ); + memcpy( szFinalRoot + iRootLen, TEXT( "\\\0" ), 2 * sizeof( TCHAR ) ); + iRootLen++; + szFinalRootBackup = szFinalRoot; + bDelOnRet = true; + } + else + { + szFinalRoot = szRootpath; + szFinalRootBackup = NULL; + bDelOnRet = false; + } + + + // Different drives? + if( _totlower( *szFilename ) != _totlower( *szFinalRoot ) ) + { + if( bDelOnRet ) delete [] szFinalRootBackup; + return false; + } + + + // Skip drive + if( _tcsnicmp( szFilename, szFinalRoot, 3 ) ) + { + szFinalRoot += 3; + iRootLen -= 3; + + memmove( szFilename, szFilename + 3, ( iFilenameLen - 2 ) * sizeof( TCHAR ) ); // Plus \0 + iFilenameLen -=3; + } + + + int iBackslashLast = -1; + + int iCompLen; // Maximum chars to compare + if( iRootLen > iFilenameLen ) + iCompLen = iFilenameLen; + else + iCompLen = iRootLen; + + + // Walk while equal + int i = 0; + while( i < iCompLen ) + { + if( ( szFilename[ i ] == TEXT( '\\' ) ) && ( szFinalRoot[ i ] == TEXT( '\\' ) ) ) + { + iBackslashLast = i; + i++; + } + else if( _totlower( szFilename[ i ] ) == _totlower( szFinalRoot[ i ] ) ) + { + i++; + } + else + { + break; + } + } + + + // Does the filename contain the full root? + int iLevelDiff = 0; + if( i != iCompLen ) + { + // Calculate level difference + for( i = iBackslashLast + 1; i < iRootLen; i++ ) + { + if( szFinalRoot[ i ] == TEXT( '\\' ) ) + { + iLevelDiff++; + } + } + } + + + if( iBackslashLast == -1 ) + { + if( bDelOnRet ) delete [] szFinalRootBackup; + return false; + } + + + TCHAR * szSource = szFilename + iBackslashLast + 1; + if( iLevelDiff > 0 ) + { + const int iExtraCharsForPrefix = ( 3 * iLevelDiff ) - iBackslashLast - 1; + const int iCharsToMove = iFilenameLen - iBackslashLast; // One more for '\0' + memmove( szSource + iExtraCharsForPrefix, szSource, sizeof( TCHAR ) * iCharsToMove ); + + TCHAR * szWalk = szFilename; + while( iLevelDiff-- ) + { + memcpy( szWalk, TEXT( "..\\" ), 3 * sizeof( TCHAR ) ); + szWalk += 3; + } + } + else + { + const int iCharsToMove = iFilenameLen - iBackslashLast; // One more for '\0' + memmove( szFilename, szSource, sizeof( TCHAR ) * iCharsToMove ); + } + + + if( bDelOnRet ) delete [] szFinalRootBackup; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Compresses filenames (inplace) +/// +/// Example: +/// Before "C:\111\222\..\333\.\444\..\..\test.mp3" +/// After "C:\111\test.mp3" +//////////////////////////////////////////////////////////////////////////////// +bool UnbloatFilename( TCHAR * szFullpath, bool bFixTooDeep ) +{ + int iLen = ( int )_tcslen( szFullpath ); + bool bModified = false; + + // Exclude drive letter from conversion "C:\" + if( ( iLen > 3 ) && !_tcsnicmp( szFullpath + 1, TEXT( ":\\" ), 2 ) ) + { + szFullpath += 3; + iLen -= 3; + } + + vector< TCHAR * > after_backslash; + TCHAR * end = szFullpath + iLen; + TCHAR * szWalk = szFullpath; + + while( true ) + { + if( !_tcsnicmp( szWalk, TEXT( "..\\" ), 3 ) ) + { + TCHAR * szAfterBackslashLast; + + if( after_backslash.empty() ) + { + // Getting here means we go deeper than root level e.g. "../test" + if( bFixTooDeep ) + { + szAfterBackslashLast = szWalk; + } + else + { + break; + } + } + else + { + szAfterBackslashLast = after_backslash.back(); + after_backslash.pop_back(); + } + + const int iBytesToCopy = end - szWalk - ( 3 * sizeof( TCHAR ) ); + const int iBytesLess = szWalk + ( 3 * sizeof( TCHAR ) ) - szAfterBackslashLast; + + memmove( szAfterBackslashLast, szWalk + 3, iBytesToCopy ); + + char * byte_end = ( char * )end; + byte_end -= iBytesLess; + end = byte_end; + *end = TEXT( '\0' ); + + szWalk = szAfterBackslashLast; + + bModified = true; + } + else if( !_tcsnicmp( szWalk, TEXT( ".\\" ), 2 ) ) + { + const int iBytesToCopy = end - szWalk - ( 2 * sizeof( TCHAR ) ); + memmove( szWalk, szWalk + 2, iBytesToCopy ); + end -= 2; + *end = TEXT( '\0' ); + + bModified = true; + } + else + { + + if( szWalk >= end ) break; + after_backslash.push_back( szWalk ); + + // Jump after next backslash + while( ( szWalk < end ) && ( *szWalk != TEXT( '\\' ) ) ) szWalk++; + szWalk++; + } + } + + return bModified; +} diff --git a/Externals/MusicMod/Player/Src/Path.h b/Externals/MusicMod/Player/Src/Path.h new file mode 100644 index 0000000000..63b510dd48 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Path.h @@ -0,0 +1,31 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_PATH_H +#define PA_PATH_H + + + +#ifndef PLAINAMP_TESTING +# include "Global.h" +#else +# include "../Testing/GlobalTest.h" +#endif + + + +bool ApplyRootToFilename( TCHAR * szRootpath, TCHAR * szFilename ); +bool UnbloatFilename( TCHAR * szFullpath, bool bFixTooDeep ); + + + +#endif // PA_PATH_H diff --git a/Externals/MusicMod/Player/Src/Playback.cpp b/Externals/MusicMod/Player/Src/Playback.cpp new file mode 100644 index 0000000000..090d2e07a5 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Playback.cpp @@ -0,0 +1,804 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Playback.h" +#include "InputPlugin.h" +#include "Output.h" +#include "Playlist.h" +#include "Console.h" +#include "Unicode.h" +#include "Rebar.h" +#include "Main.h" +#include "Config.h" +#include "Status.h" + + + +int iCurVol = 255; +ConfIntMinMax ciCurVol( &iCurVol, TEXT( "Volume" ), CONF_MODE_INTERNAL, 255, 0, 255 ); + +int iCurPan = 0; +ConfIntMinMax ciCurPan( &iCurPan, TEXT( "Panning" ), CONF_MODE_INTERNAL, 0, -127, 127 ); + + + +#define VOLUME_STEP ( 255 / 10 ) + + + +bool bPlaying = false; +bool bPaused = false; + +bool bTimerRunning = false; + + + +// Only for reference comparison!!! +TCHAR * szCurrentFilename = NULL; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +// ======================================================================================= +// The timer that calls WndprocMain every second +void EnableTimer( bool bEnabled ) +{ + // Return if the timer is already activated + if( bEnabled == bTimerRunning ) return; + + if( bEnabled ) + { + SetTimer( WindowMain, TIMER_SEEK_UPDATE, 1000, NULL ); + INFO_LOG(AUDIO, "EnableTimer > Activated\n" ); + } + else + { + KillTimer( WindowMain, TIMER_SEEK_UPDATE ); + StatusReset(); + + INFO_LOG(AUDIO, "EnableTimer > Killed\n" ); + } + + bTimerRunning = bEnabled; +} +// ======================================================================================= + + +//////////////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////////////// +bool OpenPlay( TCHAR * szFilename, int iNumber ) +{ + // ======================================================================================= + #ifdef NOGUI + //INFO_LOG(AUDIO, "Playback.cpp: OpenPlay > Begin <%i> <%s>\n" , iNumber, szFilename ); + #else + TCHAR sszBuffer[ 5000 ]; + _stprintf( sszBuffer, TEXT( "Playback.cpp: OpenPlay was called <%i> <%s>" ), iNumber, szFilename ); + Console::Append( sszBuffer ); + #endif + // ======================================================================================= + + + // ======================================================================================= + // Get the right input plugin + // --------------------------------------------------------------------------------------- + if( !szFilename ) return false; + szCurrentFilename = szFilename; + + // --------------------------------------------------------------------------------------- + // TODO: non-file support + // --------------------------------------------------------------------------------------- + // Get file extension + const int iLen = ( int )_tcslen( szFilename ); + TCHAR * walk = szFilename + iLen - 1; + while( ( walk >= szFilename ) && ( *walk != TEXT( '.' ) ) ) walk--; + walk++; + + // --------------------------------------------------------------------------------------- + const int iExtLen = ( int )_tcslen( walk ); + TCHAR * szExt = new TCHAR[ iExtLen + 1 ]; + memcpy( szExt, walk, iExtLen * sizeof( TCHAR ) ); + szExt[ iExtLen ] = TEXT( '\0' ); + + // --------------------------------------------------------------------------------------- + // Compare the extension to the supported extension by the current input plugins + map ::iterator iter = + ext_map.find( szExt ); + delete [] szExt; + if( iter == ext_map.end() ) + { + Console::Append( TEXT( "ERROR: Extension not supported" ) ); + Console::Append( " " ); + INFO_LOG(AUDIO,"OpenPlay > ERROR: Extension not supported\n"); + return false; + } + // --------------------------------------------------------------------------------------- + // Now that we know which input pugin to use we set that one as active + InputPlugin * old_input = active_input_plugin; // Save the last one, if any + active_input_plugin = iter->second; + INFO_LOG(AUDIO,"OpenPlay > Input plugin '%s' activated\n", active_input_plugin->GetFilename()); + // ======================================================================================= + + if( old_input ) + { + // if( active_input_plugin != old_input ) ----> TODO unload old plugin + + // Some output plugins require a call to Close() before each + // call to Open(). Calling Input::Stop() will make the input plugin + // call Output::Close() and thus solve this problem. + old_input->plugin->Stop(); + } + + if( !active_input_plugin->plugin ) + { + Console::Append( TEXT( "ERROR: Input plugin is NULL" ) ); + Console::Append( " " ); + INFO_LOG(AUDIO,"OpenPlay > ERROR: Input plugin is NULL\n"); + return false; + } + + // Connect + //INFO_LOG(AUDIO, "OpenPlay > OutMod\n" ); + active_input_plugin->plugin->outMod = &output_server; // output->plugin; + + // ======================================================================================= + // Re-apply volume and panning + //active_input_plugin->plugin->SetVolume( iCurVol ); + //active_input_plugin->plugin->SetPan( iCurPan ); + //Playback::Eq::Reapply(); + // ======================================================================================= + + // Play + +#ifdef PA_UNICODE + // Filename + const int iFilenameLen = _tcslen( szFilename ); + char * szTemp = new char[ iFilenameLen + 1 ]; + ToAnsi( szTemp, szFilename, iFilenameLen ); + szTemp[ iFilenameLen ] = '\0'; + + // Ansi Title + char szAnsiTitle[ 2000 ] = "\0"; + int length_in_ms; + active_input_plugin->plugin->GetFileInfo( szTemp, szAnsiTitle, &length_in_ms ); + const int iAnsiTitleLen = strlen( szAnsiTitle ); + + // Unicode title + TCHAR szTitle[ 2000 ]; + ToTchar( szTitle, szAnsiTitle, iFilenameLen, iAnsiTitleLen ); + szTitle[ iAnsiTitleLen ] = TEXT( "\0" ); + + active_input_plugin->plugin->Play( szTemp ); + delete [] szTemp; +#else + // ======================================================================================= + // Play the file + // --------------------------------------------------------------------------------------- + // Title + TCHAR szTitle[ 2000 ] = TEXT( "\0" ); + int length_in_ms; + + //INFO_LOG(AUDIO, "OpenPlay > GetFileInfo\n" ); + active_input_plugin->plugin->GetFileInfo( szFilename, szTitle, &length_in_ms ); + + //INFO_LOG(AUDIO, "OpenPlay > Play\n" ); + active_input_plugin->plugin->Play( szFilename ); + // ======================================================================================= +#endif + + bPlaying = true; + bPaused = false; + + // ======================================================================================= + // Title + + //TCHAR szBuffer[ 5000 ]; + //_stprintf( szBuffer, TEXT( "%i. %s - Plainamp" ), iNumber, szTitle ); + //SetWindowText( WindowMain, szBuffer ); + // ======================================================================================= + + /* + TCHAR * szBasename = szFilename + uLen - 1; + while( ( szBasename > szFilename ) && ( *szBasename != TEXT( '\\' ) ) ) szBasename--; + szBasename++; + */ + + // Timer ON + //EnableTimer( true ); + + //INFO_LOG(AUDIO, "OpenPlay > End\n" ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback_PrevOrNext( bool bPrevOrNext ) +{ + // todo: prev/next in pause mode? + + if( !active_input_plugin ) return false; + if( !active_input_plugin->plugin ) return false; + + int iNextIndex = playlist->GetCurIndex(); + int iMaxIndex = playlist->GetMaxIndex(); + if( iMaxIndex < 0 || iNextIndex < 0 ) return false; + + bool res; + if( bPrevOrNext ) + res = Playback::Order::Prev( iNextIndex, iMaxIndex ); + else + res = Playback::Order::Next( iNextIndex, iMaxIndex ); + + if( res ) + { + if( bPlaying ) + { + // NOT TWICE active_input_plugin->plugin->Stop(); + bPlaying = false; + bPaused = false; + + // Timer OFF + EnableTimer( false ); + } + + TCHAR * szFilename = Playlist::GetFilename( iNextIndex ); + if( !szFilename ) return false; + + playlist->SetCurIndex( iNextIndex ); + + return OpenPlay( szFilename, iNextIndex + 1 ); + } + else + { + return false; + } + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Prev() +{ + return Playback_PrevOrNext( true ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +// Play the file +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Play() +{ + static int iLastIndex = -1; + + // --------------------------------------------------------------------------------------- + + #ifndef NOGUI + TCHAR sszBuffer[ 5000 ]; + _stprintf( sszBuffer, TEXT( "Playback::Play() with bPlaying <%i>\n" ), bPlaying ); + Console::Append( sszBuffer ); + Console::Append( TEXT( " " ) ); + #else + //INFO_LOG(AUDIO, "Playback::Play() > Begin <%i>\n" , bPlaying ); + #endif + // --------------------------------------------------------------------------------------- + + // --------------------------------------------------------------------------------------- + if( bPlaying ) // If we are currently playing a file + { + if( !active_input_plugin ) return false; + if( !active_input_plugin->plugin ) return false; + + const int iIndex = playlist->GetCurIndex(); + if( iIndex < 0 ) return false; + + /* + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "OLD [%i] NEW [%i]" ), iLastIndex, iIndex ); + SetWindowText( WindowMain, szBuffer ); + */ + + // If we are not playing the same track/file as before + TCHAR * szFilename = Playlist::GetFilename( iIndex ); + if( szFilename != szCurrentFilename ) + { + // New track! + + // Stop + // NOT TWICE active_input_plugin->plugin->Stop(); + + // Timer OFF + //EnableTimer( false ); + + // Get filename + if( !szFilename ) + { + Console::Append( TEXT( "ERROR: Could not resolve filename" ) ); + Console::Append( " " ); + return false; + } + + // Play + iLastIndex = iIndex; + bPlaying = OpenPlay( szFilename, iIndex + 1 ); + bPaused = false; + } + else + { + // Same track! + if( bPaused ) + { + // Unpause + active_input_plugin->plugin->UnPause(); + bPaused = false; + + // Timer ON + //EnableTimer( true ); + } + else + { + // Restart at beginning + active_input_plugin->plugin->SetOutputTime( 0 ); + } + } + } + // ======================================================================================= + else // we are not currently playing + { + const int iIndex = playlist->GetCurIndex(); + if( iIndex < 0 ) return false; + + // Get filename + TCHAR * szFilename = Playlist::GetFilename( iIndex ); + + // ======================================================================================= + + + //bool bPlaying = false; + //TCHAR * szFilename = TEXT("C:\Files\Spel och spelfusk\Console\Gamecube\Code\vgmstream (isolate ast)\Music\demo36_02.ast"); + //bPlaying = OpenPlay( szFilename, 1 ); + //bPlaying = OpenPlay( szFilename, iIndex + 1 ); + //Console::Append( TEXT( "Playback.cpp:Playback::Play() called OpenPlay" ) ); + + // ======================================================================================= + + if( !szFilename ) + { + Console::Append( TEXT( "ERROR: Could not resolve filename" ) ); + Console::Append( " " ); + return false; + } + #ifndef NOGUI + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Playback.cpp: Play() got the filename <%s>" ), szFilename); + Console::Append( szBuffer ); + //Console::Append( TEXT( " " ) ); + #else + //INFO_LOG(AUDIO, "Playback::Play() > Filename <%s>\n", szFilename); + #endif + + // Play + iLastIndex = iIndex; + bPlaying = OpenPlay( szFilename, iIndex + 1 ); + bPaused = false; + } + return true; +} +// ======================================================================================= + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Pause() +{ + if( !bPlaying ) return false; + if( !active_input_plugin ) return false; + if( !active_input_plugin->plugin ) return false; + + if( bPaused ) + { + // Unpause + active_input_plugin->plugin->UnPause(); + bPaused = false; + + // Timer ON + //EnableTimer( true ); + } + else + { + // Pause + active_input_plugin->plugin->Pause(); + bPaused = true; + + // Timer OFF + //EnableTimer( false ); + } + +// Console::Append( TEXT( "Playback::Pause" ) ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Stop() +{ + if( !bPlaying ) return false; + + // --------------------------------------------------------------------------------------- + // Stop + if( active_input_plugin && active_input_plugin->plugin ) + { + active_input_plugin->plugin->Stop(); + } + active_input_plugin = NULL; // QUICK FIX + // --------------------------------------------------------------------------------------- + + bPlaying = false; + bPaused = false; + + // Timer OFF > It was never turned on + //EnableTimer( false ); + + // Reset seekbar > We don't need this + //Playback::UpdateSeek(); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Next() +{ + return Playback_PrevOrNext( false ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::IsPlaying() +{ + return bPlaying; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::IsPaused() +{ + return bPaused; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +//int getlength(); +//int getoutputtime(); + +bool Playback::UpdateSeek() +{ +#ifdef NOGUI + printf( "Playback::UpdateSeek > Begin\n" ); + if( active_input_plugin ) + { + printf( "GetLength() > Begin\n" ); + const int ms_len = active_input_plugin->plugin->GetLength(); + //const int ms_cur = active_input_plugin->plugin->GetOutputTime(); + + //const int ms_len = getlength(); + //const int ms_cur = getoutputtime(); + + //printf( "Current position is <%i of %i>\n", ms_cur, ms_len ); + } + return true; +#else + static bool bSliderEnabledBefore = false; + bool bSliderEnabledAfter; + + + //if( !WindowSeek ) return false; + + int iVal = 0; + + // If it has not been set + if( !active_input_plugin || !active_input_plugin->plugin ) + { + if( bSliderEnabledBefore ) + { + // Update slider + PostMessage( WindowSeek, TBM_SETPOS, ( WPARAM )( TRUE ), iVal ); + + // Disable slider + EnableWindow( WindowSeek, FALSE ); + bSliderEnabledBefore = false; + } + } + else + { + const int ms_len = active_input_plugin->plugin->GetLength(); + if( ms_len ) + { + const int ms_cur = active_input_plugin->plugin->GetOutputTime(); + iVal = ( ms_cur * 1000 ) / ms_len; + + if( iVal > 1000 ) iVal = 0; + + // ======================================================================================= + //TCHAR szBuffer[ 5000 ]; + //_stprintf( szBuffer, TEXT( "Current position is <%i of %i>" ), ms_cur, ms_len ); + //Console::Append( szBuffer ); + //Console::Append( TEXT( " " ) ); + printf( "Current position is <%i of %i>\n", ms_cur, ms_len ); + // ======================================================================================= + } + + + + if( !bSliderEnabledBefore ) + { + EnableWindow( WindowSeek, TRUE ); + bSliderEnabledBefore = true; + } + + // Update slider + PostMessage( WindowSeek, TBM_SETPOS, ( WPARAM )( TRUE ), iVal ); + } + + return true; +#endif +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playback::PercentToMs( float fPercent ) +{ + if( !active_input_plugin ) return -1; + if( !active_input_plugin->plugin ) return -1; + + const int ms_len = active_input_plugin->plugin->GetLength(); + const int ms_res = ( int )( ms_len * fPercent / 100.0f ); + return ms_res; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::SeekPercent( float fPercent ) +{ + // TODO update slider, NOT HERE!!! + + if( !bPlaying ) return false; + if( bPaused ) return false; // TODO: apply seek when unpausing + if( !active_input_plugin ) return false; + if( !active_input_plugin->plugin ) return false; + + if( fPercent < 0.0f ) + fPercent = 0.0f; + else if( fPercent > 100.0f ) + fPercent = 100.0f; + + const int ms_len = active_input_plugin->plugin->GetLength(); + const int ms_cur = ( int )( ms_len * fPercent / 100.0f ); + active_input_plugin->plugin->SetOutputTime( ms_cur ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool SeekRelative( int ms ) +{ + if( !bPlaying ) return false; + if( bPaused ) return false; // TODO: apply seek when unpausing + if( !active_input_plugin ) return false; + if( !active_input_plugin->plugin ) return false; + + const int ms_len = active_input_plugin->plugin->GetLength(); + const int ms_old = active_input_plugin->plugin->GetOutputTime(); + int ms_new = ms_old + ms; + + if( ms_new < 0 ) + ms_new = 0; + else if( ms_new > ms_len ) + ms_new = ms_len; + + if( ms_new == ms_old ) return true; + active_input_plugin->plugin->SetOutputTime( ms_new ); + + /* + // PROGRESS + // PostMessage( hwnd, PBM_SETPOS , ( WPARAM )( iVal ), 0 ); + // TARCKBAR + PostMessage( wnd_pos, TBM_SETPOS, ( WPARAM )( TRUE ), ms_cur * 1000 / ms_len ); + */ + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Forward() +{ + return SeekRelative( 5000 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Rewind() +{ + return SeekRelative( -5000 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void Playback::NotifyTrackEnd() +{ + bPlaying = false; + bPaused = false; + + // Timer + EnableTimer( false ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playback::Volume::Get() +{ + return iCurVol; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline void Playback_Volume_Set( int iVol ) +{ + if( active_input_plugin && active_input_plugin->plugin ) + { + active_input_plugin->plugin->SetVolume( iVol ); + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Volume::Set( int iVol ) +{ + const int iCurVolBackup = iCurVol; + iCurVol = iVol; + ciCurVol.MakeValidPull(); + + if( iCurVol != iCurVolBackup ) + { + Playback_Volume_Set( iCurVol ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Volume::Up() +{ + if( ciCurVol.IsMax() ) return true; + + const int iCurVolBackup = iCurVol; + iCurVol += VOLUME_STEP; + ciCurVol.MakeValidPull(); + + if( iCurVol != iCurVolBackup ) + { + Console::Append( TEXT( "Volume UP" ) ); + Playback_Volume_Set( iCurVol ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Volume::Down() +{ + if( ciCurVol.IsMin() ) return true; + + const int iCurVolBackup = iCurVol; + iCurVol -= VOLUME_STEP; + ciCurVol.MakeValidPull(); + + if( iCurVol != iCurVolBackup ) + { + Console::Append( TEXT( "Volume DOWN" ) ); + Playback_Volume_Set( iCurVol ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playback::Pan::Get() +{ + return iCurPan; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Pan::Set( int iPan ) +{ + const int iCurPanBackup = iCurPan; + iCurPan = iPan; + ciCurPan.MakeValidPull(); + + if( ( iCurPan != iCurPanBackup ) && active_input_plugin && active_input_plugin->plugin ) + { + active_input_plugin->plugin->SetPan( iCurPan ); + } + + return true; +} diff --git a/Externals/MusicMod/Player/Src/Playback.h b/Externals/MusicMod/Player/Src/Playback.h new file mode 100644 index 0000000000..558b191d1d --- /dev/null +++ b/Externals/MusicMod/Player/Src/Playback.h @@ -0,0 +1,106 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_PLAYBACK_H +#define PA_PLAYBACK_H + + + +#include "Global.h" + + + +#define ORDER_SINGLE 0 +#define ORDER_SINGLE_REPEAT 1 +#define ORDER_NORMAL 2 +#define ORDER_NORMAL_REPEAT 3 +#define ORDER_INVERSE 4 +#define ORDER_INVERSE_REPEAT 5 +#define ORDER_RANDOM 6 + +#define ORDER_FIRST ORDER_SINGLE +#define ORDER_LAST ORDER_RANDOM + +#define ORDER_DEFAULT ORDER_NORMAL_REPEAT + + +#define TIMER_SEEK_UPDATE 1 + + + +typedef bool ( * PresetCallback )( TCHAR * ); + + +namespace Playback +{ + bool Prev(); + bool Play(); + bool Pause(); + bool Stop(); + bool Next(); + + bool IsPlaying(); + bool IsPaused(); + + bool UpdateSeek(); + int PercentToMs( float fPercent ); + bool SeekPercent( float fPercent ); + bool Forward(); + bool Rewind(); + + void NotifyTrackEnd(); + + namespace Volume + { + int Get(); + bool Set( int iVol ); + bool Up(); + bool Down(); + }; + + namespace Pan + { + int Get(); + bool Set( int iPan ); + }; + + namespace Order + { + int GetCurMode(); + bool SetMode( int iMode ); + + TCHAR * GetModeName( int iMode ); +// int GetModeNameLen( int iMode ); + + bool Next( int & iCur, int iMax ); + bool Prev( int & iCur, int iMax ); + }; + + namespace Eq + { + // 63 -> -12db + // 31 -> 0 + // 0 -> +12db + // bool Get( char * eq_data ); + // bool Set( bool bOn, char * pData, int iPreamp ); + int GetCurIndex(); + bool SetIndex( int iPresetIndex ); + + bool Reapply(); + + bool ReadPresets( PresetCallback AddPreset ); + }; +}; + + + +#endif // PA_PLAYBACK_H diff --git a/Externals/MusicMod/Player/Src/PlaybackEq.cpp b/Externals/MusicMod/Player/Src/PlaybackEq.cpp new file mode 100644 index 0000000000..ee779e5545 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlaybackEq.cpp @@ -0,0 +1,223 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Playback.h" +#include "InputPlugin.h" +#include "Console.h" +#include "Config.h" +#include + +using namespace std; + + + +vector eq_vec; + + + +// Note: CurPresetFixed will be renamed to CurPreset when +// custom presets are implemented. This hopefully avoids +// migration trouble. + +int iCurPreset; // -1 means EQ off +ConfInt ciCurPreset( &iCurPreset, TEXT( "CurPresetFixed" ), CONF_MODE_INTERNAL, -1 ); + +bool bPreventDistortion; // Automatic preamp adjustment +ConfBool cbPreventDistortion( &bPreventDistortion, TEXT( "PreventDistortion" ), CONF_MODE_PUBLIC, true ); + + + +//////////////////////////////////////////////////////////////////////////////// +// PRE valid index +//////////////////////////////////////////////////////////////////////////////// +void Playback_Eq_Set( int iPresetIndex ) +{ + if( active_input_plugin && active_input_plugin->plugin ) + { + if( iPresetIndex == -1 ) // == EQ disabled + { + char data[ 10 ] = { 31, 31, 31, 31, 31, 31, 31, 31, 31, 31 }; + active_input_plugin->plugin->EQSet( 0, data, 31 ); + } + else + { + char data[ 10 ]; + memcpy( data, eq_vec[ iPresetIndex ], sizeof( char ) * 10 ); + + if( bPreventDistortion ) + { + // Search minimum (most amplifying band) + int iMin = 63; + int i; + for( i = 0; i < 10; i++ ) + { + if( data[ i ] < iMin ) iMin = data[ i ]; + } + + if( iMin < 31 ) // Possible distortion + { + // Adjust preamp to prevent distortion + active_input_plugin->plugin->EQSet( 1, data, 31 + ( 31 - iMin ) ); + } + else + { + if( iMin > 31 ) // Lower than necessary + { + // Push to zero level so we get + // more volume without distortion + const int iSub = iMin - 31; + for( i = 0; i < 10; i++ ) + { + data[ i ] -= iSub; + } + } + active_input_plugin->plugin->EQSet( 1, data, 31 ); + } + } + else + { + active_input_plugin->plugin->EQSet( 1, data, 31 ); + } + } + } +} + + + +/* +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Eq::Set( bool bOn, char * pData, int iPreamp ) +{ + if( active_input_plugin && active_input_plugin->plugin ) + { + char data[ 10 ]; + memcpy( data, pData, sizeof( char ) * 10 ); + active_input_plugin->plugin->EQSet( bOn ? 1 : 0, data, iPreamp ); + } + return true; +} +*/ + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playback::Eq::GetCurIndex() +{ + return iCurPreset; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Eq::SetIndex( int iPresetIndex ) +{ + if( iPresetIndex >= ( int )eq_vec.size() ) return false; + Playback_Eq_Set( iPresetIndex ); + iCurPreset = iPresetIndex; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Eq::Reapply() +{ + Playback_Eq_Set( iCurPreset ); + return true; +} + + + +// Note: Most of these are the exact presets used in Winamp. +// I do not expect any legal trouble with this but +// in case I am wrong please let me know and I will +// remove them. + +char eq_classical [ 10 ] = { 31, 31, 31, 31, 31, 31, 44, 44, 44, 48 }; +char eq_club [ 10 ] = { 31, 31, 26, 22, 22, 22, 26, 31, 31, 31 }; +char eq_dance [ 10 ] = { 16, 20, 28, 32, 32, 42, 44, 44, 32, 32 }; +char eq_full_bass [ 10 ] = { 16, 16, 16, 22, 29, 39, 46, 49, 50, 50 }; +char eq_full_bass_treble[ 10 ] = { 20, 22, 31, 44, 40, 29, 18, 14, 12, 12 }; +char eq_full_treble [ 10 ] = { 48, 48, 48, 39, 27, 14, 6, 6, 6, 4 }; +char eq_headphones [ 10 ] = { 24, 14, 23, 38, 36, 29, 24, 16, 11, 8 }; +char eq_laptop [ 10 ] = { 24, 14, 23, 38, 36, 29, 24, 16, 11, 8 }; +char eq_large_hall [ 10 ] = { 15, 15, 22, 22, 31, 40, 40, 40, 31, 31 }; +char eq_live [ 10 ] = { 40, 31, 25, 23, 22, 22, 25, 27, 27, 28 }; +char eq_more_bass [ 10 ] = { 22, 22, 22, 22, 22, 22, 26, 31, 31, 31 }; +char eq_party [ 10 ] = { 20, 20, 31, 31, 31, 31, 31, 31, 20, 20 }; +char eq_pop [ 10 ] = { 35, 24, 20, 19, 23, 34, 36, 36, 35, 35 }; +char eq_reggae [ 10 ] = { 31, 31, 33, 42, 31, 21, 21, 31, 31, 31 }; +char eq_rock [ 10 ] = { 19, 24, 41, 45, 38, 25, 17, 14, 14, 14 }; +char eq_ska [ 10 ] = { 36, 40, 39, 33, 25, 22, 17, 16, 14, 16 }; +char eq_soft [ 10 ] = { 24, 29, 34, 36, 34, 25, 18, 16, 14, 12 }; +char eq_soft_rock [ 10 ] = { 25, 25, 28, 33, 39, 41, 38, 33, 27, 17 }; +char eq_techno [ 10 ] = { 19, 22, 31, 41, 40, 31, 19, 16, 16, 17 }; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Eq::ReadPresets( PresetCallback AddPreset ) +{ + if( AddPreset == NULL ) return false; + + eq_vec.push_back( eq_classical ); AddPreset( TEXT( "Classical" ) ); + eq_vec.push_back( eq_club ); AddPreset( TEXT( "Club" ) ); + eq_vec.push_back( eq_dance ); AddPreset( TEXT( "Dance" ) ); + eq_vec.push_back( eq_full_bass ); AddPreset( TEXT( "Full Bass" ) ); + eq_vec.push_back( eq_full_bass_treble ); AddPreset( TEXT( "Full Bass & Treble" ) ); + eq_vec.push_back( eq_full_treble ); AddPreset( TEXT( "Full Treble" ) ); + eq_vec.push_back( eq_headphones ); AddPreset( TEXT( "Headphones" ) ); + eq_vec.push_back( eq_laptop ); AddPreset( TEXT( "Laptop Speakers" ) ); + eq_vec.push_back( eq_large_hall ); AddPreset( TEXT( "Large Hall" ) ); + eq_vec.push_back( eq_live ); AddPreset( TEXT( "Live" ) ); + eq_vec.push_back( eq_more_bass ); AddPreset( TEXT( "More Bass" ) ); + eq_vec.push_back( eq_party ); AddPreset( TEXT( "Party" ) ); + eq_vec.push_back( eq_pop ); AddPreset( TEXT( "Pop" ) ); + eq_vec.push_back( eq_reggae ); AddPreset( TEXT( "Reggae" ) ); + eq_vec.push_back( eq_rock ); AddPreset( TEXT( "Rock" ) ); + eq_vec.push_back( eq_ska ); AddPreset( TEXT( "Ska" ) ); + eq_vec.push_back( eq_soft ); AddPreset( TEXT( "Soft" ) ); + eq_vec.push_back( eq_soft_rock ); AddPreset( TEXT( "Soft Rock" ) ); + eq_vec.push_back( eq_techno ); AddPreset( TEXT( "Techno" ) ); + + // Fix invalid indices + if( iCurPreset < -1 ) + { + iCurPreset = -1; + } + else + { + const int iLen = ( int )eq_vec.size(); + if( iCurPreset >= iLen ) + { + iCurPreset = iLen - 1; + } + } + + // TODO load/save eqf files + // GPL eqf loading from koders.com + // equalizer.c / xmms2 A Gtk2 port of xmms.(xmms2) + // equalizer.c / Digital Disco System(dds) + + return true; +} diff --git a/Externals/MusicMod/Player/Src/PlaybackOrder.cpp b/Externals/MusicMod/Player/Src/PlaybackOrder.cpp new file mode 100644 index 0000000000..50e09a5da2 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlaybackOrder.cpp @@ -0,0 +1,205 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + + +#include "Playback.h" +#include "Config.h" +#include +#include + + + +int iCurOrder = ORDER_DEFAULT; +ConfIntMinMax ciCurOrder( &iCurOrder, TEXT( "Order" ), CONF_MODE_INTERNAL, ORDER_DEFAULT, ORDER_FIRST, ORDER_LAST ); + +bool bRandomReady = false; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playback::Order::GetCurMode() +{ + return iCurOrder; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Order::SetMode( int iMode ) +{ + iCurOrder = iMode; + ciCurOrder.MakeValidDefault(); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +TCHAR * Playback::Order::GetModeName( int iMode ) +{ + switch( iMode ) + { + case ORDER_SINGLE: return TEXT( " Single" ); + case ORDER_SINGLE_REPEAT: return TEXT( " Single + Repeat" ); + case ORDER_NORMAL: return TEXT( " Normal" ); + case ORDER_NORMAL_REPEAT: return TEXT( " Normal + Repeat" ); + case ORDER_INVERSE: return TEXT( " Inverse" ); + case ORDER_INVERSE_REPEAT: return TEXT( " Inverse + Repeat" ); + case ORDER_RANDOM: return TEXT( " Random" ); + default: return NULL; + } +} + +/* +int Playback::Order::GetModeNameLen( int iMode ) +{ + switch( uMode ) + { + case ORDER_SINGLE: return 7; + case ORDER_SINGLE_REPEAT: return 16; + case ORDER_NORMAL: return 7; + case ORDER_NORMAL_REPEAT: return 16; + case ORDER_INVERSE: return 8; + case ORDER_INVERSE_REPEAT: return 17; + case ORDER_RANDOM: return 7; + default: return 0; + } +} +*/ + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool NextNormal( int & iCur, int iMax ) +{ + if( iCur >= iMax ) return false; + iCur++; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool NextNormalRepeat( int & iCur, int iMax ) +{ + if( iCur >= iMax ) + iCur = 0; + else + iCur++; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool NextInverse( int & iCur, int iMax ) +{ + if( iCur <= 0 ) return false; + iCur--; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool NextInverseRepeat( int & iCur, int iMax ) +{ + if( iCur <= 0 ) + iCur = iMax; + else + iCur--; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool NextRandom( int & iCur, int iMax ) +{ + if( iMax < 2 ) return false; + + if( !bRandomReady ) + { + srand( ( unsigned )time( NULL ) ); + bRandomReady = true; + } + + const int iNew = ( int )( rand() / ( float )RAND_MAX * iMax ); + if( iNew != iCur ) + iCur = iNew; + else + { + if( iCur >= iMax ) + iCur = 0; + else + iCur++; + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Order::Next( int & iCur, int iMax ) +{ + switch( iCurOrder ) + { + case ORDER_SINGLE: return false; + case ORDER_SINGLE_REPEAT: return true; + case ORDER_NORMAL: return NextNormal( iCur, iMax ); + case ORDER_NORMAL_REPEAT: return NextNormalRepeat( iCur, iMax ); + case ORDER_INVERSE: return NextInverse( iCur, iMax ); + case ORDER_INVERSE_REPEAT: return NextInverseRepeat( iCur, iMax ); + case ORDER_RANDOM: return NextRandom( iCur, iMax ); + default: return false; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playback::Order::Prev( int & iCur, int iMax ) +{ + switch( iCurOrder ) + { + case ORDER_SINGLE: return false; + case ORDER_SINGLE_REPEAT: return true; + case ORDER_NORMAL: return NextInverse( iCur, iMax ); + case ORDER_NORMAL_REPEAT: return NextInverseRepeat( iCur, iMax ); + case ORDER_INVERSE: return NextNormal( iCur, iMax ); + case ORDER_INVERSE_REPEAT: return NextNormalRepeat( iCur, iMax ); + case ORDER_RANDOM: return NextRandom( iCur, iMax ); + default: return false; + } +} + diff --git a/Externals/MusicMod/Player/Src/Player.cpp b/Externals/MusicMod/Player/Src/Player.cpp new file mode 100644 index 0000000000..8c1075ba56 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Player.cpp @@ -0,0 +1,380 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + + +////////////////////////////////////////////////////////////////////////////////////////// +// Usage instructions +// ŻŻŻŻŻŻŻŻŻŻŻŻŻŻ + +// ======================================================================================= +// Plugins +// --------------------------------------------------------------------------------------- +// Plainamp setup > The output plugin must be manually loaded and activated the first time it's used. +// After that the ini saves OutputPluginActive___out_wave_gpl.dll=1. Input plugins are automatically +// found, loaded and activated. +// ======================================================================================= + + +// ======================================================================================= +// The ini path szIniPath +// --------------------------------------------------------------------------------------- +/* We will get problems if the program can't find the ini settings. Plugins will not be loaded, + or loadedand then unloaded before activated, or not working. */ +// ======================================================================================= + +/////////////////////////////////// + + + +////////////////////////////////////////////////////////////////////////////////////////// +// Include +// ŻŻŻŻŻŻŻŻŻŻ +#include +//#include "Unicode.h" +//#include "Output.h" +#include +#include + +#include "Global.h" // Local +#include "Font.h" +#include "InputPlugin.h" +#include "OutputPlugin.h" +#include "VisPlugin.h" +#include "DspPlugin.h" +#include "GenPlugin.h" +#include "Main.h" +#include "Rebar.h" +#include "Playlist.h" +#include "Status.h" +#include "PluginManager.h" +#include "Prefs.h" +#include "Config.h" +#include "Emabox/Emabox.h" +#include "Console.h" + +#include "PlayerExport.h" // DLL Player +///////////////////////// + + + +////////////////////////////////////////////////////////////////////////////////////////// +// Declarations and definitions +// ŻŻŻŻŻŻŻŻŻŻ + + +// ------------------------- +// Keys +// --------- +#define PLUS_ALT ( FVIRTKEY | FALT ) +#define PLUS_CONTROL ( FVIRTKEY | FCONTROL ) +#define PLUS_CONTROL_ALT ( FVIRTKEY | FCONTROL | FALT ) +#define PLUS_CONTROL_SHIFT ( FVIRTKEY | FCONTROL | FSHIFT ) +#define PLUS_SHIFT ( FVIRTKEY | FSHIFT ) +// ------------- + + +HINSTANCE g_hInstance = NULL; // extern +HINSTANCE hInstance = NULL; // extern + +TCHAR * szHomeDir = NULL; // extern +int iHomeDirLen = 0; // extern + +TCHAR * szPluginDir = NULL; // extern +int iPluginDirLen = 0; // extern + + +// ------------------------- +/* Read global settings from the ini file. They are read from the ini file through Config.h. + Usage: ( &where to place it, the option name, public or private, default value) */ +// --------- +TCHAR szCurDir[ MAX_PATH + 1 ] = TEXT( "" ); +ConfCurDir ccdCurDir( szCurDir, TEXT( "CurDir" ) ); +// ------------------------- + +bool bWarnPluginsMissing; +ConfBool cbWarnPluginsMissing( &bWarnPluginsMissing, TEXT( "WarnPluginsMissing" ), CONF_MODE_PUBLIC, false ); + +bool bLoop; +ConfBool cbLoop( &bLoop, TEXT( "Loop" ), CONF_MODE_PUBLIC, false ); +///////////////////////// + + + +//////////////////////////////////////////////////////////////////////////////// +// The first function that is called by the exe +//////////////////////////////////////////////////////////////////////////////// +void Player_Main(bool Console) +{ + // ======================================================================================= + // Set variables + //g_hInstance = hInstance; + // ======================================================================================= + + //printf( "DLL > main_dll() opened\n" ); + + if (Console) Player_Console(true); + + //MessageBox(0, "main() opened", "", 0); + //printf( "main() opened\n" ); + INFO_LOG(AUDIO,"\n=========================================================\n\n\n" ); + //INFO_LOG(AUDIO, "DLL > Player_Main() > Begin\n" ); + INFO_LOG(AUDIO, "DLL > Settings:\n", bLoop); + + + // ======================================================================================= + // Load full config from ini file + //Conf::Init( hInstance ); + Conf::Init( ); + + // --------------------------------------------------------------------------------------- + + INFO_LOG(AUDIO, "DLL > Loop: %i\n", bLoop); + INFO_LOG(AUDIO, "DLL > WarnPluginsMissing: %i\n", bWarnPluginsMissing); + // --------------------------------------------------------------------------------------- + + // ======================================================================================= + + + // ======================================================================================= + /* Get home dir. We use the TCHAR type for the dirname to insure us against foreign letters + in the dirnames */ + szHomeDir = new TCHAR[ MAX_PATH + 1 ]; + iHomeDirLen = GetModuleFileName( NULL, szHomeDir, MAX_PATH ); + //if( !iHomeDirLen ) return 1; + + // --------------------------------------------------------------------------------------- + // Walk through the pathname and look for backslashes + TCHAR * walk = szHomeDir + iHomeDirLen - 1; + while( ( walk > szHomeDir ) && ( *walk != TEXT( '\\' ) ) ) walk--; + // --------------------------------------------------------------------------------------- + + walk++; + *walk = TEXT( '\0' ); + iHomeDirLen = walk - szHomeDir; + // ======================================================================================= + + + // ======================================================================================= + /* Get plugins dir. Notice to change the number 8 in two places below if the dir name + is changed */ + szPluginDir = new TCHAR[ MAX_PATH + 1 ]; + memcpy( szPluginDir, szHomeDir, iHomeDirLen * sizeof( TCHAR ) ); + memcpy( szPluginDir + iHomeDirLen, TEXT( "PluginsMusic" ), 12 * sizeof( TCHAR ) ); + szPluginDir[ iHomeDirLen + 12 ] = TEXT( '\0' ); + INFO_LOG(AUDIO,"DLL > Plugindir: %s\n", szPluginDir); + // ======================================================================================= + #ifndef NOGUI + Font::Create(); + //Console::Append( TEXT( "Winmain.cpp called Font::Create()" ) ); + #endif + + + // --------------------------------------------------------------------------------------- + // Set volume. This must probably be done after the dll is loaded. + //GlobalVolume = Playback::Volume::Get(); // Don't bother with this for now + //GlobalCurrentVolume = GlobalVolume; + //Output_SetVolume( GlobalVolume ); + INFO_LOG(AUDIO,"DLL > Volume: %i\n\n", GlobalVolume); + // --------------------------------------------------------------------------------------- + + + // ======================================================================================= + // The only thing this function currently does is creating the Playlist. + // ======================================================================================= + BuildMainWindow(); + + //addfiletoplaylist("c:\\zelda\\demo37_01.ast"); + //addfiletoplaylist("c:\\zelda\\demo36_02.ast"); + //Console::Append( TEXT( "Winmain.cpp called BuildMainWindow()" ) ); + + //Prefs::Create(); // This creates windows preferences + //Console::Append( TEXT( "Winmain.cpp called Prefs::Create()" ) ); + + // Find plugins + Plugin::FindAll ( szPluginDir, TEXT( "in_*.dll" ), true ); + Plugin::FindAll( szPluginDir, TEXT( "out_*.dll" ), false ); + Plugin::FindAll ( szPluginDir, TEXT( "vis_*.dll" ), false ); + Plugin::FindAll ( szPluginDir, TEXT( "dsp_*.dll" ), false ); + Plugin::FindAll ( szPluginDir, TEXT( "gen_*.dll" ), true ); + + //INFO_LOG(AUDIO, "Winmain.cpp > PluginManager::Fill()\n" ); + PluginManager::Fill(); + + //INFO_LOG(AUDIO, "Winmain.cpp > PluginManager::Fill()\n" ); + + + + // ======================================================================================= + + // Check plugin presence + // ======================================================================================= + + + // ======================================================================================= + + // Todo: all the rest... + // ACCEL accels[] = { + + // ======================================================================================= + + + + // ======================================================================================= + + //Playback::Play(); + //play_file("C:\\opening.hps"); + //play_file("C:\\Files\\Spel och spelfusk\\Console\\Gamecube\\Code\\Dolphin\\Binary\\win32\\evt_x_event_00.dsp"); + //printf("Winmain.cpp called Playback.cpp:Playback::Play()\n"); + + // ======================================================================================= + // ---- Set volume and get current location ---- + // Somehow we don't have access to active_input_plugin->plugin->GetLength() from here so + // we have to call it in Playback::UpdateSeek() instead + //Sleep(1000); + + //Playback::UpdateSeek(); + //Output_SetVolume( 100 ); // volume is better set from the ini file + // --------------------------------------------------------------------------------------- + // ======================================================================================= + + + // ======================================================================================= + // Check the plugins + if( input_plugins.empty() ) + { + INFO_LOG(AUDIO,"\n *** Warning: No valid input plugins found\n\n"); + } + else + { + INFO_LOG(AUDIO," >>> These valid input plugins were found:\n"); + for(int i = 0; i < input_plugins.size(); i++) + INFO_LOG(AUDIO," %i: %s\n", (i + 1), input_plugins.at(i)->GetFilename()); + INFO_LOG(AUDIO,"\n"); + } + + // The input plugins are never activated here, they are activate for each file + if( !active_input_plugin || !active_input_plugin->plugin ) + { + // INFO_LOG(AUDIO,"The input plugin is not activated yet\n"); + } + else + { + //const int ms_len = active_input_plugin->plugin->GetLength(); + //const int ms_cur = active_input_plugin->plugin->GetOutputTime(); + //INFO_LOG(AUDIO,"We are at <%i of %i>\n", ms_cur, ms_len); + } + // --------------------------------------------------------------------------------------- + if( active_output_count > 0 ) + { + // Show current playback progress + /*int res_temp; + for( int i = 0; i < active_output_count; i++ ) + { + res_temp = active_output_plugins[ i ]->plugin->GetOutputTime(); + } + INFO_LOG(AUDIO,"Playback progress <%i>\n", res_temp);*/ + } + else + { + INFO_LOG(AUDIO,"\n *** Warning: The output plugin is not working\n\n"); + } + // ======================================================================================= + + // ======================================================================================= + // Start the timer + if(!TimerCreated && bLoop) // Only create this the first time + { + //INFO_LOG(AUDIO,"Created the timer\n"); + MakeTime(); + TimerCreated = true; + } + // ======================================================================================= + + INFO_LOG(AUDIO, "\n=========================================================\n\n" ); + //INFO_LOG(AUDIO, "DLL > main_dll() > End\n\n\n" ); + + //std::cin.get(); +} + + + + + + +// ======================================================================================= +// Should I use this? +void close() +{ + + printf( "The Winmain.cpp message loop was reached\n" ); + + // Message loop + //std::cin.get(); + + Playback::Stop(); // If we don't call this before we unload the dll we get a crash + + printf("We are now past the message loop\n" ); + + //DestroyAcceleratorTable( hAccel ); + + + // ======================================================================================= + // Input + vector ::iterator iter_input = input_plugins.begin(); + while( iter_input != input_plugins.end() ) + { + ( *iter_input )->Unload(); + iter_input++; + } + + // Output + vector ::iterator iter_output = output_plugins.begin(); + while( iter_output != output_plugins.end() ) + { + ( *iter_output )->Unload(); + iter_output++; + } + + // General + vector ::iterator iter_gen = gen_plugins.begin(); + while( iter_gen != gen_plugins.end() ) + { + ( *iter_gen )->Unload(); + iter_gen++; + } + // ======================================================================================= + + + // TODO: create main::destroy + // UnregisterClass( PA_CLASSNAME, g_hInstance ); + + //Prefs::Destroy(); + + //Font::Destroy(); + +/* + delete [] szPluginDir; + delete [] szHomeDir; +*/ + + // --------------------------------------------------------------------------------------- + // We don't save any changes + //Conf::Write(); + + //printf("Winmain.cpp called Conf::Write(), the last function\n"); + // --------------------------------------------------------------------------------------- + + //std::cin.get(); // Let use see all messages + + //return 0; +} +// ======================================================================================= diff --git a/Externals/MusicMod/Player/Src/PlayerExport.cpp b/Externals/MusicMod/Player/Src/PlayerExport.cpp new file mode 100644 index 0000000000..db35042358 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlayerExport.cpp @@ -0,0 +1,181 @@ + +////////////////////////////////////////////////////////////////////////////////////////// +// Include +// ŻŻŻŻŻŻŻŻŻŻ +#include // System + +#include "Common.h" // Global common +#include "Log.h" + +//#include "../../Common/Src/Console.h" // Local common + +#include "OutputPlugin.h" // Local +#include "Playback.h" +#include "Playlist.h" + +#define _DLL_PLAYER_H_ +#include "PlayerExport.h" // DLL Player +////////////////////////////// + + +////////////////////////////////////////////////////////////////////////////////////////// +// Declarations and definitions +// ŻŻŻŻŻŻŻŻŻŻ +std::string CurrentlyPlayingFile; +int GlobalVolume = -1; +bool GlobalMute = false; +bool GlobalPause; +bool TimerCreated = false; +bool Initialized = false; +///////////////////////////////// + + +// ------------------------- +/* We keep the file in the playlist, even though we currently only ever have one file here + at a time */ +// --------- +void AddFileToPlaylist(char * a) +{ + //playlist->RemoveAll(); + + #include "unicode.h" + const int iLen = strlen(a); // I can't do this because I don't + + printf( "iLen <%i>\n", iLen ); + + // --------------------------------------------------------------------------------------- + // Do some string conversion + TCHAR * szKeep; + szKeep = new TCHAR[ iLen + 1 ]; + ToTchar( szKeep, a, iLen ); + szKeep[ iLen ] = TEXT( '\0' ); + playlist->PushBack( szKeep ); + // --------------------------------------------------------------------------------------- + + // If we added a second file the current index = -1 so we have to change that back + playlist->SetCurIndex( 0 ); +} + + + +void Player_Play(char * FileName) +{ + INFO_LOG(AUDIO,"Play file <%s>\n", FileName); + + // Check if the file exists + if(GetFileAttributes(FileName) == INVALID_FILE_ATTRIBUTES) + { + INFO_LOG(AUDIO,"Warning: The file <%s> does not exist. Something is wrong.\n", FileName); + return; + } + + Playback::Stop(); + //INFO_LOG(AUDIO,"Stop\n"); + playlist->RemoveAll(); + //INFO_LOG(AUDIO,"RemoveAll\n"); + AddFileToPlaylist(FileName); + //INFO_LOG(AUDIO,"addfiletoplaylist\n"); + + // Play the file + Playback::Play(); + + CurrentlyPlayingFile = FileName; + + // --------------------------------------------------------------------------------------- + // Set volume. This must probably be done after the dll is loaded. + //Output_SetVolume( Playback::Volume::Get() ); + //INFO_LOG(AUDIO,"Volume(%i)\n", Playback::Volume::Get()); + // --------------------------------------------------------------------------------------- + + GlobalPause = false; +} + +void Player_Stop() +{ + Playback::Stop(); + //INFO_LOG(AUDIO,"Stop\n"); + playlist->RemoveAll(); + + CurrentlyPlayingFile = ""; + + GlobalPause = false; +} + + +void Player_Pause() +{ + if (!GlobalPause) + { + INFO_LOG(AUDIO,"DLL > Pause\n"); + Playback::Pause(); + GlobalPause = true; + } + else + { + INFO_LOG(AUDIO,"DLL > UnPause from Pause\n"); + Player_Unpause(); + GlobalPause = false; + } +} + +void Player_Unpause() +{ + INFO_LOG(AUDIO,"DLL > UnPause\n"); + Playback::Play(); + GlobalPause = false; +} + + +////////////////////////////////////////////////////////////////////////////////////////// +// Declarations and definitions +// ŻŻŻŻŻŻŻŻŻŻŻŻŻ +/* About the volume: The colume is normally update for the output plugin in Output.cpp and for the + inout plugin (why?) in Playback.cpp with Playback::Volume::Set(). But I don't know how that works + so I only use GlobalVolume to keep track of the volume */ +// ------------------------------ +void Player_Mute(int Vol) +{ + if(GlobalVolume == -1) GlobalVolume = Vol; + INFO_LOG(AUDIO,"DLL > Mute <%i> <%i>\n", GlobalVolume, GlobalMute); + + GlobalMute = !GlobalMute; + + // Set volume + if(GlobalMute) + { + Output_SetVolume( 0 ); + INFO_LOG(AUDIO,"DLL > Volume <%i>\n", GlobalMute); + } + else + { + Output_SetVolume( GlobalVolume ); + INFO_LOG(AUDIO,"DLL > Volume <%i>\n", GlobalMute); + } + + //INFO_LOG(AUDIO,"Volume(%i)\n", Playback::Volume::Get()); +} +/////////////////////////////////////// + + +void Player_Volume(int Vol) +{ + GlobalVolume = Vol; + Output_SetVolume( GlobalVolume ); + //INFO_LOG(AUDIO,"DLL > Volume <%i> <%i>\n", GlobalVolume, GlobalCurrentVolume); +} + +void ShowConsole() +{ +// Console::Open(100, 2000, "MusicMod", true); // give room for 2000 rows +} + + +void Player_Console(bool Console) +{ + if(Console) + ShowConsole(); + else + #if defined (_WIN32) + FreeConsole(); + #endif +} diff --git a/Externals/MusicMod/Player/Src/PlayerExport.h b/Externals/MusicMod/Player/Src/PlayerExport.h new file mode 100644 index 0000000000..54ed9fba31 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlayerExport.h @@ -0,0 +1,49 @@ +// Copyright (C) 2003-2008 Dolphin Project. + +// 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, version 2.0. + +// 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 2.0 for more details. + +// A copy of the GPL 2.0 should have been included with the program. +// If not, see http://www.gnu.org/licenses/ + +// Official SVN repository and contact information can be found at +// http://code.google.com/p/dolphin-emu/ + + + +////////////////////////////////////////////////////////////////////////////////////////// +// Include +// ŻŻŻŻŻŻŻŻŻŻ + +////////////////////////////////// + + +// ------------------------- +// DLL Player +// --------- +void MakeTime(); +#ifndef _DLL_PLAYER_H_ + extern int GlobalVolume; + extern int GlobalCurrentVolume; + extern bool GlobalPause; + extern bool TimerCreated; +#endif + + +// Dll export +#define EXPORT __declspec(dllexport) +EXPORT void Player_Main(bool Console); +EXPORT void Player_Console(bool Console); +EXPORT void Player_Play(char *); +EXPORT void Player_Stop(); +EXPORT void Player_Pause(); +EXPORT void Player_Unpause(); +EXPORT void Player_Mute(int Vol); +EXPORT void Player_Volume(int Vol); + diff --git a/Externals/MusicMod/Player/Src/Playlist.cpp b/Externals/MusicMod/Player/Src/Playlist.cpp new file mode 100644 index 0000000000..68060b72b4 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Playlist.cpp @@ -0,0 +1,1383 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Playlist.h" +#include "AddDirectory.h" +#include "Rebar.h" +#include "Main.h" +#include "Status.h" +#include "Console.h" +#include "Font.h" +#include "Playback.h" +#include "InputPlugin.h" +#include "Prefs.h" +#include "Config.h" +#include "Unicode.h" +#include "Path.h" +#include "commdlg.h" + + + + + + + + +HWND WindowPlaylist = NULL; // extern +// WNDPROC WndprocPlaylistBackup = NULL; + +// int iCurIndex = -1; +// int iMaxIndex = -1; + + + +// LRESULT CALLBACK WndprocPlaylist( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); +void Playlist_SelectSingle( int iIndex ); + + +struct PlaylistEntry +{ + TCHAR * szFilename; + // More to come +}; + +/* +/////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playlist::GetCurIndex() +{ + return iCurIndex; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playlist::GetMaxIndex() +{ + return iMaxIndex; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::SetCurIndex( int iIndex ) +{ + if( iIndex < 0 || iIndex > iMaxIndex ) return false; + + iCurIndex = iIndex; + if( bPlaylistFollow ) + { + Playlist_SelectSingle( iCurIndex ); + } + + return true; +} +*/ + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::Create() +{ + if( WindowPlaylist ) return false; + + // If this is not called a crash occurs + PlaylistView::Create(); + + + #ifndef NOGUI + + RECT ClientMain; + GetClientRect( WindowMain, &ClientMain ); + + const int iClientHeight = ClientMain.bottom - ClientMain.top; + const int iClientWidth = ClientMain.right - ClientMain.left; + const int iPlaylistHeight = iClientHeight - iRebarHeight - iStatusHeight; + + WindowPlaylist = CreateWindowEx( + WS_EX_CLIENTEDGE, + TEXT( "LISTBOX" ), + NULL, + WS_VSCROLL | + LBS_DISABLENOSCROLL | + LBS_EXTENDEDSEL | + LBS_HASSTRINGS | + LBS_NOTIFY | + LBS_NOINTEGRALHEIGHT | + WS_CHILD | + WS_VISIBLE, // + 0, + iRebarHeight, // + -2, + iClientWidth, + iPlaylistHeight, + WindowMain, + NULL, + g_hInstance, + NULL + ); + + // Exchange window procedure + WndprocPlaylistBackup = ( WNDPROC )GetWindowLong( WindowPlaylist, GWL_WNDPROC ); + if( WndprocPlaylistBackup != NULL ) + { + SetWindowLong( WindowPlaylist, GWL_WNDPROC, ( LONG )WndprocPlaylist ); + } + + Font::Apply( WindowPlaylist ); + + #endif NOGUI + + + + TCHAR * szPlaylistMind = new TCHAR[ iHomeDirLen + 12 + 1 ]; + memcpy( szPlaylistMind, szHomeDir, iHomeDirLen * sizeof( TCHAR ) ); + memcpy( szPlaylistMind + iHomeDirLen, TEXT( "Plainamp.m3u" ), 12 * sizeof( TCHAR ) ); + + + + szPlaylistMind[ iHomeDirLen + 12 ] = TEXT( '\0' ); + + + Playlist::AppendPlaylistFile( szPlaylistMind ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline void Playlist_SelectSingle( int iIndex ) +{ + SendMessage( + WindowPlaylist, + LB_SETSEL, + FALSE, + -1 + ); + + SendMessage( + WindowPlaylist, + LB_SETSEL, + TRUE, + iIndex + ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline bool Playlist_IsSelected( int iIndex ) +{ + return ( 0 != SendMessage( + WindowPlaylist, + LB_GETSEL, + ( WPARAM )iIndex, + 0 + ) ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline bool Playlist_SelectModify( int iIndex, bool bSelected ) +{ + return ( LB_ERR != SendMessage( + WindowPlaylist, + LB_SETSEL, + ( WPARAM )( bSelected ? TRUE : FALSE ), + iIndex + ) ); +} + + +/* +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist_Remove( int iIndex ) +{ + if( iIndex < 0 || iIndex > iMaxIndex ) return false; + + // Get entry data + PlaylistEntry * entry = ( PlaylistEntry * )SendMessage( + WindowPlaylist, + LB_GETITEMDATA, + iIndex, + 0 + ); + + // Free data + if( entry ) + { + if( entry->szFilename ) delete [] entry->szFilename; + delete entry; + } + + SendMessage( + WindowPlaylist, + LB_DELETESTRING, + iIndex, + 0 + ); + + if( iIndex < iCurIndex ) + iCurIndex--; + if( ( iIndex == iCurIndex ) && ( iCurIndex == iMaxIndex ) ) + iCurIndex = iMaxIndex - 1; + iMaxIndex--; + + return true; +} +*/ + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline int Playlist_GetCount() +{ + return ( int )SendMessage( + WindowPlaylist, + LB_GETCOUNT, + 0, + 0 + ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline int Playlist_GetCaretIndex() +{ + return ( int )SendMessage( + WindowPlaylist, + LB_GETCARETINDEX, + 0, + 0 + ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline bool Playlist_SetCaretIndex( int iIndex ) +{ + return ( LB_OKAY == SendMessage( + WindowPlaylist, + LB_SETCARETINDEX, + ( WPARAM )iIndex, + FALSE + ) ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +inline int Playlist_GetSelCount() +{ + return ( int )SendMessage( + WindowPlaylist, + LB_GETSELCOUNT, + 0, + 0 + ); +} + + +/* +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playlist_MouseToIndex() +{ + POINT p; + GetCursorPos( &p ); + ScreenToClient( WindowPlaylist, &p ); + + int iIndex = ( int )SendMessage( + WindowPlaylist, + LB_ITEMFROMPOINT, + 0, + p.x | ( p.y << 16 ) + ); + + if( ( iIndex < 0 ) || ( iIndex > iMaxIndex ) ) + return -1; + else + return iIndex; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::Clear() +{ + if( iMaxIndex < 0 ) return false; + + int iCount = iMaxIndex + 1; + while( iCount-- ) + { + Playlist_Remove( iCount ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::RemoveSelected() +{ + int iSelCount = Playlist_GetSelCount(); + if( iSelCount < 0 ) return false; + + // Which items are selected? + int * sel = new int[ iSelCount ]; + LRESULT lResult = SendMessage( + WindowPlaylist, + LB_GETSELITEMS, + ( WPARAM )iSelCount, + ( LPARAM )sel + ); + + // Remove + if( lResult > 0 ) + { + while( lResult-- ) + { + int iIndex = sel[ lResult ]; + Playlist_Remove( iIndex ); + } + } + + delete [] sel; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::Crop() +{ + int iAllCount = Playlist_GetCount(); + if( iAllCount < 1 ) return false; + + int iSelCount = Playlist_GetSelCount(); + if( iSelCount < 0 ) + { + return false; + } + else if( iSelCount == 0 ) + { + // None selected + return Clear(); + } + + // Which items are selected? + int * sel = new int[ iSelCount ]; + LRESULT lResult = SendMessage( + WindowPlaylist, + LB_GETSELITEMS, + ( WPARAM )iSelCount, + ( LPARAM )sel + ); + + int iLowerEqualIndex = iSelCount - 1; + for( int i = iAllCount - 1; i >= 0; i-- ) + { + while( ( sel[ iLowerEqualIndex ] > i ) && ( iLowerEqualIndex > 0 ) ) + { + iLowerEqualIndex--; + } + + if( i != sel[ iLowerEqualIndex ] ) + { + // Not selected -> remove + Playlist_Remove( i ); + } + } + + delete [] sel; + return true; +} +*/ + +/* +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist_MoveSel( bool bUpOrDown ) +{ + static bool bMoving = false; + if( bMoving ) return false; + + bMoving = true; + + const int iSelCount = Playlist_GetSelCount(); + if( iSelCount < 0 ) + { + // No items selected + bMoving = false; + return false; + } + + // Which items are selected? + int * sel = new int[ iSelCount ]; + LRESULT lResult = SendMessage( + WindowPlaylist, + LB_GETSELITEMS, + ( WPARAM )iSelCount, + ( LPARAM )sel + ); + + if( lResult <= 0 ) + { + // Nothing to move + delete [] sel; + bMoving = false; + return false; + } + + if( ( bUpOrDown && ( sel[ 0 ] == 0 ) ) || + ( !bUpOrDown && ( sel[ iSelCount - 1 ] == iMaxIndex ) ) ) + { + // Cannot move + delete [] sel; + bMoving = false; + return false; + } + + const int iOldTop = ( int )SendMessage( + WindowPlaylist, + LB_GETTOPINDEX, + 0, + 0 + ); + + // 1 _2_[3][4][5] 6 7 [8] 9 + // --> 1 [3][4][5]_2_ 6 [8]_7_ 9 + + // Redrawing OFF + // SendMessage( WindowPlaylist, WM_SETREDRAW, ( WPARAM )FALSE, 0 ); + + int i = ( bUpOrDown ? 0 : iSelCount - 1 ); + do + { + // Backup the jumper + PlaylistEntry * entry_old = ( PlaylistEntry * )SendMessage( + WindowPlaylist, + LB_GETITEMDATA, + sel[ i ] + ( bUpOrDown ? -1 : 1 ), + 0 + ); + + do + { + // Copy on + PlaylistEntry * entry_new = ( PlaylistEntry * )SendMessage( + WindowPlaylist, + LB_GETITEMDATA, + sel[ i ], + 0 + ); + + int iDest = sel[ i ] + ( bUpOrDown ? -1 : 1 ); + + // Update entry display == delete, insert, set data + SendMessage( + WindowPlaylist, + LB_DELETESTRING, + iDest, + 0 + ); + iDest = ( int )SendMessage( + WindowPlaylist, + LB_INSERTSTRING, + iDest, + ( LPARAM )entry_new->szFilename + ); + SendMessage( + WindowPlaylist, + LB_SETITEMDATA, + iDest, + ( LPARAM )entry_new + ); + + if( sel[ i ] == iCurIndex ) + { + iCurIndex += ( bUpOrDown ? -1 : 1 ); + } + + i += ( bUpOrDown ? 1 : -1 ); + } while( bUpOrDown + ? + ( i < iSelCount ) && ( sel[ i - 1 ] + 1 == sel[ i ] ) + : + ( i >= 0 ) && ( sel[ i + 1 ] - 1 == sel[ i ] ) + ); + + // Place the jumper + int iLast = ( bUpOrDown ? sel[ i - 1 ] : sel[ i + 1 ] ); + + // Update entry display == delete, insert, set data + SendMessage( + WindowPlaylist, + LB_DELETESTRING, + iLast, + 0 + ); + iLast = ( int )SendMessage( + WindowPlaylist, + LB_INSERTSTRING, + iLast, + ( LPARAM )entry_old->szFilename + ); + SendMessage( + WindowPlaylist, + LB_SETITEMDATA, + iLast, + ( LPARAM )entry_old + ); + } while( bUpOrDown + ? + ( i < iSelCount ) + : + ( i >= 0 ) + ); + + // Select new indices (old selection went away on insert/delete + if( bUpOrDown ) + { + for( i = 0; i < iSelCount; i++ ) + SendMessage( WindowPlaylist, LB_SETSEL, TRUE, sel[ i ] - 1 ); + } + else + { + for( i = 0; i < iSelCount; i++ ) + SendMessage( WindowPlaylist, LB_SETSEL, TRUE, sel[ i ] + 1 ); + } + + // Prevent scrolling + SendMessage( + WindowPlaylist, + LB_SETTOPINDEX, + ( WPARAM )iOldTop, + 0 + ); + + // Redrawing ON + // SendMessage( WindowPlaylist, WM_SETREDRAW, ( WPARAM )TRUE, 0 ); + + + delete [] sel; + bMoving = false; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocPlaylist( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + + static bool bDragging = false; + static bool bMoveLock = false; + static int iDragStartY; + static int iItemHeight = 0x7fffffff; + + switch( message ) + { + case WM_MOUSEMOVE: + { + if( !bDragging || bMoveLock ) break; + bMoveLock = true; + + const int y = HIWORD( lp ); + const int diff = y - iDragStartY; + if( abs( diff ) > iItemHeight / 2 ) + { + iDragStartY += ( ( diff > 0 ) ? iItemHeight : -iItemHeight ); + Playlist_MoveSel( diff < 0 ); + } + + bMoveLock = false; + break; + } + + case WM_LBUTTONDOWN: + { + if( GetKeyState( VK_MENU ) >= 0 ) break; + + // Dragging ON + iDragStartY = HIWORD( lp ); + iItemHeight = ( int )SendMessage( + WindowPlaylist, + LB_GETITEMHEIGHT, + 0, + 0 + ); + bDragging = true; + + return 0; + } + + case WM_LBUTTONUP: + // Dragging OFF + bDragging = false; + break; + + case WM_SYSKEYDOWN: + switch( wp ) // [Alt]+[...] + { + case VK_UP: + Playlist_MoveSel( true ); + break; + + case VK_DOWN: + Playlist_MoveSel( false ); + break; + } + break; + + case WM_CHAR: + case WM_KEYUP: + // SMALL LETTERS!!!!!! + switch( wp ) + { + case 'z': + case 'y': + case 'x': + case 'c': + case 'v': + case 'b': + case 'l': + return 0; + } + break; + + case WM_KEYDOWN: + { + const bool bShift = ( ( GetKeyState( VK_SHIFT ) & 0x8000 ) != 0 ); + const bool bControl = ( ( GetKeyState( VK_CONTROL ) & 0x8000 ) != 0 ); + // const bool bAlt = ( ( GetKeyState( VK_MENU ) & 0x8000 ) != 0 ); + + + switch( wp ) + { + case VK_LEFT: + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_REW5S, 0 ); + return 0; + + case VK_RIGHT: + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_FFWD5S, 0 ); + return 0; + + case VK_UP: + if( bControl && !bShift ) + { + // Move caret not selection + const int iCaretBefore = Playlist_GetCaretIndex(); + if( iCaretBefore > 0 ) + { + Playlist_SetCaretIndex( iCaretBefore - 1 ); + } + else if( ( iCaretBefore == 0 ) && bInfinitePlaylist ) + { + Playlist_SetCaretIndex( iMaxIndex ); + } + return 0; + } + else + { + if( bInfinitePlaylist ) + { + if( Playlist_GetCaretIndex() != 0 ) break; + Playlist_SelectSingle( iMaxIndex ); + return 0; // Or it will increase one more + } + } + break; + + case VK_DOWN: + if( bControl && !bShift ) + { + // Move caret not selection + const int iCaretBefore = Playlist_GetCaretIndex(); + if( ( iCaretBefore < iMaxIndex ) && ( iCaretBefore >= 0 ) ) + { + Playlist_SetCaretIndex( iCaretBefore + 1 ); + } + else if( ( iCaretBefore == iMaxIndex ) && bInfinitePlaylist ) + { + Playlist_SetCaretIndex( 0 ); + } + return 0; + } + else + { + if( bInfinitePlaylist ) + { + if( Playlist_GetCaretIndex() != iMaxIndex ) break; + Playlist_SelectSingle( 0 ); + return 0; // Or it will increase one more + } + } + break; + + case VK_SPACE: + if( bControl && !bShift ) + { + const int iCaret = Playlist_GetCaretIndex(); + if( iCaret == -1 ) return 0; + bool bSelected = Playlist_IsSelected( iCaret ); + Playlist_SelectModify( iCaret, !bSelected ); + } + return 0; + + case VK_DELETE: + { + if( bShift ) break; + + if( bControl ) + playlist->RemoveSelected( false ); + else + playlist->RemoveSelected( true ); + + break; + } + + case VK_RETURN: + iCurIndex = Playlist_GetCaretIndex(); + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON2, 0 ); + return 0; + + case 'Y': + case 'Z': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON1, 0 ); + return 0; + + case 'X': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON2, 0 ); + return 0; + + case 'C': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON3, 0 ); + return 0; + + case 'V': + // Todo modifiers pressed? -> fadeout/... + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON4, 0 ); + return 0; + + case 'B': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON5, 0 ); + return 0; + / + case 'J': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_JUMPFILE, 0 ); + return 0; + / + case 'L': + if( bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, bShift ? WINAMP_FILE_DIR : WINAMP_FILE_PLAY, 0 ); + return 0; + } + break; + } + + case WM_LBUTTONDBLCLK: + iCurIndex = Playlist_MouseToIndex(); + if( iCurIndex < 0 ) break; + Playback::Play(); + Playback::UpdateSeek(); + break; + + } + return CallWindowProc( WndprocPlaylistBackup, hwnd, message, wp, lp ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::Add( int iIndex, TCHAR * szDisplay, TCHAR * szFilename ) +{ + iMaxIndex++; + if( iIndex < 0 || iIndex > iMaxIndex ) iIndex = iMaxIndex; + if( iIndex <= iCurIndex ) + { + iCurIndex++; + } + + // Create entry data + PlaylistEntry * new_entry = new PlaylistEntry; + new_entry->szFilename = szFilename; + + iIndex = ( int )SendMessage( + WindowPlaylist, + LB_INSERTSTRING, // LB_ADDSTRING, + iIndex, + ( LPARAM )szDisplay + ); + + // Associate data + SendMessage( + WindowPlaylist, + LB_SETITEMDATA, + iIndex, + ( LPARAM )new_entry + ); + + return true; +} +*/ + + +//////////////////////////////////////////////////////////////////////////////// +/// Opens a dialog box and loads the playlist if is [true], +/// or saves the playlist if it is [false]. +//////////////////////////////////////////////////////////////////////////////// +bool Playlist_DialogBoth( bool bOpenOrSave ) +{ + TCHAR szFilters[] = TEXT( + "All files (*.*)\0*.*\0" + "Playlist files (*.M3U)\0*.m3u\0" + "\0" + ); + TCHAR szFilename[ MAX_PATH ] = TEXT( "\0" ); + + OPENFILENAME ofn; + memset( &ofn, 0, sizeof( OPENFILENAME ) ); + ofn.lStructSize = sizeof( OPENFILENAME ); + ofn.hwndOwner = WindowMain; + ofn.hInstance = g_hInstance; + ofn.lpstrFilter = szFilters; + ofn.lpstrCustomFilter = NULL; + ofn.nMaxCustFilter = 0; + ofn.nFilterIndex = 2; + ofn.lpstrFile = szFilename; + ofn.nMaxFile = MAX_PATH; + ofn.Flags = OFN_EXPLORER | + OFN_ENABLESIZING | + ( bOpenOrSave ? OFN_FILEMUSTEXIST : OFN_OVERWRITEPROMPT ) | + OFN_PATHMUSTEXIST | + OFN_HIDEREADONLY; + ofn.nMaxFileTitle = 0, + ofn.lpstrInitialDir = NULL; + ofn.lpstrTitle = ( bOpenOrSave ? TEXT( "Open playlist" ) : TEXT( "Save playlist" ) ); + + if( bOpenOrSave ) + { + if( !GetOpenFileName( &ofn ) ) return false; + } + else + { + if( !GetSaveFileName( &ofn ) ) return false; + } + + if( bOpenOrSave ) + { + // Open + const int iFilenameLen = ( int )_tcslen( szFilename ); + if( !_tcsncmp( szFilename + iFilenameLen - 3, TEXT( "m3u" ), 3 ) ) + { + // Playlist file + playlist->RemoveAll(); + Playlist::AppendPlaylistFile( szFilename ); + Playback::Play(); + + + Console::Append( TEXT( "Playlist.cpp:Playlist_DialogBoth() called Playback::Play()" ) ); + Console::Append( TEXT( " " ) ); + } + } + else + { + // TODO: Check extension, ask for appending if missing + + // Save + Playlist::ExportPlaylistFile( szFilename ); + } + + return true; + +} + + +//////////////////////////////////////////////////////////////////////////////// +/// Opens a dialog box and loads the selected playlist +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::DialogOpen() +{ + return Playlist_DialogBoth( true ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// Opens a dialog box and saves the playlist to the filename selected +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::DialogSaveAs() +{ + return Playlist_DialogBoth( false ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::AppendPlaylistFile( TCHAR * szFilename ) +{ + + + + // Open playlist file + HANDLE hFile = CreateFile( + szFilename, // LPCTSTR lpFileName + FILE_READ_DATA, // DWORD dwDesiredAccess + FILE_SHARE_READ, // DWORD dwShareMode + NULL, // LPSECURITY_ATTRIBUTES lpSecurityAttributes + OPEN_EXISTING, // DWORD dwCreationDisposition + FILE_ATTRIBUTE_NORMAL, // DWORD dwFlagsAndAttributes + NULL // HANDLE hTemplateFile + ); + + // This will happen if there is no Playlist.m3u + /* + if( hFile == INVALID_HANDLE_VALUE ) + { + // MessageBox( 0, TEXT( "Could not read playlist file" ), TEXT( "Error" ), MB_ICONERROR ); + //Console::Append( "We got INVALID_HANDLE_VALUE" ); + return false; + } + */ + // Disable this + //const bool bEmptyBefore = ( playlist->GetSize() == 0 ); + + +// ======================================================================================= + // Remove filename from so we can + // use it as relative directory root + TCHAR * szWalk = szFilename + _tcslen( szFilename ) - 1; + while( ( szWalk > szFilename ) && ( *szWalk != TEXT( '\\' ) ) ) szWalk--; + szWalk++; + *szWalk = TEXT( '\0' ); + + TCHAR * szBaseDir = szFilename; + const int iBaseDirLen = ( int )_tcslen( szBaseDir ); + + + DWORD iSizeBytes = GetFileSize( hFile, NULL ); + if( iSizeBytes <= 0 ) + { + CloseHandle( hFile ); + return false; + } + + // Allocate + char * rawdata = new char[ iSizeBytes + 1 ]; // One more so we can write '\0' on EOF + DWORD iBytesRead; + + // ======================================================================================= + // Read whole file + ReadFile( + hFile, // HANDLE hFile + rawdata, // LPVOID lpBuffer + iSizeBytes, // DWORD nNumberOfBytesToRead + &iBytesRead, // LPDWORD lpNumberOfBytesRead + NULL // LPOVERLAPPED lpOverlapped + ); + + if( iBytesRead < iSizeBytes ) + { + delete [] rawdata; + CloseHandle( hFile ); + + MessageBox( 0, TEXT( "Could not read whole file" ), TEXT( "Error" ), MB_ICONERROR ); + return false; + } + + // Parse file content + + // File must be + // * M3U + // * ANSI + + char * walk = rawdata; + const char * eof = rawdata + iSizeBytes; + + char * beg = rawdata; + char * end; + + while( true ) + { + // Find newline or eof + while( ( walk < eof ) && ( *walk != '\015' ) && ( *walk != '\012' ) ) walk++; + end = walk; + + if( ( end - beg > 2 ) && ( *beg != '#' ) ) + { + + TCHAR * szKeep; + if( beg[ 1 ] == ':' ) + { + // TODO: Better detection, network path? + + // Absolute path, skip this + /* + const int iLen = end - beg; + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "iLen <%i>" ), iLen ); + Console::Append( szBuffer ); + + szKeep = new TCHAR[ iLen + 1 ]; + ToTchar( szKeep, beg, iLen ); + szKeep[ iLen ] = TEXT( '\0' ); + */ + + } + else + { + // Skip initial so we don't get a double backslash in between + + + + while( ( beg[ 0 ] == '\\' ) && ( beg < end ) ) beg++; + + // Relative path + const int iSecondLen = end - beg; + szKeep = new TCHAR[ iBaseDirLen + iSecondLen + 1 ]; + memcpy( szKeep, szBaseDir, iBaseDirLen * sizeof( TCHAR ) ); + ToTchar( szKeep + iBaseDirLen, beg, iSecondLen ); + + szKeep[ iBaseDirLen + iSecondLen ] = TEXT( '\0' ); + + UnbloatFilename( szKeep, false ); + } + + // if( !Add( iMaxIndex + 1, szKeep, szKeep ) ) break; + playlist->PushBack( szKeep ); + } + + // Skip newlines + while( ( walk < eof ) && ( ( *walk == '\015' ) || ( *walk == '\012' ) ) ) walk++; + if( walk == eof ) + { + break; + } + + beg = walk; + } + + delete [] rawdata; + CloseHandle( hFile ); +/* + if( bEmptyBefore ) + { + iCurIndex = 0; + } +*/ + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::ExportPlaylistFile( TCHAR * szFilename ) +{ + // Open playlist file + HANDLE hFile = CreateFile( + szFilename, // LPCTSTR lpFileName + FILE_WRITE_DATA, // DWORD dwDesiredAccess + 0, // DWORD dwShareMode + NULL, // LPSECURITY_ATTRIBUTES lpSecurityAttributes + CREATE_ALWAYS, // DWORD dwCreationDisposition + FILE_ATTRIBUTE_NORMAL, // DWORD dwFlagsAndAttributes + NULL // HANDLE hTemplateFile + ); + + if( hFile == INVALID_HANDLE_VALUE ) + { + MessageBox( 0, TEXT( "Could not write playlist file" ), TEXT( "Error" ), MB_ICONERROR ); + return false; + } + + + // Remove filename from so we can + // use it as relative directory root + TCHAR * szWalk = szFilename + _tcslen( szFilename ) - 1; + while( ( szWalk > szFilename ) && ( *szWalk != TEXT( '\\' ) ) ) szWalk--; + szWalk++; + *szWalk = TEXT( '\0' ); + + TCHAR * szBaseDir = szFilename; + const int iBaseDirLen = ( int )_tcslen( szBaseDir ); + + + char * rawdata = new char[ ( playlist->GetMaxIndex() + 1 ) * ( MAX_PATH + 2 ) ]; + char * walk = rawdata; + + // Write playlist to buffer + const int iMaxMax = playlist->GetMaxIndex(); + for( int i = 0; i <= iMaxMax; i++ ) + { + // Get + TCHAR * szEntry = GetFilename( i ); + if( !szEntry ) break; + int iEntryLen = ( int )_tcslen( szEntry ); + + // Copy + TCHAR * szTemp = new TCHAR[ iEntryLen + 1 ]; + memcpy( szTemp, szEntry, iEntryLen * sizeof( TCHAR ) ); + szTemp[ iEntryLen ] = TEXT( '\0' ); + + // Convert + if( ApplyRootToFilename( szBaseDir, szTemp ) ) + { + // Update length or we are writing too much + iEntryLen = ( int )_tcslen( szTemp ); + } + + // Copy +#ifdef PA_UNICODE + ToAnsi( walk, szTemp, iEntryLen ); +#else + memcpy( walk, szTemp, iEntryLen ); +#endif + + delete [] szTemp; + + walk += iEntryLen; + memcpy( walk, "\015\012", 2 ); + walk += 2; + } + + const DWORD iSizeBytes = walk - rawdata; + DWORD iBytesRead; + WriteFile( + hFile, // HANDLE hFile, + rawdata, // LPCVOID lpBuffer, + iSizeBytes, // DWORD nNumberOfBytesToWrite, + &iBytesRead, // LPDWORD lpNumberOfBytesWritten, + NULL // LPOVERLAPPED lpOverlapped + ); + + if( iBytesRead < iSizeBytes ) + { + delete [] rawdata; + CloseHandle( hFile ); + + MessageBox( 0, TEXT( "Could not write whole file" ), TEXT( "Error" ), MB_ICONERROR ); + return false; + } + + delete [] rawdata; + CloseHandle( hFile ); + + return true; +} + + +/* +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::Append( TCHAR * szDisplay, TCHAR * szFilename ) +{ + return Add( iMaxIndex + 1, szDisplay, szFilename ); +} +*/ + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +TCHAR * Playlist::GetFilename( int iIndex ) +{ + // if( iIndex < 0 || iIndex > iMaxIndex ) return NULL; + +/* + PlaylistEntry * entry = ( PlaylistEntry * )SendMessage( + WindowPlaylist, + LB_GETITEMDATA, + iIndex, + 0 + ); + + return ( entry ? entry->szFilename : NULL ); + */ + //TCHAR * szFilename = "C:\Files\Spel och spelfusk\Console\Gamecube\Code\vgmstream (isolate ast)\Music\demo36_02.ast"; + //return szFilename; + return ( TCHAR * )playlist->Get( iIndex ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playlist::GetFilename( int iIndex, char * szAnsiFilename, int iChars ) +{ + // if( iIndex < 0 || iIndex > iMaxIndex ) return 0; + /* + PlaylistEntry * entry = ( PlaylistEntry * )SendMessage( + WindowPlaylist, + LB_GETITEMDATA, + iIndex, + 0 + ); + if( !entry || !entry->szFilename ) return 0; + + TCHAR * & szFilename = entry->szFilename; + */ + TCHAR * szFilename = ( TCHAR * )playlist->Get( iIndex ); + + const int iFilenameLen = ( int )_tcslen( szFilename ); + const int iCopyLen = ( iFilenameLen < iChars ) ? iFilenameLen : iChars; + +#ifdef PA_UNICODE + ToAnsi( szAnsiFilename, szFilename, iCopyLen ); +#else + memcpy( szAnsiFilename, szFilename, iCopyLen ); +#endif + + szAnsiFilename[ iCopyLen ] = '\0'; + return iCopyLen; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int Playlist::GetTitle( int iIndex, char * szAnsiTitle, int iChars ) +{ + // if( iIndex < 0 || iIndex > iMaxIndex ) return 0; + /* + TCHAR * szFilename = ( TCHAR * )SendMessage( + WindowPlaylist, + LB_GETITEMDATA, + iIndex, + 0 + ); + if( !szFilename ) return 0; + */ + TCHAR * szFilename = ( TCHAR * )playlist->Get( iIndex ); + + // Get extension + const int iFilenameLen = ( int )_tcslen( szFilename ); + TCHAR * szExt = szFilename + iFilenameLen - 1; + while( ( szExt > szFilename ) && ( *szExt != TEXT( '.' ) ) ) szExt--; + szExt++; + + // Get plugin for extension + map ::iterator iter = ext_map.find( szExt ); + if( iter == ext_map.end() ) return 0; + InputPlugin * input_plugin = iter->second; + +#ifdef PA_UNICODE + // Filename + char * szTemp = new char[ iFilenameLen + 1 ]; + ToAnsi( szTemp, szFilename, iFilenameLen ); + szTemp[ iFilenameLen ] = '\0'; + + // Ansi Title + char szTitle[ 2000 ] = "\0"; + int length_in_ms; + input_plugin->plugin->GetFileInfo( szTemp, szTitle, &length_in_ms ); + const int iTitleLen = strlen( szTitle ); + memcpy( szAnsiTitle, szTitle, iChars * sizeof( char ) ); + szTitle[ iChars ] = '\0'; +#else + char szTitle[ 2000 ] = "\0"; + int length_in_ms; + input_plugin->plugin->GetFileInfo( szFilename, szTitle, &length_in_ms ); + const int iTitleLen = ( int )strlen( szAnsiTitle ); + memcpy( szAnsiTitle, szTitle, iChars * sizeof( char ) ); + szTitle[ iChars ] = '\0'; +#endif + + return ( iTitleLen < iChars ) ? iTitleLen : iChars; +} + + +/* +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::SelectZero() +{ + SendMessage( + WindowPlaylist, + LB_SETSEL, + FALSE, + -1 + ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Playlist::SelectAll() +{ + SendMessage( + WindowPlaylist, + LB_SETSEL, + TRUE, + -1 + ); + return true; +} +*/ diff --git a/Externals/MusicMod/Player/Src/Playlist.h b/Externals/MusicMod/Player/Src/Playlist.h new file mode 100644 index 0000000000..82aaab7952 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Playlist.h @@ -0,0 +1,65 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_PLAYLIST_H +#define PA_PLAYLIST_H + + + +#include "Global.h" +#include "PlaylistControler.h" +#include "PlaylistView.h" + + +#define PLAINAMP_PL_REM_SEL 50004 +#define PLAINAMP_PL_REM_CROP 50005 + + + +extern HWND WindowPlaylist; + +extern PlaylistControler * playlist; + +namespace Playlist +{ + bool Create(); +/* + int GetCurIndex(); + int GetMaxIndex(); + bool SetCurIndex( int iIndex ); +*/ + TCHAR * GetFilename( int iIndex ); + + int GetFilename( int iIndex, char * szAnsiFilename, int iChars ); + int GetTitle( int iIndex, char * szAnsiTitle, int iChars ); + + bool DialogOpen(); + bool DialogSaveAs(); + + bool AppendPlaylistFile( TCHAR * szFilename ); + bool ExportPlaylistFile( TCHAR * szFilename ); +/* + bool Append( TCHAR * szDisplay, TCHAR * szFilename ); + bool Add( int iIndex, TCHAR * szDisplay, TCHAR * szFilename ); + + bool Clear(); // aka RemoveAll() + bool RemoveSelected(); + bool Crop(); // aka RemoveUnselected + + bool SelectZero(); + bool SelectAll(); +*/ +}; + + + +#endif // PA_PLAYLIST_H diff --git a/Externals/MusicMod/Player/Src/PlaylistControler.cpp b/Externals/MusicMod/Player/Src/PlaylistControler.cpp new file mode 100644 index 0000000000..a3b0f17a68 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlaylistControler.cpp @@ -0,0 +1,467 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "PlaylistControler.h" +#include "Config.h" +#include "Font.h" +#include "Console.h" + +bool bPlaylistFollow; +ConfBool cbPlaylistFollow( &bPlaylistFollow, TEXT( "PlaylistFollow" ), CONF_MODE_PUBLIC, true ); + + +void PlaylistControler::MoveSelected( int iDistance ) +{ + if( iDistance == -1 ) + { + if( ListView_GetItemState( _hView, 0, LVIS_SELECTED ) ) + { + // Cannot move upwards + return; + } + } + else if( iDistance == 1 ) + { + if( ListView_GetItemState( _hView, _database.GetMaxIndex(), LVIS_SELECTED ) ) + { + // Cannot move downwards + return; + } + } + else + { + // More distance maybe later + return; + } + + const int iFocus = ListView_GetNextItem( _hView, ( UINT )-1, LVIS_FOCUSED ); + + // Negative is to the top + LRESULT iBefore = 0; + LRESULT iAfter = -2; // Extra value to check after big for-loop + + for( ; ; ) + { + // Count + LRESULT iAfter = ListView_GetNextItem( _hView, iBefore - 1, LVNI_SELECTED ); + if( iAfter == -1 ) break; // No more selections selected + + LRESULT iFirst = iAfter; + + // Search end of selection block + iBefore = iAfter + 1; + for( ; ; ) + { + iAfter = ListView_GetNextItem( _hView, iBefore - 1, LVNI_SELECTED ); + if( iAfter == iBefore ) + { + // Keep searching + iBefore++; + } + else + { + // End found (iBefore is the first not selected) + const int iSelSize = iBefore - iFirst; + if( iDistance == -1 ) + { + // Updwards + const int iOldIndex = iFirst - 1; + const int iNewIndex = iOldIndex + iSelSize; + + TCHAR * szData = ( TCHAR * )_database.Get( iOldIndex ); + _database.Erase( iOldIndex ); + _database.Insert( iNewIndex, szData ); + + ListView_SetItemState( _hView, iOldIndex, LVIS_SELECTED, LVIS_SELECTED ); + ListView_SetItemState( _hView, iNewIndex, 0, LVIS_SELECTED ); + ListView_RedrawItems( _hView, iOldIndex, iNewIndex ); + } + else + { + // Downwards + const int iOldIndex = iFirst + iSelSize; + const int iNewIndex = iFirst; + + TCHAR * szData = ( TCHAR * )_database.Get( iOldIndex ); + _database.Erase( iOldIndex ); + _database.Insert( iNewIndex, szData ); + + ListView_SetItemState( _hView, iOldIndex, LVIS_SELECTED, LVIS_SELECTED ); + ListView_SetItemState( _hView, iNewIndex, 0, LVIS_SELECTED ); + ListView_RedrawItems( _hView, iNewIndex, iOldIndex ); + } + + iBefore++; + break; + } + } + } + + if( iAfter != -2 ) return; // Nothing was selected so nothing was moved + ListView_SetItemState( _hView, iFocus + iDistance, LVIS_FOCUSED, LVIS_FOCUSED ); + + Refresh(); +} + +int PlaylistControler::GetCurIndex() +{ + return _database.GetCurIndex(); +} + +int PlaylistControler::GetMaxIndex() +{ + return _database.GetMaxIndex(); +} + +int PlaylistControler::GetSize() +{ + return _database.GetSize(); +} + +void PlaylistControler::SetCurIndex( int iIndex ) +{ + const int iCurIndexBefore = _database.GetCurIndex(); + _database.SetCurIndex( iIndex ); + + // --------------------------------------------------------------------------------------- + // We disable the windows management + /* + if( bPlaylistFollow ) + { + ListView_SetItemState( _hView, ( UINT )-1, 0, LVIS_SELECTED | LVIS_FOCUSED ); + ListView_SetItemState( _hView, iIndex, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED ); + } + + if( iCurIndexBefore != -1 ) + ListView_RedrawItems( _hView, iCurIndexBefore, iCurIndexBefore ); + ListView_RedrawItems( _hView, iIndex, iIndex ); + */ + // --------------------------------------------------------------------------------------- +} + +// Returns on digit count change +bool PlaylistControler::FixDigitsMore() +{ + const int iCountAfter = _database.GetSize(); + const int iDigitsBefore = _iDigits; + while( iCountAfter > _iDigitMax ) + { + _iDigitMin *= 10; // 10 -> 100 + _iDigitMax = _iDigitMax * 10 + 9; // 99 -> 999 + _iDigits++; // 2 -> 3 + } + + + return ( ( _iDigits != iDigitsBefore ) + || ( iCountAfter == 1 ) ); // Force update when first item is inserted +} + +// Returns on digit count change +bool PlaylistControler::FixDigitsLess() +{ + const int iCountAfter = _database.GetSize(); + const int iDigitsBefore = _iDigits; + while( ( iCountAfter < _iDigitMin ) && ( _iDigits > 1 ) ) + { + _iDigitMin /= 10; // 999 -> 99 + _iDigitMax /= 10; // 100 -> 10 + _iDigits--; // 3 -> 2 + } + + return ( _iDigits != iDigitsBefore ); +} + + +void PlaylistControler::Refresh() +{ + AutosizeColumns(); + + const int iFirst = ListView_GetTopIndex( _hView ); + const int iLast = iFirst + ListView_GetCountPerPage( _hView ); + + ListView_RedrawItems( _hView, iFirst, iLast ); +} + +PlaylistControler::PlaylistControler( HWND hView, bool bEnableZeroPadding, int * piIndexSlave ) +{ + _hView = hView; + + _bZeroPadding = bEnableZeroPadding; + _iDigits = 1; + _iDigitMin = 1; + _iDigitMax = 9; + + _database.SetCurIndexSlave( piIndexSlave ); + + printf("PlaylistControler::PlaylistControler was called\n"); + + Refresh(); + + // TODO clear list view here??? +} + + +void PlaylistControler::PushBack( TCHAR * szText ) +{ + const int iSize = _database.GetMaxIndex(); + + _database.PushBack( szText ); + + // --------------------------------------------------------------------------------------- + // Disabled windows function + //ListView_SetItemCount( _hView, _database.GetSize() ); + + //if( FixDigitsMore() ) Refresh(); + // --------------------------------------------------------------------------------------- +} + +void PlaylistControler::Insert( int i, TCHAR * szText ) +{ + const int iSize = _database.GetMaxIndex(); + + _database.Insert( i, szText ); + ListView_SetItemCount( _hView, _database.GetSize() ); + + if( FixDigitsMore() ) Refresh(); +} + +void PlaylistControler::RemoveAll() +{ + _database.Clear(); + + // This is for the windows playlist, we don't use that + //ListView_DeleteAllItems( _hView ); + + if( FixDigitsLess() ) Refresh(); +} + +void PlaylistControler::RemoveSelected( bool bPositive ) +{ + SendMessage( _hView, WM_SETREDRAW, FALSE, 0 ); + + if( bPositive ) + { + LRESULT iWalk = 0; + for( ; ; ) + { + // MSDN: The specified item itself is excluded from the search. + iWalk = ListView_GetNextItem( _hView, iWalk - 1, LVNI_SELECTED ); + if( iWalk != -1 ) + { + _database.Erase( iWalk ); + ListView_DeleteItem( _hView, iWalk ); + } + else + { + break; + } + } + } + else + { + LRESULT iBefore = 0; + LRESULT iAfter = 0; + for( ; ; ) + { + // MSDN: The specified item itself is excluded from the search. + iAfter = ListView_GetNextItem( _hView, iBefore - 1, LVNI_SELECTED ); + if( iAfter != -1 ) + { + // is the first selected + // so we delete all before and restart + // at the beginning + const int iDelIndex = iBefore; + for( int i = iBefore; i < iAfter; i++ ) + { + _database.Erase( iDelIndex ); + ListView_DeleteItem( _hView, iDelIndex ); + } + + iBefore++; // Exclude the one we found selected right before + } + else + { + if( iBefore == 0 ) + { + // All selected + _database.Clear(); + ListView_DeleteAllItems( _hView ); + } + else + { + const int iSize = _database.GetSize(); + const int iDelIndex = iBefore; + for( int i = iBefore; i < iSize; i++ ) + { + _database.Erase( iDelIndex ); + ListView_DeleteItem( _hView, iDelIndex ); + } + } + + break; + } + } + } + + bool bRefresh = FixDigitsLess(); + + SendMessage( _hView, WM_SETREDRAW, TRUE, 0 ); + + if( bRefresh ) Refresh(); +} + +void PlaylistControler::SelectAll( bool bPositive ) +{ + ListView_SetItemState( _hView, ( UINT )-1, bPositive ? LVIS_SELECTED : 0, LVIS_SELECTED ); +} + +void PlaylistControler::SelectInvert() +{ + SendMessage( _hView, WM_SETREDRAW, FALSE, 0 ); + + const int iOneTooMuch = _database.GetSize(); + for( int i = 0; i < iOneTooMuch; i++ ) + { + ListView_SetItemState( + _hView, + i, + ( ListView_GetItemState( _hView, i, LVIS_SELECTED ) == LVIS_SELECTED ) + ? 0 + : LVIS_SELECTED, + LVIS_SELECTED + ); + + } + + SendMessage( _hView, WM_SETREDRAW, TRUE, 0 ); +} + +const TCHAR * PlaylistControler::Get( int i ) +{ + Console::Append( TEXT("We are in PlaylistControler::Get()") ); + + return _database.Get( i ); +} + +void PlaylistControler::Fill( LVITEM & request ) +{ + if( ( request.mask & LVIF_TEXT ) == 0 ) return; + if( request.iSubItem ) + { + // Text + _sntprintf( request.pszText, request.cchTextMax, TEXT( "%s" ), _database.Get( request.iItem ) ); + } + else + { + // Number + if( _bZeroPadding ) + { + TCHAR szFormat[ 6 ]; + _stprintf( szFormat, TEXT( "%%0%dd" ), _iDigits ); + _sntprintf( request.pszText, request.cchTextMax, szFormat, request.iItem + 1 ); + } + else + { + _sntprintf( request.pszText, request.cchTextMax, TEXT( "%d" ), request.iItem + 1 ); + } + } +} + +void PlaylistControler::EnableZeroPadding( bool bActive ) +{ + if( bActive == _bZeroPadding ) return; + + LVCOLUMN col; + memset( &col, 0, sizeof( LVCOLUMN ) ); + col.mask = LVCF_FMT; + + if( bActive ) + { + // TODO recalculation yes/no? + /* + int iSize = _database.GetSize(); + if( iSize != 0 ) + { + int iDigits = 0; + while( iSize > 0 ) + { + iSize /= 10; + iDigits++; + } + + _iDigits = iSize; + _iDigitMin = 1; + _iDigitMax = 9; + while( iSize-- > 0 ) + { + _iDigitMin *= 10; + _iDigitMax = _iDigitMax * 10 + 9; + } + } + else + { + _iDigits = 1; + _iDigitMin = 1; + _iDigitMax = 9; + } + */ + + col.fmt = LVCFMT_LEFT; + ListView_SetColumn( _hView, 0, &col ); + + Refresh(); + } + else + { + col.fmt = LVCFMT_RIGHT; + ListView_SetColumn( _hView, 0, &col ); + + Refresh(); + } + _bZeroPadding = bActive; +} + +void PlaylistControler::AutosizeColumns() +{ + RECT r; + if( !GetClientRect( _hView, &r ) ) return; + + if( _bZeroPadding ) + { + ListView_SetColumnWidth( _hView, 0, LVSCW_AUTOSIZE ); + const int iWidth = ListView_GetColumnWidth( _hView, 0 ); + ListView_SetColumnWidth( _hView, 1, r.right - r.left - iWidth ); + } + else + { + HDC hdc = GetDC( _hView ); + const HFONT hOldFont = ( HFONT )SelectObject( hdc, Font::Get() ); + SIZE size; + BOOL res = GetTextExtentPoint32( hdc, TEXT( "0" ), 1, &size ); + SelectObject( hdc, hOldFont ); + ReleaseDC( _hView, hdc ); + const int iWidth = res ? ( int )( size.cx * ( _iDigits + 0.25f ) ) : 120; + ListView_SetColumnWidth( _hView, 0, iWidth ); + ListView_SetColumnWidth( _hView, 1, r.right - r.left - iWidth ); + } +} + +void PlaylistControler::Resize( HWND hParent ) +{ + /* + RECT rc; + GetClientRect( hParent, &rc ); + MoveWindow( _hView, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, TRUE ); + */ + + AutosizeColumns(); +} diff --git a/Externals/MusicMod/Player/Src/PlaylistControler.h b/Externals/MusicMod/Player/Src/PlaylistControler.h new file mode 100644 index 0000000000..cc85f5e317 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlaylistControler.h @@ -0,0 +1,65 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PLAYLIST_CONTROLER_H +#define PLAYLIST_CONTROLER_H 1 + +#include +#include +#include +#include "PlaylistModel.h" + + + +// TODO: column width update on scrollbar show/hide + +class PlaylistControler +{ + PlaylistModel _database; + HWND _hView; + + bool _bZeroPadding; + int _iDigits; // 3 + int _iDigitMin; // 100 + int _iDigitMax; // 999 + +private: + bool FixDigitsMore(); + bool FixDigitsLess(); + void Refresh(); + void AutosizeColumns(); + +public: + PlaylistControler( HWND hView, bool bEnableZeroPadding, int * piIndexSlave ); + + void MoveSelected( int iDistance ); + + int GetCurIndex(); + int GetMaxIndex(); + int GetSize(); + void SetCurIndex( int iIndex ); + + void PushBack( TCHAR * szText ); + void Insert( int i, TCHAR * szText ); + void RemoveAll(); + void RemoveSelected( bool bPositive ); + void SelectAll( bool bPositive ); + void SelectInvert(); + + const TCHAR * Get( int i ); + + void Fill( LVITEM & request ); + void EnableZeroPadding( bool bActive ); + void Resize( HWND hParent ); +}; + +#endif // PLAYLIST_CONTROLER_H diff --git a/Externals/MusicMod/Player/Src/PlaylistModel.h b/Externals/MusicMod/Player/Src/PlaylistModel.h new file mode 100644 index 0000000000..1a914c8f3a --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlaylistModel.h @@ -0,0 +1,122 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + +// ======================================================================================= +#include "Console.h" +#include "Global.h" +// ======================================================================================= + + +#ifndef PLAYLIST_MODEL_H +#define PLAYLIST_MODEL_H 1 + +#include +#include +#include +using namespace std; + + +class PlaylistModel +{ + vector _database; + int _iCurIndex; + int * _piCurIndex; + +public: + PlaylistModel() + { + _piCurIndex = &_iCurIndex; + } + + PlaylistModel( int * piIndexSlave ) + { + _piCurIndex = piIndexSlave; + } + + void SetCurIndexSlave( int * piIndexSlave ) + { + // *piIndexSlave = *_piCurIndex; + _piCurIndex = piIndexSlave; + } + + // ======================================================================================= + // This is where the _database is populated + // ======================================================================================= + void PushBack( TCHAR * szText ) + { + Console::Append( TEXT( "PlaylistModel.h:_database.PushBack() was called " ) ); + + _database.push_back( szText ); // this is a API call + } + + void Insert( int i, TCHAR * szText ) + { + Console::Append( TEXT( "PlaylistModel.h:_database.Insert() was called " ) ); + + if( i <= *_piCurIndex ) ( *_piCurIndex )++; + _database.insert( _database.begin() + i, szText ); + } + + void Erase( int i ) + { + if( i < *_piCurIndex ) ( *_piCurIndex )--; + _database.erase( _database.begin() + i ); + } + + const TCHAR * Get( int i ) + { + //Console::Append( TEXT( "PlaylistModel.h:_database.Get() was called " ) ); + + if( 0 > i || i >= ( int )_database.size() ) + { + static const TCHAR * szError = TEXT( "INDEX OUT OF RANGE" ); + return szError; + } + + return _database[ i ]; + } + + void Clear() + { + _database.clear(); + *_piCurIndex = -1; + } + + int GetMaxIndex() + { + return _database.size() - 1; + } + + int GetCurIndex() + { + return *_piCurIndex; + } + + void SetCurIndex( int iIndex ) + { + + if( 0 > iIndex || iIndex >= ( int )_database.size() ) + { + INFO_LOG(AUDIO,"SetCurIndex > Return"); + return; + } + + *_piCurIndex = iIndex; + } + + int GetSize() + { + return _database.size(); + } +}; + + +#endif // PLAYLIST_MODEL_H diff --git a/Externals/MusicMod/Player/Src/PlaylistView.cpp b/Externals/MusicMod/Player/Src/PlaylistView.cpp new file mode 100644 index 0000000000..f2ba322d8d --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlaylistView.cpp @@ -0,0 +1,782 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + +// ======================================================================================= +#include "Playlist.h" +#include "Main.h" +#include "Status.h" +#include "Rebar.h" +#include "Playback.h" +#include "Config.h" +#include "Util.h" + +WNDPROC WndprocPlaylistBackup = NULL; +LRESULT CALLBACK WndprocPlaylist( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); +// ======================================================================================= + + +// ======================================================================================= +// This gives the values from the ini file. +// ConfBool = class +PlaylistControler * playlist = NULL; // extern + +bool bPlaylistEntryNumberZeroPadding; +ConfBool cbPlaylistEntryNumberZeroPadding( &bPlaylistEntryNumberZeroPadding, TEXT( "PlaylistEntryNumberZeroPadding" ), CONF_MODE_PUBLIC, true ); + +int iCurPlaylistPosition; +ConfInt ciCurPlaylistPosition( &iCurPlaylistPosition, TEXT( "CurPlaylistPosition" ), CONF_MODE_INTERNAL, -1 ); + +bool bInfinitePlaylist; +ConfBool cbInfinitePlaylist( &bInfinitePlaylist, TEXT( "InfinitePlaylist" ), CONF_MODE_PUBLIC, false ); +// ======================================================================================= + + +// ======================================================================================= + +void PlaylistView::Create() +{ + #ifndef NOGUI + + RECT ClientMain; + GetClientRect( WindowMain, &ClientMain ); + + const int iClientHeight = ClientMain.bottom - ClientMain.top; + const int iClientWidth = ClientMain.right - ClientMain.left; + const int iPlaylistHeight = iClientHeight - iRebarHeight - iStatusHeight; + + + //LoadCommonControls(); + + + DWORD dwStyle; + // HWND WindowPlaylist; + BOOL bSuccess = TRUE; + + dwStyle = WS_TABSTOP | + WS_CHILD | + WS_BORDER | + WS_VISIBLE | + LVS_AUTOARRANGE | // TODO + LVS_REPORT | + LVS_OWNERDATA | + LVS_NOCOLUMNHEADER ; + + WindowPlaylist = CreateWindowEx( WS_EX_CLIENTEDGE, // ex style + WC_LISTVIEW, // class name - defined in commctrl.h + TEXT( "" ), // dummy text + dwStyle, // style + 0, + iRebarHeight, // + -2, + iClientWidth, + iPlaylistHeight, + WindowMain, // parent + NULL, // ID + g_hInstance, // instance + NULL); // no extra data + + if(!WindowPlaylist) return; // TODO + + // This calls PlaylistControler::PlaylistControler() + playlist = new PlaylistControler( WindowPlaylist, bPlaylistEntryNumberZeroPadding, &iCurPlaylistPosition ); + + #else + + HWND WindowPlaylist = NULL; + playlist = new PlaylistControler( WindowPlaylist, bPlaylistEntryNumberZeroPadding, &iCurPlaylistPosition ); + + #endif + + // Exchange window procedure + //WndprocPlaylistBackup = ( WNDPROC )GetWindowLong( WindowPlaylist, GWL_WNDPROC ); + //if( WndprocPlaylistBackup != NULL ) + //{ + // SetWindowLong( WindowPlaylist, GWL_WNDPROC, ( LONG )WndprocPlaylist ); + //} + + + //ListView_SetExtendedListViewStyle( WindowPlaylist, LVS_EX_FULLROWSELECT ); // | LVS_EX_GRIDLINES ); + //playlist->Resize( WindowMain ); + + /* + * http://msdn.microsoft.com/library/default.asp?url=/library/en-us/shellcc/platform/commctls/listview/structures/lvcolumn.asp + * + * Remarks: + * If a column is added to a list-view control with index 0 (the leftmost column) + * and with LVCFMT_RIGHT or LVCFMT_CENTER specified, the text is not right-aligned + * or centered. The text in the index 0 column is left-aligned. Therefore if you + * keep inserting columns with index 0, the text in all columns are left-aligned. + * If you want the first column to be right-aligned or centered you can make a dummy + * column, then insert one or more columns with index 1 or higher and specify the + * alignment you require. Finally delete the dummy column. + */ + + //LV_COLUMN lvColumn; + //memset( &lvColumn, 0, sizeof( LV_COLUMN ) ); + //lvColumn.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT; + // + //// Number column (with dummy hack) + //lvColumn.fmt = LVCFMT_LEFT; + //lvColumn.cx = 0; + //lvColumn.pszText = TEXT( "" ); + //ListView_InsertColumn( WindowPlaylist, 0, &lvColumn ); + //lvColumn.fmt = LVCFMT_RIGHT; + //lvColumn.cx = 120; + //lvColumn.pszText = TEXT( "" ); + //ListView_InsertColumn( WindowPlaylist, 1, &lvColumn ); + //ListView_DeleteColumn( WindowPlaylist, 0 ); + + // Entry + //lvColumn.fmt = LVCFMT_LEFT; + //lvColumn.cx = 120; + //lvColumn.pszText = TEXT( "Filename" ); + //ListView_InsertColumn(WindowPlaylist, 1, &lvColumn); + + + + /* + stupid test code + + SCROLLINFO scrollinfo; + ZeroMemory( &scrollinfo, sizeof( SCROLLINFO ) ); + scrollinfo.cbSize = sizeof( SCROLLINFO ); + scrollinfo.fMask = 0; // SIF_DISABLENOSCROLL; + + if( !GetScrollInfo( WindowPlaylist, SB_VERT, &scrollinfo ) ) + { + MessageBox( 0, "ERROR", "", 0 ); + } + else + { + MessageBox( 0, "OKAY", "", 0 ); + scrollinfo.fMask = SIF_DISABLENOSCROLL; + SetScrollInfo( WindowPlaylist, SB_VERT, &scrollinfo, TRUE ); + } + + if( !ShowScrollBar( WindowPlaylist, SB_VERT, TRUE ) ) + { + MessageBox( 0, "ERROR ShowScrollBar", "", 0 ); + } + + + SCROLLBARINFO scrollbarinfo; + scrollbarinfo.cbSize = sizeof( SCROLLBARINFO ); + if( !GetScrollBarInfo( WindowPlaylist, OBJID_VSCROLL, &scrollbarinfo ) ) + { + MessageBox( 0, "ERROR GetScrollBarInfo", "", 0 ); + } + */ + +} + + + + + +// Dragging +static int iItemHeight = 15; +static int iDragStartY = 0; +static bool bDragging = false; + +// Liquid selection +static bool bLiquidSelecting = false; +static int iLastTouched = -1; + +// Liquid or range selection +static int iSelAnchor = -1; + +LRESULT CALLBACK WndprocPlaylist( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + /* + * Click, click and click + * + * [Alt] [Ctrl] [Shift] Action + * -------+---------+---------+----------------------- + * X | X | X | + * X | X | | + * X | | X | + * X | | | + * | X | X | Range selection + * | X | | Toggle selection + * | | X | Range selection + * | | | Single selection + * + * + * Click, hold and move + * + * [Alt] [Ctrl] [Shift] Action + * -------+---------+---------+----------------------- + * X | X | X | Selection move + * X | X | | Selection move + * X | | X | Selection move + * X | | | Selection move + * | X | X | + * | X | | + * | | X | + * | | | Liquid selection + */ + + static bool bCapturing = true; + + switch( message ) + { +/* + case WM_CAPTURECHANGED: + if( bCapturing && ( GetCapture() != WindowPlaylist ) ) + { + MessageBox( 0, TEXT( "Capture stolen" ), TEXT( "" ), 0 ); + } + break; +*/ + case WM_MOUSEMOVE: + if( bLiquidSelecting ) + { + LVHITTESTINFO hittest; + memset( &hittest, 0, sizeof( LVHITTESTINFO ) ); + hittest.pt.x = LOWORD( lp ); + hittest.pt.y = HIWORD( lp ); + const int iIndex = ( int )ListView_HitTest( WindowPlaylist, &hittest ); + if( iIndex == -1 ) return 0; + if( iIndex == iLastTouched ) return 0; + + // Note: Update this as early as possible! + // We cannot be sure this code is + // not called two or three times at the + // same time without losing much speed + // but this at least lowers the chance + const int iLastTouchedBackup = iLastTouched; + iLastTouched = iIndex; + + const bool bControl = ( ( GetKeyState( VK_CONTROL ) & 0x8000 ) != 0 ); + + + + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVIS_FOCUSED ); + ListView_SetItemState( WindowPlaylist, iIndex, LVIS_FOCUSED, LVIS_FOCUSED ); + + // Below anchor? + if( iIndex > iSelAnchor ) + { + if( iIndex > iLastTouchedBackup ) + { + // iSelAnchor + // .. + // iLastTouchedBackup + // .. + // >> iIndex << + if( iLastTouchedBackup > iSelAnchor ) + { + // Select downwards + for( int i = iLastTouchedBackup + 1; i <= iIndex; i++ ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + + // iLastTouchedBackup + // .. + // iSelAnchor + // .. + // >> iIndex << + else + { + // Unselect downwards + for( int i = iLastTouchedBackup; i < iSelAnchor; i++ ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + + // Select downwards + for( int i = iSelAnchor + 1; i <= iIndex; i++ ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + } + else // iIndex < iLastTouchedBackup + { + // iSelAnchor + // .. + // >> iIndex << + // .. + // iLastTouchedBackup + + // Unselect upwards + for( int i = iLastTouchedBackup; i > iIndex; i-- ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + } + + // Above anchor? + else if( iIndex < iSelAnchor ) + { + if( iIndex < iLastTouchedBackup ) + { + // >> iIndex << + // .. + // iSelAnchor + // .. + // iLastTouchedBackup + if( iIndex < iLastTouchedBackup ) + { + // Unselect upwards + for( int i = iLastTouchedBackup; i > iSelAnchor; i-- ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + + // Select upwards + for( int i = iSelAnchor - 1; i >= iIndex; i-- ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + + // >> iIndex << + // .. + // iLastTouchedBackup + // .. + // iSelAnchor + else // iIndex < iLastTouchedBackup + { + // Select upwards + for( int i = iLastTouchedBackup - 1; i >= iIndex; i-- ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + } + else // iIndex > iLastTouchedBackup + { + // iLastTouchedBackup + // .. + // >> iIndex << + // .. + // iSelAnchor + + // Unselect downwards + for( int i = iLastTouchedBackup; i < iIndex; i++ ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + } + + // At anchor + else // iIndex == iSelAnchor + { + if( iIndex < iLastTouchedBackup ) + { + // iSelAnchor / >> iIndex << + // .. + // iLastTouchedBackup + + // Unselect upwards + for( int i = iLastTouchedBackup; i > iSelAnchor; i-- ) + { + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + } + else // iIndex > iLastTouchedBackup + { + // iLastTouchedBackup + // .. + // iSelAnchor / >> iIndex << + + // Unselect downwards + for( int i = iLastTouchedBackup; i < iSelAnchor; i++ ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + } + } + else if( bDragging ) + { + static bool bMoveLock = false; + + if( bMoveLock ) return 0; + bMoveLock = true; + + const int y = HIWORD( lp ); + const int diff = y - iDragStartY; + if( abs( diff ) > iItemHeight / 2 ) + { + iDragStartY += ( ( diff > 0 ) ? iItemHeight : -iItemHeight ); + playlist->MoveSelected( ( diff > 0 ) ? +1 : -1 ); + } + + bMoveLock = false; + } + return 0; + + case WM_LBUTTONDOWN: + { + static int iLastClicked = -1; + static bool bLastClickNoneOrShift = true; + + + SetFocus( hwnd ); // TODO monitor focus loss + + LVHITTESTINFO hittest; + memset( &hittest, 0, sizeof( LVHITTESTINFO ) ); + GetCursorPos( &hittest.pt ); + ScreenToClient( hwnd, &hittest.pt ); + const int iIndex = ( int )ListView_HitTest( WindowPlaylist, &hittest ); + if( iIndex == -1 ) return 0; + + const bool bShift = ( ( GetKeyState( VK_SHIFT ) & 0x8000 ) != 0 ); + const bool bControl = ( ( GetKeyState( VK_CONTROL ) & 0x8000 ) != 0 ); + const bool bAlt = ( ( GetKeyState( VK_MENU ) & 0x8000 ) != 0 ); + + // [Shift] or [Shift]+[Ctrl]? + if( bShift ) + { + if( bAlt ) return 0; + + // Last click usable as selection anchor? + if( !bLastClickNoneOrShift ) + { + // Treat as normal click + iSelAnchor = iIndex; + iLastClicked = iIndex; + + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVIS_SELECTED | LVIS_FOCUSED ); + ListView_SetItemState( WindowPlaylist, iIndex, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED ); + + bLiquidSelecting = true; + bLastClickNoneOrShift = true; + + return 0; + } + + if( iIndex != iLastClicked ) + { + // Below anchor? + if( iIndex > iSelAnchor ) + { + // Below last click? + if( iIndex > iLastClicked ) + { + // Other side of anchor? + if( iLastClicked < iSelAnchor ) + { + // Unselect downwards + for( int i = iLastClicked; i < iSelAnchor; i++ ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + + // Select downwards + for( int i = iSelAnchor; i <= iIndex; i++ ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + + // Same side of anchor? + else + { + // Select downwards + for( int i = iLastClicked + 1; i <= iIndex; i++ ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + } + + // Above last click? + else // iIndex < iLastClicked + { + // Unselect upwards + for( int i = iLastClicked; i > iIndex; i-- ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + } + + // Above anchor? + else if( iIndex < iSelAnchor ) + { + // Above last clicked? + if( iIndex < iLastClicked ) + { + // Other side of anchor? + if( iLastClicked > iSelAnchor ) + { + // Unselect upwards + for( int i = iLastClicked; i > iSelAnchor; i-- ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + + // Select upwards + for( int i = iSelAnchor; i >= iIndex; i-- ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + + // Same side of anchor? + else + { + // Select upwards + for( int i = iLastClicked - 1; i >= iIndex; i-- ) + ListView_SetItemState( WindowPlaylist, i, LVIS_SELECTED, LVIS_SELECTED ); + } + } + + // Below last clicked? + else // iIndex > iLastClicked + { + // Unselect downwards + for( int i = iLastClicked; i < iIndex; i++ ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + } + + // At anchor + else // iIndex == iSelAnchor + { + if( iLastClicked < iSelAnchor ) + { + // Unselect downwards + for( int i = iLastClicked; i < iIndex; i++ ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + else if( iLastClicked > iSelAnchor ) + { + // Unselect upwards + for( int i = iLastClicked; i > iIndex; i-- ) + ListView_SetItemState( WindowPlaylist, i, 0, LVIS_SELECTED ); + } + } + + iLastClicked = iIndex; + + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVIS_FOCUSED ); + ListView_SetItemState( WindowPlaylist, iIndex, LVIS_FOCUSED, LVIS_FOCUSED ); + } + + bLastClickNoneOrShift = true; + } + + // [Ctrl]? + else if( bControl ) + { + if( bAlt ) return 0; + + iLastTouched = iIndex; + + // Toggle selection + const bool bSelected = ( ListView_GetItemState( WindowPlaylist, iIndex, LVIS_SELECTED ) == LVIS_SELECTED ); + ListView_SetItemState( WindowPlaylist, iIndex, bSelected ? 0 : LVIS_SELECTED, LVIS_SELECTED ); + + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVIS_FOCUSED ); + ListView_SetItemState( WindowPlaylist, iIndex, LVIS_FOCUSED, LVIS_FOCUSED ); + + bLastClickNoneOrShift = false; + } + + // [Alt]? + else if( bAlt ) + { + // Update item height + RECT rc; + GetClientRect( WindowPlaylist, &rc ); + const int iItemsPerPage = ListView_GetCountPerPage( WindowPlaylist ); + iItemHeight = ( rc.bottom - rc.top ) / iItemsPerPage; + + iDragStartY = hittest.pt.y; + bDragging = true; + + bCapturing = true; + SetCapture( WindowPlaylist ); + } + + // No modifiers? + else + { + iSelAnchor = iIndex; + iLastClicked = iIndex; + + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVIS_SELECTED | LVIS_FOCUSED ); + ListView_SetItemState( WindowPlaylist, iIndex, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED ); + + bLiquidSelecting = true; + bLastClickNoneOrShift = true; + + bCapturing = true; + SetCapture( WindowPlaylist ); + } + } + return 0; + + case WM_LBUTTONUP: + bLiquidSelecting = false; + bDragging = false; + + bCapturing = false; + ReleaseCapture(); + + return 0; + + case WM_SYSKEYDOWN: + switch( wp ) // [Alt]+[...] + { + case VK_UP: + playlist->MoveSelected( -1 ); + break; + + case VK_DOWN: + playlist->MoveSelected( +1 ); + break; + + } + break; + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + + case WM_CHAR: + case WM_KEYUP: + // SMALL LETTERS!!!!!! + switch( wp ) + { + case 'z': + case 'y': + case 'x': + case 'c': + case 'v': + case 'b': + case 'l': + return 0; + } + break; + + case WM_KEYDOWN: + { + const bool bShift = ( ( GetKeyState( VK_SHIFT ) & 0x8000 ) != 0 ); + const bool bControl = ( ( GetKeyState( VK_CONTROL ) & 0x8000 ) != 0 ); + + // Note: [Alt] goes to WM_SYSKEYDOWN instead + // const bool bAlt = ( ( GetKeyState( VK_MENU ) & 0x8000 ) != 0 ); + + + switch( wp ) + { + case VK_LEFT: + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_REW5S, 0 ); + return 0; + + case VK_RIGHT: + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_FFWD5S, 0 ); + return 0; + + case VK_UP: + if( bInfinitePlaylist ) + { + // First item has focus? + if( ListView_GetNextItem( WindowPlaylist, ( UINT )-1, LVNI_FOCUSED ) != 0 ) break; + + if( bControl && !bShift ) + { + // Move caret only + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVNI_FOCUSED ); + ListView_SetItemState( WindowPlaylist, playlist->GetMaxIndex(), LVNI_FOCUSED, LVNI_FOCUSED ); + return 0; + } + else + { + // Move Caret and selection + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVNI_FOCUSED ); + ListView_SetItemState( WindowPlaylist, playlist->GetMaxIndex() - 1, LVNI_FOCUSED, LVNI_FOCUSED ); + wp = VK_DOWN; + } + } + break; + + case VK_DOWN: + if( bInfinitePlaylist ) + { + // Last item has focus? + if( ListView_GetNextItem( WindowPlaylist, playlist->GetMaxIndex() - 1, LVNI_FOCUSED ) == -1 ) break; + + if( bControl && !bShift ) + { + // Move caret only + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVNI_FOCUSED ); + ListView_SetItemState( WindowPlaylist, 0, LVNI_FOCUSED, LVNI_FOCUSED ); + return 0; + } + else + { + // Workaround + ListView_SetItemState( WindowPlaylist, ( UINT )-1, 0, LVNI_FOCUSED ); + ListView_SetItemState( WindowPlaylist, 1, LVNI_FOCUSED, LVNI_FOCUSED ); + wp = VK_UP; + } + } + break; + + case VK_DELETE: + { + if( bShift ) break; + + if( bControl ) + playlist->RemoveSelected( false ); + else + playlist->RemoveSelected( true ); + + break; + } + + case VK_RETURN: + playlist->SetCurIndex( ListView_GetNextItem( WindowPlaylist, ( UINT )-1, LVIS_FOCUSED ) ); + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON2, 0 ); + return 0; + + case 'Y': + case 'Z': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON1, 0 ); + return 0; + + case 'X': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON2, 0 ); + return 0; + + case 'C': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON3, 0 ); + return 0; + + case 'V': + // Todo modifiers pressed? -> fadeout/... + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON4, 0 ); + return 0; + + case 'B': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_BUTTON5, 0 ); + return 0; + /* + case 'J': + if( bShift || bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, WINAMP_JUMPFILE, 0 ); + return 0; + */ + case 'L': + if( bControl ) return 0; + SendMessage( WindowMain, WM_COMMAND, bShift ? WINAMP_FILE_DIR : WINAMP_FILE_PLAY, 0 ); + return 0; + } + break; + } + + case WM_LBUTTONDBLCLK: + // iCurIndex = Playlist_MouseToIndex(); + + LVHITTESTINFO hittest; + memset( &hittest, 0, sizeof( LVHITTESTINFO ) ); + GetCursorPos( &hittest.pt ); + ScreenToClient( hwnd, &hittest.pt ); + const int iIndex = ( int )ListView_HitTest( WindowPlaylist, &hittest ); + if( iIndex == -1 ) break; + + playlist->SetCurIndex( iIndex ); + + Playback::Play(); + Playback::UpdateSeek(); + + break; + + } + + return CallWindowProc( WndprocPlaylistBackup, hwnd, message, wp, lp ); +} diff --git a/Externals/MusicMod/Player/Src/PlaylistView.h b/Externals/MusicMod/Player/Src/PlaylistView.h new file mode 100644 index 0000000000..5783ef9075 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PlaylistView.h @@ -0,0 +1,27 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PLAYLIST_VIEW_H +#define PLAYLIST_VIEW_H 1 + + +#include "Global.h" + +namespace PlaylistView +{ + void Create(); + + +}; + + +#endif // PLAYLIST_VIEW_H diff --git a/Externals/MusicMod/Player/Src/Plugin.cpp b/Externals/MusicMod/Player/Src/Plugin.cpp new file mode 100644 index 0000000000..fcad94bb74 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Plugin.cpp @@ -0,0 +1,63 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Plugin.h" +#include "InputPlugin.h" +#include "OutputPlugin.h" +#include "VisPlugin.h" +#include "DspPlugin.h" +#include "GenPlugin.h" + + + +vector plugins; // extern + +int Plugin::iWndprocHookCounter = 0; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +Plugin::Plugin( TCHAR * szDllpath ) +{ + hDLL = NULL; + szName = NULL; + iNameLen = 0; + + + iFullpathLen = ( int )_tcslen( szDllpath ); + szFullpath = new TCHAR[ iFullpathLen + 1 ]; + memcpy( szFullpath, szDllpath, iFullpathLen * sizeof( TCHAR ) ); + szFullpath[ iFullpathLen ] = TEXT( '\0' ); + + TCHAR * walk = szFullpath + iFullpathLen - 1; + while( ( *walk != TEXT( '\\') ) && ( walk >= szFullpath ) ) walk--; + if( *walk == TEXT( '\\') ) walk++; + + szFilename = walk; + iFilenameLen = iFullpathLen - ( walk - szFullpath ); + _tcslwr( szFilename ); + + plugins.push_back( this ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +Plugin::~Plugin() +{ + if( szFullpath ) delete [] szFullpath; + if( szName ) delete [] szName; +} diff --git a/Externals/MusicMod/Player/Src/Plugin.h b/Externals/MusicMod/Player/Src/Plugin.h new file mode 100644 index 0000000000..eab9cff84c --- /dev/null +++ b/Externals/MusicMod/Player/Src/Plugin.h @@ -0,0 +1,142 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_PLUGIN_H +#define PA_PLUGIN_H + + + +#include "Global.h" +#include + +using namespace std; + + +class Plugin; +extern vector plugins; + + + +enum PluginType +{ + PLUGIN_TYPE_INPUT, + PLUGIN_TYPE_OUTPUT, + PLUGIN_TYPE_VIS, + PLUGIN_TYPE_DSP, + PLUGIN_TYPE_GEN +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp plugin wrapper +//////////////////////////////////////////////////////////////////////////////// +class Plugin +{ +public: + Plugin( TCHAR * szDllpath ); + ~Plugin(); + + virtual bool Load() = 0; + virtual bool Unload() = 0; + + virtual TCHAR * GetTypeString() = 0; + virtual int GetTypeStringLen() = 0; + virtual PluginType GetType() = 0; + +// void AllowUnload( bool bAllow ) { bAllowUnload = bAllow; } + inline bool IsLoaded() { return ( hDLL != NULL ); } + virtual bool IsActive() = 0; + + inline TCHAR * GetFullpath() { return szFullpath; } + // inline int GetFullpathLen() { return iFilenameLen; } + + inline TCHAR * GetFilename() { return szFilename; } + inline int GetFilenameLen() { return iFilenameLen; } + inline TCHAR * GetName() { return szName; } + inline int GetNameLen() { return iNameLen; } + + template< class PluginKind > + static bool FindAll( TCHAR * szPath, TCHAR * szPattern, bool bKeepLoaded ); + +protected: + HINSTANCE hDLL; ///< Library handle + TCHAR * szName; ///< Name + int iNameLen; ///< Length of name (in characters) + + BOOL iHookerIndex; ///< Window hook index (0..HC-1). Only last can be unloaded + WNDPROC WndprocBackup; ///< Window procedure backup. Is restored when unloading. Only valid for + static int iWndprocHookCounter; ///< Number of window hooks (=HC) + +private: + TCHAR * szFullpath; ///< Full path e.g. "C:\test.dll" + TCHAR * szFilename; ///< Filename e.g. "test.dll" + int iFullpathLen; ///< Length of full path (in characters) + int iFilenameLen; ///< Length of filename (in characters) +}; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +template< class PluginKind > +bool Plugin::FindAll( TCHAR * szPath, TCHAR * szPattern, bool bKeepLoaded ) +{ + const int uPathLen = ( int )_tcslen( szPath ); + const int uPatternLen = ( int )_tcslen( szPattern ); + + TCHAR * szFullPattern = new TCHAR[ uPathLen + 1 + uPatternLen + 1 ]; + memcpy( szFullPattern, szPath, uPathLen * sizeof( TCHAR ) ); + szFullPattern[ uPathLen ] = TEXT( '\\' ); + memcpy( szFullPattern + uPathLen + 1, szPattern, uPatternLen * sizeof( TCHAR ) ); + szFullPattern[ uPathLen + 1 + uPatternLen ] = TEXT( '\0' ); + + + WIN32_FIND_DATA fd; + HANDLE hFind = FindFirstFile( szFullPattern, &fd ); + if( hFind == INVALID_HANDLE_VALUE ) + { + delete [] szFullPattern; + return false; + } + + do + { + const int iFilenameLen = ( int )_tcslen( fd.cFileName ); + + TCHAR * szFullpath = new TCHAR[ uPathLen + 1 + iFilenameLen + 1 ]; + memcpy( szFullpath, szPath, uPathLen * sizeof( TCHAR ) ); + szFullpath[ uPathLen ] = TEXT( '\\' ); + memcpy( szFullpath + uPathLen + 1, fd.cFileName, iFilenameLen * sizeof( TCHAR ) ); + szFullpath[ uPathLen + 1 + iFilenameLen ] = TEXT( '\0' ); + + +//////////////////////////////////////////////////////////////////////////////// + new PluginKind( szFullpath, bKeepLoaded ); +//////////////////////////////////////////////////////////////////////////////// + + + delete [] szFullpath; + } + while( FindNextFile( hFind, &fd ) ); + + FindClose( hFind ); + + + delete [] szFullPattern; + return true; +} + + + +#endif // PA_PLUGIN_H diff --git a/Externals/MusicMod/Player/Src/PluginManager.cpp b/Externals/MusicMod/Player/Src/PluginManager.cpp new file mode 100644 index 0000000000..ab5ac603fe --- /dev/null +++ b/Externals/MusicMod/Player/Src/PluginManager.cpp @@ -0,0 +1,1143 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "PluginManager.h" +#include "Plugin.h" +#include "InputPlugin.h" +#include "OutputPlugin.h" +#include "VisPlugin.h" +#include "DspPlugin.h" +#include "GenPlugin.h" +#include "Console.h" +#include "Main.h" +#include "Util.h" +#include "Config.h" + + + +#define MENU_INPUT_CONFIG 1 +#define MENU_INPUT_ABOUT 2 + +#define MENU_OUTPUT_CONFIG 1 +#define MENU_OUTPUT_ABOUT 2 +#define MENU_OUTPUT_SEP 3 +#define MENU_OUTPUT_ACTIVE 4 +#define MENU_OUTPUT_LOAD 5 + +#define MENU_GEN_CONFIG 1 +#define MENU_GEN_SEP 2 +#define MENU_GEN_LOAD 3 + + + +HWND WindowManager = NULL; // extern +HWND WindowListView = NULL; + + +WNDPROC WndprocListViewBackup = NULL; +LRESULT CALLBACK WndprocListView( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + +LRESULT CALLBACK WndprocManager( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + +HMENU input_menu = NULL; +HMENU output_menu = NULL; +HMENU gen_menu = NULL; + + + +bool bManagerGrid; +ConfBool cbManagerGrid( &bManagerGrid, "ManagerGrid", CONF_MODE_PUBLIC, true ); + +bool bManagerVisible; +WINDOWPLACEMENT WinPlaceManager; + +void WinPlaceManagerCallback( ConfVar * var ) +{ + if( !IsWindow( WindowManager ) ) return; + + GetWindowPlacement( WindowManager, &WinPlaceManager ); + + // MSDN: If the window identified by the hWnd parameter + // is maximized, the showCmd member is SW_SHOWMAXIMIZED. + // If the window is minimized, showCmd is SW_SHOWMINIMIZED. + // Otherwise, it is SW_SHOWNORMAL. + if( !bManagerVisible ) + { + WinPlaceManager.showCmd = SW_HIDE; + } +} + +RECT rManagerDefault = { 500, 400, 1000, 700 }; + +ConfWinPlaceCallback cwpcWinPlaceManager( + &WinPlaceManager, + TEXT( "WinPlaceManager" ), + &rManagerDefault, + WinPlaceManagerCallback +); + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +#define CLASSNAME_PAPMC TEXT( "PLAINAMP_PMC" ) +#define PAPMC_TITLE TEXT( "Plugin Manager" ) + +bool PluginManager::Build() +{ + LoadCommonControls(); + + // Register class + WNDCLASS wc = { + 0, // UINT style + WndprocManager, // WNDPROC lpfnWndProc + 0, // int cbClsExtra + 0, // int cbWndExtra + g_hInstance, // HINSTANCE hInstance + NULL, // HICON hIcon + LoadCursor( NULL, IDC_ARROW ), // HCURSOR hCursor + ( HBRUSH )COLOR_WINDOW, // HBRUSH hbrBackground + NULL, // LPCTSTR lpszMenuName + CLASSNAME_PAPMC // LPCTSTR lpszClassName + }; + + if( !RegisterClass( &wc ) ) return false; + + WindowManager = CreateWindowEx( + WS_EX_TOOLWINDOW, // DWORD dwExStyle + CLASSNAME_PAPMC, // LPCTSTR lpClassName + PAPMC_TITLE, // LPCTSTR lpWindowName + WS_OVERLAPPEDWINDOW | // DWORD dwStyle + WS_CLIPCHILDREN, // + rManagerDefault.left, // int x + rManagerDefault.top, // int y + rManagerDefault.right - rManagerDefault.left, // int nWidth + rManagerDefault.bottom - rManagerDefault.top, // int nHeight + NULL, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + RECT r; + GetClientRect( WindowManager, &r ); + + WindowListView = CreateWindowEx( + WS_EX_CLIENTEDGE, // DWORD dwExStyle + WC_LISTVIEW, // LPCTSTR lpClassName + NULL, // LPCTSTR lpWindowName + WS_VSCROLL | // DWORD dwStyle + LVS_REPORT | // + LVS_SORTASCENDING | // + LVS_SINGLESEL | // + WS_CHILD | // + WS_VISIBLE, // + 0, // int x + 0, // int y + r.right - r.left, // int nWidth + r.bottom - r.top, // int nHeight + WindowManager, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + if( !WindowListView ) return false; + + ListView_SetExtendedListViewStyle( + WindowListView, + LVS_EX_FULLROWSELECT | + ( bManagerGrid ? LVS_EX_GRIDLINES : 0 ) | + LVS_EX_HEADERDRAGDROP + ); + + // (0) File + LVCOLUMN lvc = { + LVCF_TEXT | LVCF_WIDTH | LVCF_FMT, // UINT mask; + LVCFMT_LEFT, // int fmt; + 100, // int cx; + TEXT( "File" ), // LPTSTR pszText; + 5, // int cchTextMax; + 0, // int iSubItem; + 0, // int iOrder; + 0 // int iImage; + }; + ListView_InsertColumn( WindowListView, 0, &lvc ); + + // (1) Status + lvc.pszText = TEXT( "Status" ); + lvc.cchTextMax = 7; + ListView_InsertColumn( WindowListView, 1, &lvc ); + + // (2) Name + lvc.cx = 220; + lvc.pszText = TEXT( "Name" ); + lvc.cchTextMax = 5; + ListView_InsertColumn( WindowListView, 2, &lvc ); + + + // (3) Type + lvc.pszText = TEXT( "Type" ); + lvc.cchTextMax = 5; + ListView_InsertColumn( WindowListView, 3, &lvc ); + + + bManagerVisible = ( WinPlaceManager.showCmd != SW_HIDE ); + SetWindowPlacement( WindowManager, &WinPlaceManager ); + + + // Exchange window procedure + WndprocListViewBackup = ( WNDPROC )GetWindowLong( WindowListView, GWL_WNDPROC ); + if( WndprocListViewBackup != NULL ) + { + SetWindowLong( WindowListView, GWL_WNDPROC, ( LONG )WndprocListView ); + } + + + // Build context menu + input_menu = CreatePopupMenu(); + AppendMenu( input_menu, MF_STRING, MENU_INPUT_CONFIG, TEXT( "Config" ) ); + AppendMenu( input_menu, MF_STRING, MENU_INPUT_ABOUT, TEXT( "About" ) ); + + output_menu = CreatePopupMenu(); + AppendMenu( output_menu, MF_STRING, MENU_OUTPUT_CONFIG, TEXT( "Config" ) ); + AppendMenu( output_menu, MF_STRING, MENU_OUTPUT_ABOUT, TEXT( "About" ) ); + AppendMenu( output_menu, MF_SEPARATOR | MF_GRAYED | MF_DISABLED, MENU_OUTPUT_SEP, NULL ); + AppendMenu( output_menu, MF_STRING, MENU_OUTPUT_ACTIVE, TEXT( "Active" ) ); + AppendMenu( output_menu, MF_STRING, MENU_OUTPUT_LOAD, TEXT( " " ) ); + + gen_menu = CreatePopupMenu(); + AppendMenu( gen_menu, MF_STRING, MENU_GEN_CONFIG, TEXT( "Config" ) ); + AppendMenu( gen_menu, MF_SEPARATOR | MF_GRAYED | MF_DISABLED, MENU_GEN_SEP, NULL ); + AppendMenu( gen_menu, MF_STRING, MENU_GEN_LOAD, TEXT( " " ) ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool PluginManager::Fill() +{ + if( !WindowListView ) return false; + + LVITEM lvi; + memset( &lvi, 0, sizeof( LVITEM ) ); + lvi.mask = LVIF_TEXT | LVIF_PARAM; + + Plugin * plugin; + int iIndex; + vector::iterator iter = plugins.begin(); + while( iter != plugins.end() ) + { +lvi.mask = LVIF_TEXT | LVIF_PARAM; + + plugin = *iter; + lvi.iItem = 0; + lvi.lParam = ( LPARAM )plugin; + + // (0) File + + lvi.iSubItem = 0; + lvi.pszText = plugin->GetFilename(); + lvi.cchTextMax = plugin->GetFilenameLen() + 1; + iIndex = ListView_InsertItem( WindowListView, &lvi ); + lvi.iItem = iIndex; +lvi.mask = LVIF_TEXT; + + // (1) Status + lvi.iSubItem = 1; + if( plugin->IsLoaded() ) + { + if( plugin->IsActive() ) + { + lvi.pszText = TEXT( "Active" ); + } + else + { + lvi.pszText = TEXT( "Loaded" ); + } + lvi.cchTextMax = 7; + } + else + { + lvi.pszText = TEXT( "Not loaded" ); + lvi.cchTextMax = 11; + } + ListView_SetItem( WindowListView, &lvi ); + + // (2) Name + lvi.iSubItem = 2; + lvi.pszText = plugin->GetName(); + lvi.cchTextMax = plugin->GetNameLen() + 1; + ListView_SetItem( WindowListView, &lvi ); + + // (3) Type + lvi.iSubItem = 3; + lvi.pszText = plugin->GetTypeString(); + lvi.cchTextMax = plugin->GetTypeStringLen() + 1; + ListView_SetItem( WindowListView, &lvi ); + + iter++; + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool PluginManager::Destroy() +{ + if( !WindowListView ) return false; + + DestroyWindow( WindowManager ); + DestroyMenu( input_menu ); + DestroyMenu( output_menu ); + DestroyMenu( gen_menu ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool PluginManager::Popup() +{ + if( !WindowListView ) return false; + if( !IsWindowVisible( WindowManager ) ) + ShowWindow( WindowManager, SW_SHOW ); + + SetActiveWindow( WindowManager ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void UpdatePluginStatus( Plugin * plugin, bool bLoaded, bool bActive ) +{ + LVFINDINFO lvfi = { + LVFI_PARAM, // UINT flags + NULL, // LPCTSTR psz + ( LPARAM )plugin, // LPARAM lParam + POINT(), // POINT pt + 0 // UINT vkDirection + }; + + int iIndex = ListView_FindItem( WindowListView, -1, &lvfi ); + if( iIndex != -1 ) + { + LVITEM lvi; + memset( &lvi, 0, sizeof( LVITEM ) ); + lvi.mask = LVIF_TEXT; + lvi.iItem = iIndex; + lvi.iSubItem = 1; + if( bLoaded ) + { + if( bActive ) + { + lvi.pszText = TEXT( "Active" ); + } + else + { + lvi.pszText = TEXT( "Loaded" ); + } + lvi.cchTextMax = 7; + } + else + { + lvi.pszText = TEXT( "Not loaded" ); + lvi.cchTextMax = 11; + } + ListView_SetItem( WindowListView, &lvi ); + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ContextMenuInput( InputPlugin * input, POINT * p ) +{ + BOOL iIndex = TrackPopupMenu( + input_menu, + TPM_LEFTALIGN | + TPM_TOPALIGN | + TPM_NONOTIFY | + TPM_RETURNCMD | + TPM_RIGHTBUTTON, + p->x, + p->y, + 0, + WindowManager, + NULL + ); + + switch( iIndex ) + { + case MENU_INPUT_ABOUT: + input->About( WindowManager ); + break; + + case MENU_INPUT_CONFIG: + input->Config( WindowManager ); + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ContextMenuOutput( OutputPlugin * output, POINT * p ) +{ + const bool bLoaded = output->IsLoaded(); + const bool bActive = output->IsActive(); + + EnableMenuItem( + output_menu, + MENU_OUTPUT_CONFIG, + bLoaded ? MF_ENABLED : MF_GRAYED | MF_DISABLED + ); + EnableMenuItem( + output_menu, + MENU_OUTPUT_ABOUT, + bLoaded ? MF_ENABLED : MF_GRAYED | MF_DISABLED + ); + EnableMenuItem( + output_menu, + MENU_OUTPUT_ACTIVE, + bLoaded ? MF_ENABLED : MF_GRAYED | MF_DISABLED + ); + CheckMenuItem( + output_menu, + MENU_OUTPUT_ACTIVE, + bActive ? MF_CHECKED : MF_UNCHECKED + ); + + + MENUITEMINFO mii = { 0 }; + mii.cbSize = sizeof( MENUITEMINFO ); + mii.fMask = MIIM_STATE | MIIM_STRING; + mii.fState = bActive ? MFS_GRAYED | MFS_DISABLED : MFS_ENABLED; + mii.dwTypeData = ( LPTSTR )( bLoaded ? TEXT( "Unload" ) : TEXT( "Load" ) ); + mii.cch = bLoaded ? 7 : 5; + + SetMenuItemInfo( + output_menu, // HMENU hMenu + MENU_OUTPUT_LOAD, // UINT uItem + FALSE, // BOOL fByPosition, + &mii // LPMENUITEMINFO lpmii + ); + + + BOOL iIndex = TrackPopupMenu( + output_menu, + TPM_LEFTALIGN | + TPM_TOPALIGN | + TPM_NONOTIFY | + TPM_RETURNCMD | + TPM_RIGHTBUTTON, + p->x, + p->y, + 0, + WindowManager, + NULL + ); + + switch( iIndex ) + { + case MENU_OUTPUT_CONFIG: + output->Config( WindowManager ); + break; + + case MENU_OUTPUT_ABOUT: + output->About( WindowManager ); + break; + + case MENU_OUTPUT_ACTIVE: + { + if( Playback::IsPlaying() ) + { + MessageBox( WindowManager, TEXT( "Cannot do this while playing!" ), TEXT( "Error" ), MB_ICONERROR ); + break; + } + + if( bActive ) + output->Stop(); + else + output->Start(); + + const bool bActiveNew = output->IsActive(); + if( bActiveNew != bActive ) + { + UpdatePluginStatus( output, bLoaded, bActiveNew ); + } + + break; + } + + case MENU_OUTPUT_LOAD: + { + if( bLoaded ) + output->Unload(); + else + output->Load(); + + const bool bLoadedNew = output->IsLoaded(); + if( bLoadedNew != bLoaded ) + { + UpdatePluginStatus( output, bLoadedNew, bActive ); + } + + break; + } + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ContextMenuVis( VisPlugin * vis, POINT * p ) +{ + HMENU vis_menu = CreatePopupMenu(); + + const bool bLoaded = vis->IsLoaded(); + const bool bPluginActive = vis->IsActive(); + + VisModule * mod; + HMENU * menus = NULL; + + const int iModCount = ( int )vis->modules.size(); + if( iModCount == 1 ) + { + // Single module + mod = vis->modules[ 0 ]; + const UINT uFlagsActive = MF_STRING | + ( mod->IsActive() ? MF_CHECKED : MF_UNCHECKED ) | + ( bLoaded ? MF_ENABLED : MF_DISABLED | MF_GRAYED ); + const UINT uFlagsConfig = MF_STRING | + ( bLoaded ? MF_ENABLED : MF_DISABLED | MF_GRAYED ); + AppendMenu( vis_menu, uFlagsActive, 1, TEXT( "Active" ) ); + AppendMenu( vis_menu, uFlagsConfig, 2, TEXT( "Config" ) ); + } + else + { + // Two or more + menus = new HMENU[ iModCount ]; + int iIndex = 0; + vector ::iterator iter = vis->modules.begin(); + while( iter != vis->modules.end() ) + { + mod = *iter; + + menus[ iIndex ] = CreatePopupMenu(); + UINT uFlags = MF_STRING | ( mod->IsActive() ? MF_CHECKED : MF_UNCHECKED ); + AppendMenu( menus[ iIndex ], uFlags, iIndex * 2 + 1, TEXT( "Active" ) ); + AppendMenu( menus[ iIndex ], MF_STRING, iIndex * 2 + 2, TEXT( "Config" ) ); + + uFlags = MF_STRING | MF_POPUP | ( ( !bLoaded || ( bPluginActive && !mod->IsActive() ) ) ? MF_GRAYED | MF_DISABLED : 0 ); + AppendMenu( vis_menu, uFlags, ( UINT_PTR )menus[ iIndex ], mod->GetName() ); + + iIndex++; + iter++; + } + } + + + + const int iPos = ( int )vis->modules.size() * 2 + 1; + AppendMenu( vis_menu, MF_SEPARATOR | MF_GRAYED | MF_DISABLED, iPos, NULL ); + + AppendMenu( + vis_menu, + MF_STRING | ( bPluginActive ? MF_DISABLED | MF_GRAYED : MF_ENABLED ), + iPos + 1, + bLoaded ? TEXT( "Unload" ) : TEXT( "Load" ) + ); + + + BOOL iIndex = TrackPopupMenu( + vis_menu, // HMENU hMenu + TPM_LEFTALIGN | // UINT uFlags + TPM_TOPALIGN | // . + TPM_NONOTIFY | // . + TPM_RETURNCMD | // . + TPM_RIGHTBUTTON, // . + p->x, // int x + p->y, // int y + 0, // int nReserved + WindowManager, // HWND hWnd + NULL // HWND prcRect + ); + + if( iIndex ) + { + if( iIndex >= iPos ) + { + // Load/unload + if( bLoaded ) + vis->Unload(); + else + vis->Load(); + + const bool bLoadedNew = vis->IsLoaded(); + + if( bLoadedNew != bLoaded ) + UpdatePluginStatus( vis, bLoadedNew, FALSE ); + } + else + { + int iWhich = ( iIndex - 1 ) / 2; + if( iIndex & 1 ) + { + // Active + const bool bPluginActive = vis->IsActive(); + + if( !vis->modules[ iWhich ]->IsActive() ) + vis->modules[ iWhich ]->Start(); + else + vis->modules[ iWhich ]->Stop(); + + const bool bPluginActiveNew = vis->IsActive(); + + if( bPluginActiveNew && ( bPluginActiveNew != bPluginActive ) ) + UpdatePluginStatus( vis, TRUE, bPluginActiveNew ); + } + else + { + // Config + vis->modules[ iWhich ]->Config(); + } + } + } + + DestroyMenu( vis_menu ); + if( iModCount > 1 ) + { + for( int i = 0; i < iModCount; i++ ) + DestroyMenu( menus[ i ] ); + delete [] menus; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ContextMenuDsp( DspPlugin * dsp, POINT * p ) +{ + HMENU context_menu = CreatePopupMenu(); + + const bool bLoaded = dsp->IsLoaded(); + const bool bPluginActive = dsp->IsActive(); + + DspModule * mod; + + HMENU * menus = NULL; + + HMENU * add_rem_menus = NULL; + + const int iEach = ( 3 * active_dsp_count + 2 ); + + const int iModCount = ( int )dsp->modules.size(); + if( iModCount == 1 ) + { + // Single module + mod = dsp->modules[ 0 ]; + + add_rem_menus = new HMENU[ 2 ]; + add_rem_menus[ 0 ] = CreatePopupMenu(); + add_rem_menus[ 1 ] = CreatePopupMenu(); + + int i; + TCHAR szHere[ 12 ]; + for( i = 0; i < active_dsp_count; i++ ) + { + TCHAR * szName = active_dsp_mods[ i ]->GetName(); + + // Entry for "Add" + _stprintf( szHere, TEXT( "Here (%i)" ), i + 1 ); + AppendMenu( + add_rem_menus[ 0 ], + MF_STRING, + 2 * i + 1, + szHere + ); + AppendMenu( + add_rem_menus[ 0 ], + MF_STRING | MF_DISABLED | MF_GRAYED, + 2 * i + 2, + szName + ); + + // Entry for "Remove" + AppendMenu( + add_rem_menus[ 1 ], + MF_STRING, + 2 * active_dsp_count + 2 + i, + szName + ); + } + _stprintf( szHere, TEXT( "Here (%i)" ), i + 1 ); + AppendMenu( + add_rem_menus[ 0 ], + MF_STRING, + 2 * i + 1, + szHere + ); + + + // Main entries + const UINT uFlagsAdd = MF_STRING | MF_POPUP | + ( ( bLoaded && !mod->IsActive() ) ? 0 : MF_DISABLED | MF_GRAYED ); + const UINT uFlagsRemove = MF_STRING | MF_POPUP | + ( ( bLoaded && active_dsp_count ) ? 0 : MF_GRAYED | MF_DISABLED ); + const UINT uFlagsConfig = MF_STRING | + ( ( bLoaded && mod->IsActive() ) ? 0 : MF_DISABLED | MF_GRAYED ); + + AppendMenu( + context_menu, + uFlagsAdd, + ( UINT_PTR )add_rem_menus[ 0 ], + TEXT( "Add" ) + ); + AppendMenu( + context_menu, + uFlagsRemove, + ( UINT_PTR )add_rem_menus[ 1 ], + TEXT( "Remove" ) + ); + AppendMenu( + context_menu, + uFlagsConfig, + 3 * active_dsp_count + 1 + 1, + TEXT( "Config" ) + ); + } + else + { + // Two or more + menus = new HMENU[ iModCount ]; + add_rem_menus = new HMENU[ 2 * iModCount ]; + + int iIndex = 0; + vector ::iterator iter = dsp->modules.begin(); + while( iter != dsp->modules.end() ) + { + mod = *iter; + + menus[ iIndex ] = CreatePopupMenu(); + + add_rem_menus[ 2 * iIndex ] = CreatePopupMenu(); + add_rem_menus[ 2 * iIndex + 1 ] = CreatePopupMenu(); + + int i; + TCHAR szHere[ 12 ]; + for( i = 0; i < active_dsp_count; i++ ) + { + TCHAR * szName = active_dsp_mods[ i ]->GetName(); + + // Entry for "Add" + _stprintf( szHere, TEXT( "Here (%i)" ), i + 1 ); + AppendMenu( + add_rem_menus[ 2 * iIndex ], + MF_STRING, + ( iIndex * iEach ) + ( 2 * i + 1 ), + szHere + ); + AppendMenu( + add_rem_menus[ 2 * iIndex ], + MF_STRING | MF_DISABLED | MF_GRAYED, + ( iIndex * iEach ) + ( 2 * i + 2 ), + szName + ); + + // Entry for "Remove" + AppendMenu( + add_rem_menus[ 2 * iIndex + 1 ], + MF_STRING, + ( iIndex * iEach ) + ( 2 * active_dsp_count + 2 + i ), + szName + ); + } + _stprintf( szHere, TEXT( "Here (%i)" ), i + 1 ); + AppendMenu( + add_rem_menus[ 2 * iIndex ], + MF_STRING, + ( iIndex * iEach ) + ( 2 * i + 1 ), + szHere + ); + + AppendMenu( + menus[ iIndex ], + MF_STRING | MF_POPUP | ( mod->IsActive() ? MF_DISABLED | MF_GRAYED : 0 ), + ( UINT_PTR )add_rem_menus[ 2 * iIndex ], + TEXT( "Add" ) + ); + AppendMenu( + menus[ iIndex ], + MF_STRING | MF_POPUP | ( active_dsp_count ? 0 : MF_GRAYED | MF_DISABLED ), + ( UINT_PTR )add_rem_menus[ 2 * iIndex + 1 ], + TEXT( "Remove" ) + ); + AppendMenu( + menus[ iIndex ], + MF_STRING | ( mod->IsActive() ? 0 : MF_DISABLED | MF_GRAYED ), + ( iIndex * iEach ) + ( 3 * active_dsp_count + 1 + 1 ), + TEXT( "Config" ) + ); + AppendMenu( context_menu, MF_STRING | MF_POPUP | ( bLoaded ? 0 : ( MF_DISABLED | MF_GRAYED ) ), ( UINT_PTR )menus[ iIndex ], mod->GetName() ); + + iIndex++; + iter++; + } + } + + + const int iPos = ( int )dsp->modules.size() * ( 3 * active_dsp_count + 1 + 1 ) + 1; + AppendMenu( context_menu, MF_SEPARATOR | MF_GRAYED | MF_DISABLED, iPos, NULL ); + + UINT uFlags = MF_STRING; + + AppendMenu( + context_menu, + uFlags | ( bPluginActive ? MF_DISABLED | MF_GRAYED : MF_ENABLED ), + iPos + 1, + bLoaded ? TEXT( "Unload" ) : TEXT( "Load" ) + ); + + + BOOL iIndex = TrackPopupMenu( + context_menu, // HMENU hMenu + TPM_LEFTALIGN | // UINT uFlags + TPM_TOPALIGN | // . + TPM_NONOTIFY | // . + TPM_RETURNCMD | // . + TPM_RIGHTBUTTON, // . + p->x, // int x + p->y, // int y + 0, // int nReserved + WindowManager, // HWND hWnd + NULL // HWND prcRect + ); + + /* + + Example menu + + Mod1 + Add + Here [ 1] + Active1 [ 2] + Here [ 3] + Active2 [ 4] + Here [ 5] + Remove + Active1 [ 6] + Active2 [ 7] + Config [ 8] + Mod2 + Add + Here [ 9] + Active1 [10] + ... + --- + Load/Unload [11] + + */ + + + if( iIndex ) + { + if( iIndex >= iPos ) + { + // Load/unload + if( bLoaded ) + dsp->Unload(); + else + dsp->Load(); + + const bool bLoadedNew = dsp->IsLoaded(); + + if( bLoadedNew != bLoaded ) + UpdatePluginStatus( dsp, bLoadedNew, FALSE ); + } + else + { + int iWhichMod = ( iIndex - 1 ) / iEach; + int iRem = iIndex % iEach; + if( iRem == 0 ) + { + // Config + dsp->modules[ iWhichMod ]->Config(); + + } + else if( iRem < 2 * active_dsp_count + 2 ) + { + // Add module + const bool bPluginActive = dsp->IsActive(); + + dsp->modules[ iWhichMod ]->Start( iRem / 2 ); + + const bool bPluginActiveNew = dsp->IsActive(); + if( bPluginActiveNew != bPluginActive ) + { + UpdatePluginStatus( dsp, TRUE, bPluginActiveNew ); + } + } + else + { + // Remove module + const bool bPluginActive = dsp->IsActive(); + + iWhichMod = iRem - ( 2 * active_dsp_count + 2 ); +/* + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "REM <%i> <%i>" ), iIndex, iWhichMod ); + Console::Append( szBuffer ); +*/ + active_dsp_mods[ iWhichMod ]->Stop(); + + const bool bPluginActiveNew = dsp->IsActive(); + if( bPluginActiveNew != bPluginActive ) + { + UpdatePluginStatus( dsp, TRUE, bPluginActiveNew ); + } + } + } + } + + DestroyMenu( context_menu ); + if( iModCount > 1 ) + { + for( int i = 0; i < iModCount; i++ ) + DestroyMenu( menus[ i ] ); + delete [] menus; + } + + for( int j = 0; j < iModCount; j++ ) + DestroyMenu( add_rem_menus[ j ] ); + delete [] add_rem_menus; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ContextMenuGen( GenPlugin * gen, POINT * p ) +{ + const bool bLoaded = gen->IsLoaded(); + + EnableMenuItem( + gen_menu, + MENU_GEN_CONFIG, + bLoaded ? MF_ENABLED : MF_GRAYED | MF_DISABLED + ); + + + MENUITEMINFO mii = { 0 }; + mii.cbSize = sizeof( MENUITEMINFO ); + mii.fMask = MIIM_STATE | MIIM_STRING; + mii.fState = gen->AllowRuntimeUnload() ? MFS_ENABLED : MFS_GRAYED | MFS_DISABLED; + mii.dwTypeData = ( LPTSTR )( bLoaded ? TEXT( "Unload" ) : TEXT( "Load and activate" ) ); + mii.cch = bLoaded ? 7 : 18; + + SetMenuItemInfo( + gen_menu, // HMENU hMenu + MENU_GEN_LOAD, // UINT uItem + FALSE, // BOOL fByPosition, + &mii // LPMENUITEMINFO lpmii + ); + + + BOOL iIndex = TrackPopupMenu( + gen_menu, + TPM_LEFTALIGN | + TPM_TOPALIGN | + TPM_NONOTIFY | + TPM_RETURNCMD | + TPM_RIGHTBUTTON, + p->x, + p->y, + 0, + WindowManager, + NULL + ); + + switch( iIndex ) + { + case MENU_GEN_CONFIG: + gen->Config(); + break; + + case MENU_GEN_LOAD: + { + if( bLoaded ) + gen->Unload(); + else + gen->Load(); + + const bool bLoadedNew = gen->IsLoaded(); + if( bLoadedNew != bLoaded ) + { + UpdatePluginStatus( gen, bLoadedNew, bLoadedNew ); + } + + break; + } + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ContextMenu( Plugin * plugin, POINT * p ) +{ + if( !plugin ) return; + + switch( plugin->GetType() ) + { + case PLUGIN_TYPE_INPUT: + ContextMenuInput( ( InputPlugin * )plugin, p ); + break; + + case PLUGIN_TYPE_OUTPUT: + ContextMenuOutput( ( OutputPlugin * )plugin, p ); + break; + + case PLUGIN_TYPE_VIS: + ContextMenuVis( ( VisPlugin * )plugin, p ); + break; + + case PLUGIN_TYPE_DSP: + ContextMenuDsp( ( DspPlugin * )plugin, p ); + break; + + case PLUGIN_TYPE_GEN: + ContextMenuGen( ( GenPlugin * )plugin, p ); + break; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocListView( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + static bool bContextOpen = false; + + switch( message ) + { + case WM_CONTEXTMENU: + { + if( ( HWND )wp != WindowListView ) break; + if( bContextOpen ) break; + bContextOpen = true; +//////////////////////////////////////////////////////////////////////////////// + + POINT p = { LOWORD( lp ), HIWORD( lp ) }; + ScreenToClient( WindowListView, &p ); + + // Which item? + LVHITTESTINFO hti; + memset( &hti, 0, sizeof( LVHITTESTINFO ) ); + hti.pt = p; + ListView_HitTest( WindowListView, &hti ); + + // Which plugin? + LVITEM lvi; + memset( &lvi, 0, sizeof( LVITEM ) ); + lvi.mask = LVIF_PARAM; + lvi.iItem = hti.iItem; + lvi.iSubItem = 0; + + if( !ListView_GetItem( WindowListView, &lvi ) ) + { + bContextOpen = false; + break; + } + Plugin * plugin = ( Plugin * )lvi.lParam; + + // Context menu + p.x = LOWORD( lp ); + p.y = HIWORD( lp ); + ContextMenu( plugin, &p ); + +//////////////////////////////////////////////////////////////////////////////// + bContextOpen = false; + break; + } + } + return CallWindowProc( WndprocListViewBackup, hwnd, message, wp, lp ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocManager( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + switch( message ) + { + case WM_SIZE: + { + // Resize children + RECT client; + GetClientRect( WindowManager, &client ); + + if( WindowListView ) + MoveWindow( WindowListView, 0, 0, client.right - client.left, client.bottom - client.top, TRUE ); + break; + } + + case WM_SYSCOMMAND: + if( ( wp & 0xFFF0 ) == SC_CLOSE ) + { + ShowWindow( hwnd, SW_HIDE ); + return 0; + } + break; + + case WM_DESTROY: + cwpcWinPlaceManager.TriggerCallback(); + cwpcWinPlaceManager.RemoveCallback(); + break; + + case WM_SHOWWINDOW: + bManagerVisible = ( wp == TRUE ); + break; + } + return DefWindowProc( hwnd, message, wp, lp ); +} diff --git a/Externals/MusicMod/Player/Src/PluginManager.h b/Externals/MusicMod/Player/Src/PluginManager.h new file mode 100644 index 0000000000..e68fb6d311 --- /dev/null +++ b/Externals/MusicMod/Player/Src/PluginManager.h @@ -0,0 +1,42 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_PLUGIN_MANAGER_H +#define PA_PLUGIN_MANAGER_H + + + +#include "Global.h" +#include "Plugin.h" + + + +extern HWND WindowManager; + + + +void UpdatePluginStatus( Plugin * plugin, bool bLoaded, bool bActive ); + + + +namespace PluginManager +{ + bool Build(); + bool Fill(); + bool Destroy(); + + bool Popup(); +}; + + + +#endif // PA_PLUGIN_MANAGER_H diff --git a/Externals/MusicMod/Player/Src/Prefs.cpp b/Externals/MusicMod/Player/Src/Prefs.cpp new file mode 100644 index 0000000000..359fa40b5f --- /dev/null +++ b/Externals/MusicMod/Player/Src/Prefs.cpp @@ -0,0 +1,565 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Prefs.h" +#include "Util.h" +#include "Font.h" +#include "Console.h" +#include + +using namespace std; + + +#define CLASSNAME_PREFS TEXT( "CLASSNAME_PREFS" ) +#define TITLE_PREFS TEXT( "Preferences" ) + + +#define PAGE_WIDTH 409 +#define PAGE_HEIGHT 400 + + +#define GAP_LEFT 4 +#define GAP_RIGHT 5 +#define GAP_TOP 4 +#define GAP_BOTTOM 5 +#define SPLITTER_WIDTH 6 +#define SPLITTER_HEIGHT 6 + +#define BOTTOM_SPACE 36 + +#define BUTTON_WIDTH 80 + + +#define PREFS_WIDTH 606 +#define PREFS_HEIGHT ( GAP_TOP + PAGE_HEIGHT + BOTTOM_SPACE ) + + + +struct PrefRecCompare +{ + bool operator()( const prefsDlgRec * a, const prefsDlgRec * b ) const + { + if( a->hInst < b->hInst ) return true; + if( a->dlgID < b->dlgID ) return true; + return strcmp( a->name, b->name ) < 0; + } +}; + + +map rec_to_item; + + +struct AllWeNeed +{ + prefsDlgRec * PageData; + HWND hwnd; +}; + + + +HWND WindowPrefs = NULL; +HWND WindowPage = NULL; +HWND WindowTree = NULL; +HWND ButtonClose = NULL; + +HTREEITEM root = NULL; + +HWND hCurrentPage = NULL; +// bool bKeepFocus = false; + + +LRESULT CALLBACK WndprocPrefs( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); +LRESULT CALLBACK WndprocTree( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); +WNDPROC WndprocTreeBackup = NULL; + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Prefs::Create() +{ + if( WindowPrefs ) return false; + + // Register class + WNDCLASS wc = { + 0, // UINT style + WndprocPrefs, // WNDPROC lpfnWndProc + 0, // int cbClsExtra + 0, // int cbWndExtra + g_hInstance, // HINSTANCE hInstance + NULL, // HICON hIcon + LoadCursor( NULL, IDC_ARROW ), // HCURSOR hCursor + ( HBRUSH )COLOR_WINDOW, // HBRUSH hbrBackground + NULL, // LPCTSTR lpszMenuName + CLASSNAME_PREFS // LPCTSTR lpszClassName + }; + + if( !RegisterClass( &wc ) ) return false; + + const int cxScreen = GetSystemMetrics( SM_CXFULLSCREEN ); + const int cyScreen = GetSystemMetrics( SM_CYFULLSCREEN ); + + // Create window + WindowPrefs = CreateWindowEx( + WS_EX_WINDOWEDGE | // DWORD dwExStyle + WS_EX_TOOLWINDOW, // + CLASSNAME_PREFS, // LPCTSTR lpClassName + TITLE_PREFS, // LPCTSTR lpWindowName + WS_OVERLAPPED | // DWORD dwStyle +// WS_VISIBLE | // + WS_CLIPCHILDREN | // + WS_SYSMENU, // + ( cxScreen - PREFS_WIDTH ) / 2, // int x + ( cyScreen - PREFS_HEIGHT ) / 2, // int y + PREFS_WIDTH, // int nWidth + PREFS_HEIGHT, // int nHeight + NULL, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + + if( !WindowPrefs ) + { + UnregisterClass( CLASSNAME_PREFS, g_hInstance ); + return false; + } + + + RECT r; + GetClientRect( WindowPrefs, &r ); + + +//////////////////////////////////////////////////////////////////////////////// + const int iWidth = PREFS_WIDTH * 2 - r.right; + const int iHeight = PREFS_HEIGHT * 2 - r.bottom; + SetWindowPos( + WindowPrefs, + NULL, + ( cxScreen - iWidth ) / 2, + ( cyScreen - iHeight ) / 2, + iWidth, + iHeight, + SWP_NOZORDER + ); + GetClientRect( WindowPrefs, &r ); +//////////////////////////////////////////////////////////////////////////////// + + + LoadCommonControls(); + + const int iTreeWidth = ( r.right - r.left ) - PAGE_WIDTH - GAP_LEFT - GAP_RIGHT - SPLITTER_WIDTH; + + WindowTree = CreateWindowEx( + WS_EX_CLIENTEDGE, // DWORD dwExStyle + WC_TREEVIEW, // LPCTSTR lpClassName + NULL, // LPCTSTR lpWindowName + WS_VSCROLL | // DWORD dwStyle + WS_VISIBLE | // + WS_CHILD | // + WS_TABSTOP | // + TVS_HASLINES | // + TVS_LINESATROOT | // + TVS_HASBUTTONS | // + TVS_SHOWSELALWAYS | // + TVS_DISABLEDRAGDROP | // + TVS_NONEVENHEIGHT, // + GAP_LEFT, // int x + GAP_TOP, // int y + iTreeWidth, // int nWidth + PAGE_HEIGHT, // int nHeight + WindowPrefs, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + if( !WindowTree ) + { + ; //... + } + + Font::Apply( WindowTree ); + + // Exchange window procedure + WndprocTreeBackup = ( WNDPROC )GetWindowLong( WindowTree, GWL_WNDPROC ); + if( WndprocTreeBackup != NULL ) + { + SetWindowLong( WindowTree, GWL_WNDPROC, ( LONG )WndprocTree ); + } + + + const int iPageLeft = GAP_LEFT + iTreeWidth + SPLITTER_WIDTH; + + WindowPage = CreateWindowEx( + 0, // DWORD dwExStyle + TEXT( "Static" ), // LPCTSTR lpClassName + TEXT( "Nothing to see here" ), // LPCTSTR lpWindowName + WS_TABSTOP | // DWORD dwStyle + WS_VISIBLE | // + WS_CHILD | // + SS_CENTER | // + SS_CENTERIMAGE, // + iPageLeft, // int x + GAP_TOP, // int y + PAGE_WIDTH, // int nWidth + PAGE_HEIGHT, // int nHeight + WindowPrefs, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + if( !WindowPage ) + { + ; //... + } + + Font::Apply( WindowPage ); + + + const int iButtonLeft = ( r.right - r.left ) - GAP_RIGHT - BUTTON_WIDTH - 1; + const int iButtonTop = ( r.bottom - r.top ) - BOTTOM_SPACE + SPLITTER_HEIGHT; + + + ButtonClose = CreateWindowEx( + 0, // DWORD dwExStyle + TEXT( "Button" ), // LPCTSTR lpClassName + TEXT( "Close" ), // LPCTSTR lpWindowName + WS_TABSTOP | // DWORD dwStyle + WS_VISIBLE | // + WS_CHILD | // + BS_PUSHBUTTON | // + BS_TEXT | // + BS_VCENTER, // + iButtonLeft, // int x + iButtonTop, // int y + BUTTON_WIDTH, // int nWidth + BOTTOM_SPACE - SPLITTER_HEIGHT - GAP_BOTTOM, // int nHeight + WindowPrefs, // HWND hWndParent + NULL, // HMENU hMenu + g_hInstance, // HINSTANCE hInstance + NULL // LPVOID lpParam + ); + + Font::Apply( ButtonClose ); + + AllWeNeed * awn_root = new AllWeNeed; +// memset( &awn_root->PageData, 0, sizeof( prefsDlgRec ) ); + awn_root->PageData = NULL; + awn_root->hwnd = NULL; + + TV_INSERTSTRUCT tvi = { + root, // HTREEITEM hParent + TVI_SORT, // HTREEITEM hInsertAfter + { + TVIF_PARAM | TVIF_STATE | TVIF_TEXT, // UINT mask + NULL, // HTREEITEM hItem + TVIS_EXPANDED | TVIS_SELECTED, // UINT state + 0, // UINT stateMask + TEXT( "General" ), // LPSTR pszText + 8, // int cchTextMax + 0, // int iImage + 0, // int iSelectedImage + 0, // int cChildren + ( LPARAM )awn_root // LPARAM lParam + } + }; + + root = TreeView_InsertItem( WindowTree, &tvi ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Prefs::Destroy() +{ + if( !WindowPrefs ) return false; + + DestroyWindow( WindowPrefs ); + UnregisterClass( CLASSNAME_PREFS, g_hInstance ); + WindowPrefs = NULL; + + DestroyWindow( WindowTree ); + WindowTree = NULL; + + DestroyWindow( WindowPage ); + WindowPage = NULL; + + DestroyWindow( ButtonClose ); + ButtonClose = NULL; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Prefs_Show( HTREEITEM item ) +{ + if( !WindowPrefs ) return false; + + // Select and load associated page + TreeView_SelectItem( WindowTree, item ); + + if( !IsWindowVisible( WindowPrefs ) ) + { + ShowWindow( WindowPrefs, SW_SHOW ); + } + + SetActiveWindow( WindowPrefs ); + SetFocus( WindowTree ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Prefs::Show() +{ + return Prefs_Show( root ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Prefs::Show( prefsDlgRec * PageData ) +{ + map::iterator iter = rec_to_item.find( PageData ); + if( iter != rec_to_item.end() ) + { + return Prefs_Show( iter->second ); + } + else + { + return Prefs_Show( root ); + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Prefs::AddPage( prefsDlgRec * PageData ) +{ + // TODO unicode! + if( !WindowPrefs ) return false; + + // Backup + char * NameBackup = new char[ strlen( PageData->name ) + 1 ]; + strcpy( NameBackup, PageData->name ); + prefsDlgRec * PageDataBackup = new prefsDlgRec; + memcpy( PageDataBackup, PageData, sizeof( prefsDlgRec ) ); + PageDataBackup->name = NameBackup; + + AllWeNeed * awn = new AllWeNeed; + awn->PageData = PageDataBackup; + awn->hwnd = NULL; + + TV_INSERTSTRUCT tvi = { + root, // HTREEITEM hParent + TVI_SORT, // HTREEITEM hInsertAfter + { // + TVIF_PARAM | TVIF_STATE | TVIF_TEXT, // UINT mask + NULL, // HTREEITEM hItem + TVIS_EXPANDED | TVIS_SELECTED, // UINT state + 0, // UINT stateMask + PageDataBackup->name, // LPSTR pszText + ( int )strlen( PageDataBackup->name ) + 1, // int cchTextMax + 0, // int iImage + 0, // int iSelectedImage + 0, // int cChildren + ( LPARAM )awn // LPARAM lParam + } + }; + + HTREEITEM new_item = TreeView_InsertItem( WindowTree, &tvi ); + + TreeView_Expand( WindowTree, root, TVE_EXPAND ); + + rec_to_item.insert( pair( PageDataBackup, new_item ) ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocPrefs( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + switch( message ) + { + case WM_NOTIFY: + { + NMHDR * nmhdr = ( NMHDR * )lp; + if( ( nmhdr->hwndFrom == WindowTree ) && ( nmhdr->code == TVN_SELCHANGING ) ) + { + NMTREEVIEW * nmtv = ( NMTREEVIEW * )lp; + TVITEM * OldPage = &nmtv->itemOld; + TVITEM * NewPage = &nmtv->itemNew; + + // Destroy old window + AllWeNeed * old_awn = ( AllWeNeed * )OldPage->lParam; + if( old_awn && old_awn->hwnd && IsWindow( old_awn->hwnd ) ) + { + DestroyWindow( old_awn->hwnd ); + old_awn->hwnd = NULL; + } + + // Create new window + AllWeNeed * new_awn = ( AllWeNeed * )NewPage->lParam; + if( new_awn ) + { + prefsDlgRec * PageData = new_awn->PageData; + if( PageData && PageData->hInst ) // root has NULL here + { + if( !PageData->proc ) + { + MessageBox( 0, TEXT( "proc NULL" ), TEXT( "" ), 0 ); + PageData->proc = ( void * )WndprocPrefs; + } + +/* + RECT r; + GetWindowRect( WindowPage, &r ); + const int iWidth = r.right - r.left; + const int iHeight = r.bottom - r.top; + POINT p = { r.left, r.top }; + ScreenToClient( WindowPrefs, &p ); + MoveWindow( WindowPage, p.x, p.y, iWidth - 10, iHeight - 10, FALSE ); +*/ +// bKeepFocus = true; + + HWND hPage = CreateDialog( + PageData->hInst, // HINSTANCE hInstance, + ( LPCTSTR )PageData->dlgID, // LPCTSTR lpTemplate, + WindowPage, // HWND hWndParent, + ( DLGPROC )PageData->proc // DLGPROC lpDialogFunc + ); + new_awn->hwnd = hPage; + +// MoveWindow( WindowPage, p.x, p.y, iWidth, iHeight, FALSE ); + + ShowWindow( hPage, SW_SHOW ); + UpdateWindow( hPage ); + SetFocus( WindowTree ); +/* + SetActiveWindow( hPage ); + SetActiveWindow( hwnd ); +*/ + hCurrentPage = hPage; +// bKeepFocus = false; + } +/* + else + { + MessageBox( 0, TEXT( "hInst NULL" ), TEXT( "" ), 0 ); + } +*/ + } + else + { + MessageBox( 0, TEXT( "awn NULL" ), TEXT( "" ), 0 ); + } + } + break; + } + + case WM_COMMAND: + switch( HIWORD( wp ) ) + { + case BN_CLICKED: + if( ( HWND )lp == ButtonClose ) + { + PostMessage( hwnd, WM_SYSCOMMAND, SC_CLOSE, 0 ); + } + break; + } + break; + + case WM_SYSCOMMAND: + if( ( wp & 0xFFF0 ) == SC_CLOSE ) + { + ShowWindow( hwnd, SW_HIDE ); + + // Destroy current page so the settings are saved + // (currently be selecting the empty page + TreeView_SelectItem( WindowTree, root ); + +/* + if( hCurrentPage && IsWindow( hCurrentPage ) ) + { + DestroyWindow( hCurrentPage ); + hCurrentPage = NULL; + } +*/ + return 0; + } + break; + + case WM_KEYDOWN: + switch( wp ) + { + case VK_ESCAPE: + PostMessage( WindowPrefs, WM_SYSCOMMAND, SC_CLOSE, 0 ); + break; + + } + break; +/* + case WM_KILLFOCUS: + if( bKeepFocus ) + { + return 1; + } + break; +*/ + } + + return DefWindowProc( hwnd, message, wp, lp ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocTree( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + switch( message ) + { + case WM_KEYDOWN: + switch( wp ) + { + case VK_ESCAPE: + PostMessage( WindowPrefs, WM_SYSCOMMAND, SC_CLOSE, 0 ); + break; + + } + break; + + } + return CallWindowProc( WndprocTreeBackup, hwnd, message, wp, lp ); +} diff --git a/Externals/MusicMod/Player/Src/Prefs.h b/Externals/MusicMod/Player/Src/Prefs.h new file mode 100644 index 0000000000..4100d48760 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Prefs.h @@ -0,0 +1,36 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_PREFS_H +#define PA_PREFS_H + + + +#include "Global.h" +#include "Winamp/wa_ipc.h" + + + +namespace Prefs +{ + bool Create(); + bool Destroy(); + + bool Show(); + bool Show( prefsDlgRec * PageData ); + + bool AddPage( prefsDlgRec * PageData ); +}; + + + +#endif // PA_PREFS_H diff --git a/Externals/MusicMod/Player/Src/Rebar.cpp b/Externals/MusicMod/Player/Src/Rebar.cpp new file mode 100644 index 0000000000..b4c62ebeaf --- /dev/null +++ b/Externals/MusicMod/Player/Src/Rebar.cpp @@ -0,0 +1,1457 @@ + + +// ======================================================================================= +// WndprocRebar is called once every second during playback +// ======================================================================================= +// In Toolbar::Create() the progress bar is called the Seekbar + + + + +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Rebar.h" +#include "Font.h" +#include "Main.h" +#include "Playlist.h" +#include "Playback.h" +#include "Console.h" +#include "Util.h" +#include "Config.h" + +#include "Winamp/wa_ipc.h" +#include "Resources/resrc1.h" + + + +#define BAND_ORDER 1 +#define BAND_EQ 2 +#define BAND_SEEK 3 +#define BAND_VOL 4 +#define BAND_PAN 5 +#define BAND_BUTTONS 6 +#define BAND_VIS 7 + +#define BAND_FIRST BAND_ORDER +#define BAND_LAST BAND_VIS + + +#define IMAGE_PREV 0 +#define IMAGE_PLAY 1 +#define IMAGE_PAUSE 2 +#define IMAGE_STOP 3 +#define IMAGE_NEXT 4 +#define IMAGE_OPEN 5 + + +HWND WindowRebar = NULL; // extern +int iRebarHeight = 40; // extern + + +HMENU rebar_menu = NULL; + + +WNDPROC WndprocRebarBackup = NULL; +LRESULT CALLBACK WndprocRebar( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + +HWND WindowOrder = NULL; // extern +HWND WindowEq = NULL; // extern + + +HWND WindowSeek = NULL; // extern +WNDPROC WndprocSeekBackup = NULL; +LRESULT CALLBACK WndprocSeek( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + +HWND WindowVol = NULL; +WNDPROC WndprocVolBackup = NULL; +LRESULT CALLBACK WndprocVol( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + +HWND WindowPan = NULL; +WNDPROC WndprocPanBackup = NULL; +LRESULT CALLBACK WndprocPan( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + +HWND WindowButtons = NULL; + +HWND WindowVis = NULL; // extern +WNDPROC WndprocVisBackup = NULL; +LRESULT CALLBACK WndprocVis( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + + + +bool Rebar_BuildOrderBand(); +bool Rebar_BuildEqBand(); +bool Rebar_BuildSeekBand(); +bool Rebar_BuildVolBand(); +bool Rebar_BuildPanBand(); +bool Rebar_BuildButtonsBand(); +bool Rebar_BuildVisBand(); + +BandInfo biOrderBand = { 0 }; +BandInfo biEqBand = { 0 }; +BandInfo biSeekBand = { 0 }; +BandInfo biVolBand = { 0 }; +BandInfo biPanBand = { 0 }; +BandInfo biButtonsBand = { 0 }; +BandInfo biVisBand = { 0 }; + +const BandInfo biOrderBandDefault = { 0, 184, true, true }; +const BandInfo biVolBandDefault = { 1, 450, false, true }; +const BandInfo biPanBandDefault = { 2, 138, false, true }; +const BandInfo biEqBandDefault = { 3, 170, true, true }; +const BandInfo biVisBandDefault = { 4, 350, false, true }; +const BandInfo biButtonsBandDefault = { 5, 147, true, true }; +const BandInfo biSeekBandDefault = { 6, 373, false, true }; + + +void BandCallback( ConfVar * var ); + +ConfBandInfoCallback cbiOrderBand ( &biOrderBand, TEXT( "OrderBand" ), ( BandInfo * )&biOrderBandDefault, BandCallback ); +ConfBandInfoCallback cbiEqBand ( &biEqBand, TEXT( "EqBand" ), ( BandInfo * )&biEqBandDefault, BandCallback ); +ConfBandInfoCallback cbiSeekBand ( &biSeekBand, TEXT( "SeekBand" ), ( BandInfo * )&biSeekBandDefault, BandCallback ); +ConfBandInfoCallback cbiVolBand ( &biVolBand, TEXT( "VolBand" ), ( BandInfo * )&biVolBandDefault, BandCallback ); +ConfBandInfoCallback cbiPanBand ( &biPanBand, TEXT( "PanBand" ), ( BandInfo * )&biPanBandDefault, BandCallback ); +ConfBandInfoCallback cbiButtonsBand( &biButtonsBand, TEXT( "ButtonsBand" ), ( BandInfo * )&biButtonsBandDefault, BandCallback ); +ConfBandInfoCallback cbiVisBand ( &biVisBand, TEXT( "VisBand" ), ( BandInfo * )&biVisBandDefault, BandCallback ); + +bool bInvertPanSlider; +ConfBool cbInvertPanSlider( &bInvertPanSlider, TEXT( "InvertPanSlider" ), CONF_MODE_PUBLIC, false ); + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void BandCallback( ConfVar * var ) +{ + BandInfo * band; + int id; + + if( var == &cbiOrderBand ) + { + band = &biOrderBand; + id = BAND_ORDER; + } + else if( var == &cbiEqBand ) + { + band = &biEqBand; + id = BAND_EQ; + } + else if( var == &cbiSeekBand ) + { + band = &biSeekBand; + id = BAND_SEEK; + } + else if( var == &cbiVolBand ) + { + band = &biVolBand; + id = BAND_VOL; + } + else if( var == &cbiPanBand ) + { + band = &biPanBand; + id = BAND_PAN; + } + else if( var == &cbiButtonsBand ) + { + band = &biButtonsBand; + id = BAND_BUTTONS; + } + else if( var == &cbiVisBand ) + { + band = &biVisBand; + id = BAND_VIS; + } + else + { + return; + } + + band->m_iIndex = ( int )SendMessage( WindowRebar, RB_IDTOINDEX, id, 0 ); + if( band->m_iIndex == -1 ) return; + + REBARBANDINFO rbbi; + rbbi.cbSize = sizeof( REBARBANDINFO ); + rbbi.fMask = RBBIM_SIZE | RBBIM_STYLE; + SendMessage( WindowRebar, RB_GETBANDINFO, band->m_iIndex, ( LPARAM )&rbbi ); + + band->m_iWidth = rbbi.cx; + band->m_bBreak = ( ( rbbi.fStyle & RBBS_BREAK ) == RBBS_BREAK ); + band->m_bVisible = ( ( rbbi.fStyle & RBBS_HIDDEN ) != RBBS_HIDDEN ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Toolbar::Create() +{ + LoadCommonControls(); + + WindowRebar = CreateWindowEx( + WS_EX_TOOLWINDOW | + WS_EX_LEFT | + WS_EX_LTRREADING | + WS_EX_RIGHTSCROLLBAR, + REBARCLASSNAME, + NULL, + WS_CHILD | + WS_VISIBLE | + WS_CLIPSIBLINGS | + WS_CLIPCHILDREN | + WS_BORDER | + RBS_VARHEIGHT | + RBS_BANDBORDERS | + RBS_DBLCLKTOGGLE | // RBS_AUTOSIZE + CCS_TOP | + CCS_NOPARENTALIGN | + CCS_NODIVIDER, // | CCS_ADJUSTABLE, + 0, + 0, + 0, + 0, + WindowMain, + NULL, + g_hInstance, + NULL + ); + +/* + #define STYLE 1 + #if STYLE == 1 + // Normal + MoveWindow( rebar, 0, 0, width, 30, TRUE ); + #elif STYLE == 2 + // Left ONLY + MoveWindow( rebar, 1, 0, width - 1, 30, TRUE ); + #endif +*/ + + REBARINFO rbi = { + sizeof( REBARINFO ), // UINT cbSize; + 0, // UINT fMask + NULL // HIMAGELIST himl + }; + + if( !SendMessage( WindowRebar, RB_SETBARINFO, 0, ( LPARAM )&rbi ) ) + { + return false; + } + + // Exchange window procedure + WndprocRebarBackup = ( WNDPROC )GetWindowLong( WindowRebar, GWL_WNDPROC ); + //WndprocRebarBackup = ( WNDPROC )GetWindowLongPtr( WindowRebar, GWLP_WNDPROC ); // 64 bit + if( WndprocRebarBackup != NULL ) + { + SetWindowLong( WindowRebar, GWL_WNDPROC, ( LONG )WndprocRebar ); + //SetWindowLongPtr( WindowRebar, GWLP_WNDPROC, ( LONG )WndprocRebar ); + } + + rebar_menu = CreatePopupMenu(); + AppendMenu( rebar_menu, MF_STRING, BAND_BUTTONS, TEXT( "Buttons" ) ); + AppendMenu( rebar_menu, MF_STRING, BAND_EQ, TEXT( "Equalizer" ) ); + AppendMenu( rebar_menu, MF_STRING, BAND_ORDER, TEXT( "Order" ) ); + AppendMenu( rebar_menu, MF_STRING, BAND_PAN, TEXT( "Panning" ) ); + AppendMenu( rebar_menu, MF_STRING, BAND_SEEK, TEXT( "Seekbar" ) ); + AppendMenu( rebar_menu, MF_STRING, BAND_VIS, TEXT( "Visualization" ) ); + AppendMenu( rebar_menu, MF_STRING, BAND_VOL, TEXT( "Volume" ) ); + + Rebar_BuildOrderBand(); + Rebar_BuildEqBand(); + Rebar_BuildSeekBand(); + Rebar_BuildVolBand(); + Rebar_BuildPanBand(); + Rebar_BuildButtonsBand(); + Rebar_BuildVisBand(); + + cbiOrderBand.Apply ( WindowRebar, BAND_ORDER ); + cbiEqBand.Apply ( WindowRebar, BAND_EQ ); + cbiSeekBand.Apply ( WindowRebar, BAND_SEEK ); + cbiVolBand.Apply ( WindowRebar, BAND_VOL ); + cbiPanBand.Apply ( WindowRebar, BAND_PAN ); + cbiButtonsBand.Apply( WindowRebar, BAND_BUTTONS ); + cbiVisBand.Apply ( WindowRebar, BAND_VIS ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Rebar_BuildOrderBand() +{ + if( !WindowRebar ) return false; + + WindowOrder = CreateWindowEx( + 0, + TEXT( "COMBOBOX" ), + TEXT( "" ), + CBS_HASSTRINGS | + CBS_DROPDOWNLIST | + WS_CHILD, + 0, + 0, + 133, + 200, + WindowRebar, + NULL, + g_hInstance, + NULL + ); + + if( !WindowOrder ) return false; + + // Add mode names + TCHAR * szName; + for( int i = ORDER_FIRST; i <= ORDER_LAST; i++ ) + { + szName = Playback::Order::GetModeName( i ); + SendMessage( WindowOrder, CB_ADDSTRING, 0, ( LPARAM )szName ); + } + + // Initial value + SendMessage( WindowOrder, CB_SETCURSEL, Playback::Order::GetCurMode(), 0 ); + + RECT rc; + GetWindowRect( WindowOrder, &rc ); + + REBARBANDINFO rbBand; + rbBand.cbSize = sizeof( REBARBANDINFO ); + rbBand.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + RBBIM_CHILD | // The hwndChild member is valid or must be filled. + RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbBand.fStyle = RBBS_GRIPPERALWAYS | + RBBS_CHILDEDGE | + ( biOrderBand.m_bBreak ? RBBS_BREAK : 0 ); + + rbBand.lpText = TEXT( " Order " ); + rbBand.hwndChild = WindowOrder; + rbBand.cxMinChild = rc.right - rc.left; + rbBand.cyMinChild = 21; // IMP + rbBand.cx = biOrderBand.m_iWidth; + rbBand.wID = BAND_ORDER; + rbBand.cyChild = 21; + rbBand.cyMaxChild = 21; + rbBand.cyIntegral = 1; + rbBand.cxIdeal = 200; + + // Add band + SendMessage( WindowRebar, RB_INSERTBAND, ( WPARAM )-1, ( LPARAM )&rbBand ); + + Font::Apply( WindowOrder ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool AddPreset( TCHAR * szPresetName ) +{ + const int iLen = _tcslen( szPresetName ); + TCHAR * szFinal = new TCHAR[ iLen + 2 + 1 ]; + szFinal[ 0 ] = TEXT( ' ' ); + szFinal[ iLen + 1 ] = TEXT( ' ' ); + szFinal[ iLen + 2 ] = TEXT( '\0' ); + memcpy( szFinal + 1, szPresetName, iLen * sizeof( TCHAR ) ); + + SendMessage( WindowEq, CB_ADDSTRING, 0, ( LPARAM )szFinal ); + + delete [] szFinal; + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Rebar_BuildEqBand() +{ + if( !WindowRebar ) return false; + + WindowEq = CreateWindowEx( + 0, + TEXT( "COMBOBOX" ), + TEXT( "" ), + CBS_HASSTRINGS | + CBS_DROPDOWNLIST | + WS_CHILD, + 0, + 0, + 133, + 300, + WindowRebar, + NULL, + g_hInstance, + NULL + ); + + if( !WindowEq ) return false; + + // Add preset names + SendMessage( WindowEq, CB_ADDSTRING, 0, ( LPARAM )TEXT( " Disabled " ) ); + Playback::Eq::ReadPresets( AddPreset ); + + // Initial value + SendMessage( WindowEq, CB_SETCURSEL, Playback::Eq::GetCurIndex() + 1, 0 ); + + // TODO Disabled + // EnableWindow( WindowEq, FALSE ); + + + RECT rc; + GetWindowRect( WindowEq, &rc ); + + REBARBANDINFO rbBand; + rbBand.cbSize = sizeof( REBARBANDINFO ); + rbBand.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + RBBIM_CHILD | // The hwndChild member is valid or must be filled. + RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbBand.fStyle = RBBS_GRIPPERALWAYS | + RBBS_CHILDEDGE | + ( biEqBand.m_bBreak ? RBBS_BREAK : 0 ); + + rbBand.lpText = TEXT( " EQ " ); + rbBand.hwndChild = WindowEq; + rbBand.cxMinChild = rc.right - rc.left; + rbBand.cyMinChild = 21; // IMP + rbBand.cx = biEqBand.m_iWidth; + rbBand.wID = BAND_EQ; + rbBand.cyChild = 21; + rbBand.cyMaxChild = 21; + rbBand.cyIntegral = 1; + rbBand.cxIdeal = 200; + + // Add band + SendMessage( WindowRebar, RB_INSERTBAND, ( WPARAM )-1, ( LPARAM )&rbBand ); + + Font::Apply( WindowEq ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Rebar_BuildSeekBand() +{ + if( !WindowRebar ) return false; + + WindowSeek = CreateWindowEx( + 0, + TRACKBAR_CLASS, + TEXT( "" ), + WS_CHILD | + WS_DISABLED | + TBS_HORZ | + TBS_NOTICKS | + TBS_FIXEDLENGTH | + TBS_BOTH, + 0, + 0, + 300, + 21, + WindowRebar, + NULL, + g_hInstance, + NULL + ); + + // Range + SendMessage( + WindowSeek, + TBM_SETRANGE, + TRUE, + ( LPARAM )MAKELONG( 0, 999 ) + ); + + // Thumb size + SendMessage( + WindowSeek, + TBM_SETTHUMBLENGTH, + 14, + 0 + ); + + // Exchange window procedure + WndprocSeekBackup = ( WNDPROC )GetWindowLong( WindowSeek, GWL_WNDPROC ); + if( WndprocSeekBackup != NULL ) + { + SetWindowLong( WindowSeek, GWL_WNDPROC, ( LONG )WndprocSeek ); + } + + + REBARBANDINFO rbbi_seek; + rbbi_seek.cbSize = sizeof( REBARBANDINFO ); + rbbi_seek.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + RBBIM_CHILD | // The hwndChild member is valid or must be filled. + RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbbi_seek.fStyle = RBBS_GRIPPERALWAYS | + RBBS_CHILDEDGE | + ( biSeekBand.m_bBreak ? RBBS_BREAK : 0 ); + + rbbi_seek.lpText = " Pos"; + rbbi_seek.hwndChild = WindowSeek; + rbbi_seek.cxMinChild = 100; + rbbi_seek.cyMinChild = 21; // IMP + rbbi_seek.cx = biSeekBand.m_iWidth; + rbbi_seek.wID = BAND_SEEK; + rbbi_seek.cyChild = 21; + rbbi_seek.cyMaxChild = 21; + rbbi_seek.cyIntegral = 1; + rbbi_seek.cxIdeal = 300; + + + // Add band + SendMessage( + WindowRebar, + RB_INSERTBAND, + ( WPARAM )-1, + ( LPARAM )&rbbi_seek + ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Rebar_BuildVolBand() +{ + if( !WindowRebar ) return false; + + WindowVol = CreateWindowEx( + 0, + TRACKBAR_CLASS, + TEXT( "" ), + WS_CHILD | + TBS_HORZ | + TBS_NOTICKS | + TBS_FIXEDLENGTH | + TBS_BOTH, + 0, + 0, + 200, + 21, + WindowRebar, + NULL, + g_hInstance, + NULL + ); + + if( !WindowVol ) return false; + + // Range + SendMessage( WindowVol, TBM_SETRANGE, TRUE, ( LPARAM )MAKELONG( 0, 255 ) ); + + // Initial value + SendMessage( WindowVol, TBM_SETPOS, TRUE, Playback::Volume::Get() ); + + // Thumbs size + SendMessage( WindowVol, TBM_SETTHUMBLENGTH, 14, 0 ); + + // Exchange window procedure + WndprocVolBackup = ( WNDPROC )GetWindowLong( WindowVol, GWL_WNDPROC ); + if( WndprocVolBackup != NULL ) + { + SetWindowLong( WindowVol, GWL_WNDPROC, ( LONG )WndprocVol ); + } + + + REBARBANDINFO rbbi_vol; + rbbi_vol.cbSize = sizeof( REBARBANDINFO ); + rbbi_vol.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + RBBIM_CHILD | // The hwndChild member is valid or must be filled. + RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbbi_vol.fStyle = RBBS_GRIPPERALWAYS | + RBBS_CHILDEDGE | + ( biVolBand.m_bBreak ? RBBS_BREAK : 0 ); + + rbbi_vol.lpText = TEXT( " Vol" ); + rbbi_vol.hwndChild = WindowVol; + rbbi_vol.cxMinChild = 100; + rbbi_vol.cyMinChild = 21; // IMP + rbbi_vol.cx = biVolBand.m_iWidth; + rbbi_vol.wID = BAND_VOL; + rbbi_vol.cyChild = 21; + rbbi_vol.cyMaxChild = 21; + rbbi_vol.cyIntegral = 1; + rbbi_vol.cxIdeal = 100; + + // Add band + SendMessage( WindowRebar, RB_INSERTBAND, ( WPARAM )-1, ( LPARAM )&rbbi_vol ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Rebar_BuildPanBand() +{ + if( !WindowRebar ) return false; + + WindowPan = CreateWindowEx( + 0, + TRACKBAR_CLASS, + TEXT( "" ), + WS_CHILD | + TBS_HORZ | + TBS_NOTICKS | + TBS_FIXEDLENGTH | + TBS_BOTH, + 0, + 0, + 200, + 21, + WindowRebar, + NULL, + g_hInstance, + NULL + ); + + if( !WindowPan ) return false; + + // Range + SendMessage( WindowPan, TBM_SETRANGE, TRUE, ( LPARAM )MAKELONG( -127, 127 ) ); + + // Initial value + const int iCurPan = Playback::Pan::Get(); + SendMessage( WindowPan, TBM_SETPOS, TRUE, bInvertPanSlider ? -iCurPan : iCurPan ); + + // Thumb size + SendMessage( WindowPan, TBM_SETTHUMBLENGTH, 14, 0 ); + + + // Exchange window procedure + WndprocPanBackup = ( WNDPROC )GetWindowLong( WindowPan, GWL_WNDPROC ); + if( WndprocPanBackup != NULL ) + { + SetWindowLong( WindowPan, GWL_WNDPROC, ( LONG )WndprocPan ); + } + + + REBARBANDINFO rbbi_pan; + rbbi_pan.cbSize = sizeof( REBARBANDINFO ); + rbbi_pan.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + RBBIM_CHILD | // The hwndChild member is valid or must be filled. + RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbbi_pan.fStyle = RBBS_GRIPPERALWAYS | + RBBS_CHILDEDGE | + ( biPanBand.m_bBreak ? RBBS_BREAK : 0 ); + + rbbi_pan.lpText = TEXT( " Pan" ); + rbbi_pan.hwndChild = WindowPan; + rbbi_pan.cxMinChild = 100; + rbbi_pan.cyMinChild = 21; // IMP + rbbi_pan.cx = biPanBand.m_iWidth; + rbbi_pan.wID = BAND_PAN; + rbbi_pan.cyChild = 21; + rbbi_pan.cyMaxChild = 21; + rbbi_pan.cyIntegral = 1; + rbbi_pan.cxIdeal = 100; + + // Add band + SendMessage( WindowRebar, RB_INSERTBAND, ( WPARAM )-1, ( LPARAM )&rbbi_pan ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Rebar_BuildButtonsBand() +{ + if( !WindowRebar ) return false; + + WindowButtons = CreateWindowEx( + 0, + TOOLBARCLASSNAME, + TEXT( "" ), + WS_CHILD | + TBSTYLE_FLAT | + CCS_NORESIZE | // Means we care about the size ourselves + CCS_NOPARENTALIGN | // Make it work with the rebar control + CCS_NODIVIDER, // No divider on top + 0, + 0, + 100, + 21, + WindowRebar, + NULL, + g_hInstance, + NULL + ); + + if( !WindowButtons ) return false; + + SendMessage( WindowButtons, TB_BUTTONSTRUCTSIZE, ( WPARAM )sizeof( TBBUTTON ), 0 ); + + TBBUTTON tb_button[ 7 ]; + + // Make image list, TODO delete later + HIMAGELIST hImages = ImageList_LoadBitmap( + GetModuleHandle( NULL ), // HINSTANCE hi + MAKEINTRESOURCE( IDB_BITMAP1 ), // LPCTSTR lpbmp + 14, // int cx + 6, // int cGrow + RGB( 255, 000, 255 ) // COLORREF crMask + ); + + SendMessage( WindowButtons, TB_SETIMAGELIST, 0, ( LPARAM )hImages ); + // SendMessage( WindowButtons, TB_SETHOTIMAGELIST, 0, ( LPARAM )hImages ); + // SendMessage( WindowButtons, TB_SETDISABLEDIMAGELIST, 0, ( LPARAM )hImages ); + + // Build buttons + tb_button[ 0 ].iBitmap = IMAGE_PREV; + tb_button[ 0 ].idCommand = WINAMP_BUTTON1; + tb_button[ 0 ].fsState = TBSTATE_ENABLED; + tb_button[ 0 ].fsStyle = TBSTYLE_BUTTON; + tb_button[ 0 ].dwData = 0; + tb_button[ 0 ].iString = SendMessage( WindowButtons, TB_ADDSTRING, 0, ( LPARAM )TEXT( "Previous" )); + + tb_button[ 1 ].iBitmap = IMAGE_PLAY; + tb_button[ 1 ].idCommand = WINAMP_BUTTON2; + tb_button[ 1 ].fsState = TBSTATE_ENABLED; + tb_button[ 1 ].fsStyle = TBSTYLE_BUTTON; + tb_button[ 1 ].dwData = 0; + tb_button[ 1 ].iString = SendMessage( WindowButtons, TB_ADDSTRING, 0, ( LPARAM )TEXT( "Play" )); + + tb_button[ 2 ].iBitmap = IMAGE_PAUSE; + tb_button[ 2 ].idCommand = WINAMP_BUTTON3; + tb_button[ 2 ].fsState = TBSTATE_ENABLED; + tb_button[ 2 ].fsStyle = TBSTYLE_BUTTON; + tb_button[ 2 ].dwData = 0; + tb_button[ 2 ].iString = SendMessage( WindowButtons, TB_ADDSTRING, 0, ( LPARAM )TEXT( "Pause" )); + + tb_button[ 3 ].iBitmap = IMAGE_STOP; + tb_button[ 3 ].idCommand = WINAMP_BUTTON4; + tb_button[ 3 ].fsState = TBSTATE_ENABLED; + tb_button[ 3 ].fsStyle = TBSTYLE_BUTTON; + tb_button[ 3 ].dwData = 0; + tb_button[ 3 ].iString = SendMessage( WindowButtons, TB_ADDSTRING, 0, ( LPARAM )TEXT( "Stop" )); + + tb_button[ 4 ].iBitmap = IMAGE_NEXT; + tb_button[ 4 ].idCommand = WINAMP_BUTTON5; + tb_button[ 4 ].fsState = TBSTATE_ENABLED; + tb_button[ 4 ].fsStyle = TBSTYLE_BUTTON; + tb_button[ 4 ].dwData = 0; + tb_button[ 4 ].iString = SendMessage( WindowButtons, TB_ADDSTRING, 0, ( LPARAM )TEXT( "Next" )); + + tb_button[ 5 ].iBitmap = 0; + tb_button[ 5 ].idCommand = -1; + tb_button[ 5 ].fsState = TBSTATE_ENABLED; + tb_button[ 5 ].fsStyle = TBSTYLE_SEP; + tb_button[ 5 ].dwData = 0; + tb_button[ 5 ].iString = 0; + + tb_button[ 6 ].iBitmap = IMAGE_OPEN; + tb_button[ 6 ].idCommand = WINAMP_FILE_PLAY; + tb_button[ 6 ].fsState = TBSTATE_ENABLED; + tb_button[ 6 ].fsStyle = TBSTYLE_BUTTON; + tb_button[ 6 ].dwData = 0; + tb_button[ 6 ].iString = SendMessage( WindowButtons, TB_ADDSTRING, 0, ( LPARAM )TEXT( "Open" )); + + // Add buttons + SendMessage( WindowButtons, TB_SETBUTTONSIZE, 0, MAKELONG( 14, 14 ) ); + SendMessage( WindowButtons, TB_ADDBUTTONS, 7, ( LPARAM )( LPTBBUTTON )&tb_button ); + + // Disable labels + SendMessage( WindowButtons, TB_SETMAXTEXTROWS, 0, 0 ); + + // Resize + RECT r; + GetWindowRect( WindowButtons, &r ); + SetWindowPos( WindowButtons, NULL, 0, 0, 134, r.bottom - r.top, SWP_NOMOVE ); + GetWindowRect( WindowButtons, &r ); + + REBARBANDINFO rbbi_buttons; + rbbi_buttons.cbSize = sizeof( REBARBANDINFO ); + rbbi_buttons.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + RBBIM_CHILD | // The hwndChild member is valid or must be filled. + RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + // RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbbi_buttons.fStyle = RBBS_GRIPPERALWAYS | + RBBS_CHILDEDGE | + ( biButtonsBand.m_bBreak ? RBBS_BREAK : 0 ); + + // rbbi_buttons.lpText = TEXT( " Playback" ); + rbbi_buttons.hwndChild = WindowButtons; + rbbi_buttons.cxMinChild = r.right - r.left; + rbbi_buttons.cyMinChild = 21; // IMP + rbbi_buttons.cx = r.right - r.left; + rbbi_buttons.wID = BAND_BUTTONS; + rbbi_buttons.cyChild = 21; + rbbi_buttons.cyMaxChild = 21; + rbbi_buttons.cyIntegral = 1; + rbbi_buttons.cxIdeal = r.right - r.left; + + // Add band + SendMessage( WindowRebar, RB_INSERTBAND, ( WPARAM )-1, ( LPARAM )&rbbi_buttons ); + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool Rebar_BuildVisBand() +{ + if( !WindowRebar ) return false; + + WindowVis = CreateWindowEx( + WS_EX_STATICEDGE, + TEXT( "STATIC" ), + TEXT( "" ), + WS_CHILD | SS_CENTER, + 0, + 0, + 100, + 21, + WindowRebar, + NULL, + g_hInstance, + NULL + ); + + if( !WindowVis ) return false; + Font::Apply( WindowVis ); + + // Resize + RECT r; + GetWindowRect( WindowButtons, &r ); +// SetWindowPos( WindowButtons, NULL, 0, 0, 146, r.bottom - r.top, SWP_NOMOVE ); +// GetWindowRect( WindowButtons, &r ); + + REBARBANDINFO rbbi_buttons; + rbbi_buttons.cbSize = sizeof( REBARBANDINFO ); + rbbi_buttons.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + RBBIM_CHILD | // The hwndChild member is valid or must be filled. + RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbbi_buttons.fStyle = RBBS_GRIPPERALWAYS | + RBBS_CHILDEDGE | + ( biVisBand.m_bBreak ? RBBS_BREAK : 0 ); + + rbbi_buttons.lpText = TEXT( " Vis " ); + rbbi_buttons.hwndChild = WindowVis; + rbbi_buttons.cxMinChild = r.right - r.left; + rbbi_buttons.cyMinChild = 21; // IMP + rbbi_buttons.cx = r.right - r.left; + rbbi_buttons.wID = BAND_VIS; + rbbi_buttons.cyChild = 21; + rbbi_buttons.cyMaxChild = 21; + rbbi_buttons.cyIntegral = 1; + rbbi_buttons.cxIdeal = r.right - r.left; + + // Add band + SendMessage( WindowRebar, RB_INSERTBAND, ( WPARAM )-1, ( LPARAM )&rbbi_buttons ); + + // Exchange window procedure + WndprocVisBackup = ( WNDPROC )GetWindowLong( WindowVis, GWL_WNDPROC ); + if( WndprocVisBackup != NULL ) + { + SetWindowLong( WindowVis, GWL_WNDPROC, ( LONG )WndprocVis ); + } + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ContextMenuRebar( POINT * p ) +{ + REBARBANDINFO rbbi; + rbbi.cbSize = sizeof( REBARBANDINFO ); + rbbi.fMask = RBBIM_STYLE; + + bool bBandVisible[ BAND_LAST - BAND_FIRST + 1 ] = { 0 }; // ID to visibility + int iBandIndex [ BAND_LAST - BAND_FIRST + 1 ] = { 0 }; // ID to Index + + for( int i = BAND_FIRST; i <= BAND_LAST; i++ ) + { + const int iArrayIndex = i - BAND_FIRST; + + // Get index + iBandIndex[ iArrayIndex ] = ( int )SendMessage( WindowRebar, RB_IDTOINDEX, i, 0 ); + if( iBandIndex[ iArrayIndex ] == -1 ) break; + + // Get info + if( !SendMessage( WindowRebar, RB_GETBANDINFO, iBandIndex[ iArrayIndex ], ( LPARAM )&rbbi ) ) + { + break; + } + + bBandVisible[ iArrayIndex ] = ( ( rbbi.fStyle & RBBS_HIDDEN ) != RBBS_HIDDEN ); + + // Update menu item + CheckMenuItem( + rebar_menu, + i, + bBandVisible[ iArrayIndex ] ? MF_CHECKED : MF_UNCHECKED + ); + } + + BOOL iIndex = TrackPopupMenu( + rebar_menu, + TPM_LEFTALIGN | + TPM_TOPALIGN | + TPM_NONOTIFY | + TPM_RETURNCMD | + TPM_RIGHTBUTTON, + p->x, + p->y, + 0, + WindowRebar, + NULL + ); + + // Toggle visibility + if( ( iIndex >= BAND_FIRST ) && ( iIndex <= BAND_LAST ) ) + { + const int iArrayIndex = iIndex - BAND_FIRST; + SendMessage( WindowRebar, RB_SHOWBAND, iBandIndex[ iArrayIndex ], bBandVisible[ iArrayIndex ] ? FALSE : TRUE ); + + // Turn off vis child + if( iIndex == BAND_VIS ) + { + HWND hChild = GetWindow( WindowVis, GW_CHILD ); + if( IsWindow( hChild ) ) + { + SendMessage( hChild, WM_DESTROY, 0, 0 ); + } + } + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocRebar( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + //Console::Append( TEXT("Rebar.cpp: WndprocRebar called") ); + + switch( message ) + { + case WM_CONTEXTMENU: // This affects the placement of the progress bar + { + //POINT p; + //GetCursorPos( &p ); + //ContextMenuRebar( &p ); + return 0; + } + + case WM_DESTROY: + cbiOrderBand.TriggerCallback(); + cbiOrderBand.RemoveCallback(); + cbiEqBand.TriggerCallback(); + cbiEqBand.RemoveCallback(); + cbiSeekBand.TriggerCallback(); + cbiSeekBand.RemoveCallback(); + cbiVolBand.TriggerCallback(); + cbiVolBand.RemoveCallback(); + cbiPanBand.TriggerCallback(); + cbiPanBand.RemoveCallback(); + cbiButtonsBand.TriggerCallback(); + cbiButtonsBand.RemoveCallback(); + cbiVisBand.TriggerCallback(); + cbiVisBand.RemoveCallback(); + break; + + } + return CallWindowProc( WndprocRebarBackup, hwnd, message, wp, lp ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocSeek( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + static bool bDown = false; + static int iLastVal = 0; + + //Console::Append( TEXT("Rebar.cpp: WndprocSeek called") ); + + switch( message ) + { + case WM_LBUTTONDOWN: + { + SetFocus( hwnd ); + + // Capture mouse + SetCapture( hwnd ); + bDown = true; + + // NO BREAK! + } + case WM_MOUSEMOVE: + { + if( !bDown ) break; + if( !Playback::IsPlaying() ) return 0; // Deny slider move + + RECT r; + GetWindowRect( hwnd, &r ); + + const int iWidth = r.right - r.left; + if( !iWidth ) return 0; + int iVal = 1000 * MAX( 0, MIN( ( short )LOWORD( lp ), iWidth ) ) / iWidth; + + // Snap + if( ( short )LOWORD( lp ) < 7 ) + { + iVal = 0; + } + else if( abs( iWidth - ( short )LOWORD( lp ) ) < 7 ) + { + iVal = 999; + } + + // Update slider + PostMessage( hwnd, TBM_SETPOS, ( WPARAM )( TRUE ), iVal ); + + // Seek + iLastVal = iVal; + Playback::SeekPercent( iLastVal / 10.0f ); + + // TODO: Seek on slide ON/OFF + + + return 0; + } + case WM_LBUTTONUP: + case WM_NCLBUTTONUP: + { + // Release capture + bDown = false; + ReleaseCapture(); + + if( !Playback::IsPlaying() ) return 0; + + int ms = Playback::PercentToMs( iLastVal / 10.0f ); + if( ms < 0 ) break; + + int h = ms / 3600000; + int m = ( ms / 60000 ) % 60; + int s = ( ms / 1000 ) % 60; + ms = ms % 1000; + + + TCHAR szBuffer[ 5000 ]; + + /* + bool bShowMs = false; + + if( h ) + { + if( bShowMs ) + { + */ + + // 00:00:00.000 + _stprintf( szBuffer, TEXT( "Jumped to %02i:%02i:%02i.%03i" ), h, m, s, ms ); + Console::Append( szBuffer ); + Console::Append( " " ); + + /* + + } + else + { + // 00:00:00 + } + } + else if( m ) + { + if( bShowMs ) + { + // 00:00.000 + } + else + { + // 00:00 + } + } + else + { + if( bShowMs ) + { + // 00.000 + } + else + { + // 00s + } + } + */ + + return 0; + } + + case WM_KEYDOWN: + case WM_KEYUP: + SendMessage( WindowPlaylist, message, wp, lp ); + return 0; + + } + return CallWindowProc( WndprocSeekBackup, hwnd, message, wp, lp ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocVol( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + static bool bDown = false; + static int iLastVal = 0; + + switch( message ) + { + case WM_LBUTTONDOWN: + SetFocus( hwnd ); + + // Capture mouse + SetCapture( hwnd ); + bDown = true; + + // NO BREAK! + + case WM_MOUSEMOVE: + { + if( !bDown ) break; + + RECT r; + GetWindowRect( hwnd, &r ); + + const int iWidth = r.right - r.left; + if( !iWidth ) return 0; + const int x = ( int )( short )LOWORD( lp ); + int iVal = 255 * x / ( int )iWidth; + + if( iVal < 0 ) + iVal = 0; + else if( ( iVal > 255 ) || ( abs( iWidth - x ) < 7 ) ) // Snap + iVal = 255; + + // Update slider + PostMessage( hwnd, TBM_SETPOS , ( WPARAM )( TRUE ), iVal ); + + // Apply volume + iLastVal = iVal; + Playback::Volume::Set( iLastVal ); + + return 0; + } + case WM_LBUTTONUP: + case WM_NCLBUTTONUP: + { + // Release capture + bDown = false; + ReleaseCapture(); + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Volume changed to %i%%" ), iLastVal * 100 / 255 ); + Console::Append( szBuffer ); + Console::Append( " " ); + + return 0; + } + } + return CallWindowProc( WndprocVolBackup, hwnd, message, wp, lp ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocPan( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + static bool bDown = false; + static int iLastVal = 0; + + switch( message ) + { + case WM_LBUTTONDOWN: + SetFocus( hwnd ); + + // Capture mouse + SetCapture( hwnd ); + bDown = true; + + // NO BREAK! + + case WM_MOUSEMOVE: + { + if( !bDown ) break; + + RECT r; + GetWindowRect( hwnd, &r ); + + const int iWidth = r.right - r.left; + if( !iWidth ) return 0; + const int x = ( int )( short )LOWORD( lp ); + int iVal = 254 * x / ( int )iWidth - 127; + + if( iVal < -127 ) + iVal = -127; + else if( iVal > 127 ) + iVal = 127; + else if( abs( ( int )iWidth / 2 - x ) < 7 ) // Snap + iVal = 0; + + // Update slider + PostMessage( hwnd, TBM_SETPOS , ( WPARAM )( TRUE ), iVal ); + + // Invert if needed (slider needs original value!) + if( bInvertPanSlider ) + { + iVal = -iVal; + } + + // Apply panning + iLastVal = iVal; + Playback::Pan::Set( iVal ); + + return 0; + } + case WM_LBUTTONUP: + case WM_NCLBUTTONUP: + { + // Release capture + bDown = false; + ReleaseCapture(); + + TCHAR szBuffer[ 5000 ]; + if( iLastVal < 0 ) + _stprintf( szBuffer, TEXT( "Panning changed to %i%% left" ), -iLastVal * 100 / 127 ); + else if( iLastVal == 0 ) + _stprintf( szBuffer, TEXT( "Panning changed to center" ) ); + else // if( iLastVal > 0 ) + _stprintf( szBuffer, TEXT( "Panning changed to %i%% right" ), iLastVal * 100 / 127 ); + Console::Append( szBuffer ); + Console::Append( " " ); + + return 0; + } + } + return CallWindowProc( WndprocPanBackup, hwnd, message, wp, lp ); +} + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocVis( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + switch( message ) + { + case WM_SHOWWINDOW: + { + if( wp == FALSE ) + { + const HWND hChild = GetWindow( hwnd, GW_CHILD ); + if( !IsWindow( hChild ) ) break; + + // Strange workaround + ShowWindow( hChild, FALSE ); + SetParent( hChild, NULL ); + PostMessage( hChild, WM_SYSCOMMAND, SC_CLOSE, 0 ); + } + } + break; + + case WM_SIZE: + { + // Resize vis child + HWND hChild = GetWindow( hwnd, GW_CHILD ); + if( !IsWindow( hChild ) ) break; + MoveWindow( hChild, 0, 0, LOWORD( lp ), HIWORD( lp ), TRUE ); + } + break; + } + return CallWindowProc( WndprocVisBackup, hwnd, message, wp, lp ); +} + + +///////////////////////// + + + + /* + // Create the toolbar control to be added. + // hwndTB = CreateToolbar(hwndOwner, dwStyle); + + // Get the height of the toolbar. + dwBtnSize = SendMessage(hwndTB, TB_GETBUTTONSIZE, 0,0); + + // Set values unique to the band with the toolbar. + rbBand.lpText = "Tool Bar"; + rbBand.hwndChild = hwndTB; + rbBand.cxMinChild = 0; + rbBand.cyMinChild = HIWORD(dwBtnSize); + rbBand.cx = 250; + + // Add the band that has the toolbar. + SendMessage(rebar, RB_INSERTBAND, (WPARAM)-1, (LPARAM)&rbBand); + */ + + +/////////////////////////////////////// + +/* +void band_dummy() +{ + REBARBANDINFO rbbi3; + rbbi3.cbSize = sizeof( REBARBANDINFO ); + rbbi3.fMask = 0 | + // RBBIM_BACKGROUND | // The hbmBack member is valid or must be filled. + // RBBIM_CHILD | // The hwndChild member is valid or must be filled. + // RBBIM_CHILDSIZE | // The cxMinChild, cyMinChild, cyChild, cyMaxChild, and cyIntegral members are valid or must be filled. + // RBBIM_COLORS | // The clrFore and clrBack members are valid or must be filled. + // RBBIM_HEADERSIZE | // Version 4.71. The cxHeader member is valid or must be filled. + RBBIM_IDEALSIZE | // Version 4.71. The cxIdeal member is valid or must be filled. + RBBIM_ID | // The wID member is valid or must be filled. + // RBBIM_IMAGE | // The iImage member is valid or must be filled. + // RBBIM_LPARAM | // Version 4.71. The lParam member is valid or must be filled. + RBBIM_SIZE | // The cx member is valid or must be filled. + RBBIM_STYLE | // The fStyle member is valid or must be filled. + RBBIM_TEXT | // The lpText member is valid or must be filled. + 0; + + rbbi3.fStyle = RBBS_GRIPPERALWAYS; // | RBBS_BREAK; + rbbi3.lpText = " "; //NULL; // "Dummy"; + rbbi3.hwndChild = NULL; + rbbi3.cxMinChild = 10; //rc.right - rc.left; + rbbi3.cyMinChild = 21; // IMP + rbbi3.cx = 300; + rbbi3.wID = BAND_ID_DUMMY; + rbbi3.cyChild = 21; //rc.bottom - rc.top; + rbbi3.cyMaxChild = 21; // rc.bottom - rc.top; + rbbi3.cyIntegral = 1; + rbbi3.cxIdeal = 700; //rc.right - rc.left; + + // Add the band that has the combo box. + SendMessage( + rebar, + RB_INSERTBAND, + ( WPARAM )-1, + ( LPARAM )&rbbi3 + ); +} +*/ diff --git a/Externals/MusicMod/Player/Src/Rebar.h b/Externals/MusicMod/Player/Src/Rebar.h new file mode 100644 index 0000000000..e2f1feb685 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Rebar.h @@ -0,0 +1,40 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_TOOLBAR_H +#define PA_TOOLBAR_H + + + +#include "Global.h" + + + +extern HWND WindowRebar; +extern HWND WindowOrder; +extern HWND WindowEq; +extern HWND WindowSeek; + +extern HWND WindowVis; + +extern int iRebarHeight; + + + +namespace Toolbar +{ + bool Create(); +}; + + + +#endif // PA_TOOLBAR_H diff --git a/Externals/MusicMod/Player/Src/Resources/Buttons.bmp b/Externals/MusicMod/Player/Src/Resources/Buttons.bmp new file mode 100644 index 0000000000000000000000000000000000000000..3fb32f69cd10e66d25c5c4fd53e3403728275f08 GIT binary patch literal 734 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 literal 0 HcmV?d00001 diff --git a/Externals/MusicMod/Player/Src/Resources/resrc1.h b/Externals/MusicMod/Player/Src/Resources/resrc1.h new file mode 100644 index 0000000000..832fdbce47 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Resources/resrc1.h @@ -0,0 +1,17 @@ +//{{NO_DEPENDENCIES}} +// Microsoft Visual C++ generated include file. +// Used by resrc1.rc +// +#define IDB_BITMAP1 102 + +// Next default values for new objects +// +#ifdef APSTUDIO_INVOKED +#ifndef APSTUDIO_READONLY_SYMBOLS +#define _APS_NO_MFC 1 +#define _APS_NEXT_RESOURCE_VALUE 103 +#define _APS_NEXT_COMMAND_VALUE 40001 +#define _APS_NEXT_CONTROL_VALUE 1000 +#define _APS_NEXT_SYMED_VALUE 101 +#endif +#endif diff --git a/Externals/MusicMod/Player/Src/Resources/resrc1.rc b/Externals/MusicMod/Player/Src/Resources/resrc1.rc new file mode 100644 index 0000000000..83bbad7f33 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Resources/resrc1.rc @@ -0,0 +1,131 @@ +// Microsoft Visual C++ generated resource script. +// +#include "resrc1.h" + +#define APSTUDIO_READONLY_SYMBOLS +///////////////////////////////////////////////////////////////////////////// +// +// Generated from the TEXTINCLUDE 2 resource. +// +//#include "resource.h" +#include "afxres.h" + +///////////////////////////////////////////////////////////////////////////// +#undef APSTUDIO_READONLY_SYMBOLS + +///////////////////////////////////////////////////////////////////////////// +// Neutral resources + +#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_NEU) +#ifdef _WIN32 +LANGUAGE LANG_NEUTRAL, SUBLANG_NEUTRAL +#pragma code_page(1252) +#endif //_WIN32 + +///////////////////////////////////////////////////////////////////////////// +// +// Icon +// + +// Icon with lowest ID value placed first to ensure application icon +// remains consistent on all systems. +IDI_ICON1 ICON "Plainamp.ico" + +///////////////////////////////////////////////////////////////////////////// +// +// Version +// + +VS_VERSION_INFO VERSIONINFO + FILEVERSION 0,2,3,1 + PRODUCTVERSION 0,2,3,1 + FILEFLAGSMASK 0x17L +#ifdef _DEBUG + FILEFLAGS 0x1L +#else + FILEFLAGS 0x0L +#endif + FILEOS 0x4L + FILETYPE 0x1L + FILESUBTYPE 0x0L +BEGIN + BLOCK "StringFileInfo" + BEGIN + BLOCK "000004b0" + BEGIN + VALUE "CompanyName", "Hartwork Project (http://www.hartwork.org)" + VALUE "FileDescription", "Plainamp" + VALUE "FileVersion", "0, 2, 3, 1" + VALUE "InternalName", "Plainamp" + VALUE "LegalCopyright", "Copyright (C) 2005 Sebastian Pipping" + VALUE "ProductName", "Plainamp" + VALUE "ProductVersion", "0, 2, 3, 1" + END + END + BLOCK "VarFileInfo" + BEGIN + VALUE "Translation", 0x0, 1200 + END +END + + +///////////////////////////////////////////////////////////////////////////// +// +// Bitmap +// + +IDB_BITMAP1 BITMAP "Buttons.bmp" +#endif // Neutral resources +///////////////////////////////////////////////////////////////////////////// + + +///////////////////////////////////////////////////////////////////////////// +// German (Germany) resources + +#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_DEU) +#ifdef _WIN32 +LANGUAGE LANG_GERMAN, SUBLANG_GERMAN +#pragma code_page(1252) +#endif //_WIN32 + +#ifdef APSTUDIO_INVOKED +///////////////////////////////////////////////////////////////////////////// +// +// TEXTINCLUDE +// + +1 TEXTINCLUDE +BEGIN + "resrc1.h\0" +END + +2 TEXTINCLUDE +BEGIN + "#include ""resource.h""\r\n" + "#include ""afxres.h""\r\n" + "\0" +END + +3 TEXTINCLUDE +BEGIN + "\r\n" + "\0" +END + +#endif // APSTUDIO_INVOKED + +#endif // German (Germany) resources +///////////////////////////////////////////////////////////////////////////// + + + +#ifndef APSTUDIO_INVOKED +///////////////////////////////////////////////////////////////////////////// +// +// Generated from the TEXTINCLUDE 3 resource. +// + + +///////////////////////////////////////////////////////////////////////////// +#endif // not APSTUDIO_INVOKED + diff --git a/Externals/MusicMod/Player/Src/Status.cpp b/Externals/MusicMod/Player/Src/Status.cpp new file mode 100644 index 0000000000..b12b98e128 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Status.cpp @@ -0,0 +1,80 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Status.h" +#include "Main.h" +#include "Util.h" +#include "GlobalVersion.h" + + + +int iStatusHeight = 40; // extern +HWND WindowStatus = NULL; // extern + +const TCHAR * const szStatusDefault = TEXT( " " ) PLAINAMP_LONG_TITLE; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool BuildMainStatus() +{ + LoadCommonControls(); + + WindowStatus = CreateWindowEx( + 0, + STATUSCLASSNAME, + szStatusDefault, + WS_CHILD | + WS_VISIBLE, + CW_USEDEFAULT, + CW_USEDEFAULT, + CW_USEDEFAULT, + CW_USEDEFAULT, + WindowMain, + NULL, + g_hInstance, + NULL + ); + + if( !WindowStatus ) return false; + + RECT r = { 0, 0, 0, 0 }; + GetWindowRect( WindowStatus, &r ); + iStatusHeight = r.bottom - r.top; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool StatusUpdate( TCHAR * szText ) +{ + if( !WindowStatus ) return false; + SetWindowText( WindowStatus, szText ); + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void StatusReset() +{ + if( !WindowStatus ) return; + SetWindowText( WindowStatus, szStatusDefault ); +} diff --git a/Externals/MusicMod/Player/Src/Status.h b/Externals/MusicMod/Player/Src/Status.h new file mode 100644 index 0000000000..08e12d3527 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Status.h @@ -0,0 +1,33 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_STATUS_H +#define PA_STATUS_H + + + +#include "Global.h" + + + +extern int iStatusHeight; +extern HWND WindowStatus; + + + +bool BuildMainStatus(); +bool StatusUpdate( TCHAR * szText ); +void StatusReset(); + + + +#endif // PA_STATUS_H diff --git a/Externals/MusicMod/Player/Src/Timer.cpp b/Externals/MusicMod/Player/Src/Timer.cpp new file mode 100644 index 0000000000..ca6aa21526 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Timer.cpp @@ -0,0 +1,126 @@ +// ================================================================================================ +// File description +// ---------------- +/* In the GUI build there is a timer that is initiated with SetTimer() that will setup a timer that + calls the lpTimerFunc function pointer as long as your program is running a message loop. If it + doesn't run one, like in a console application (the NOGUI build), you'll have to use a different + kind of timer API, like timeSetEvent() or CreateThreadPoolTimer(). These timers use a different + thread to make the callback so be careful to properly lock and synchronize. */ +// ================================================================================================ + +// ================================================================================================ +// Library requirements +// ---------------- +// This program needs winmm.lib. There's no simpler or better way to make a timer withouth it. +// ================================================================================================ + +// ================================================================================================ +// Includes +// ---------------- +#include +//using namespace std; +#include +#include +#include "Global.h" +#include "PlayerExport.h" + +#include "InputPlugin.h" +// ================================================================================================ + + +//////////////////////////////////////////////////////////////////////////////////////////// +// Declarations and definitions +// ŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻ + +void MakeTime(); + +int g_Stop = 0; +extern std::string CurrentlyPlayingFile; +extern bool GlobalPause; +/////////////////////////////////// + +//////////////////////////////////////////////////////////////////////////////////////////// +// Manage restart when playback for a file has reached the end of the file +// ŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻ +#ifdef _M_X64 + void CALLBACK Update() +#else + void CALLBACK Update(unsigned int,unsigned int,unsigned long,unsigned long,unsigned long) +#endif +{ + //INFO_LOG(AUDIO,"DLL > Update() > Begin (%i)\n", active_input_plugin); + + // -------------------------------- + // Manage restart when playback for a file has reached the end of the file + // -------------- + // Check if the input plugin is activated + if(!active_input_plugin || !active_input_plugin->plugin) + { + //INFO_LOG(AUDIO,"The input plugin is not activated yet\n"); + } + else + { + const int ms_len = active_input_plugin->plugin->GetLength(); + const int ms_cur = active_input_plugin->plugin->GetOutputTime(); + + // Get the current playback progress + float progress; + if(ms_len > 0) + progress = (float)ms_cur / ms_len; + else + progress = 0; + + if ( progress > 0.7 ) // Only show this if we are getting close to the end, for bugtesting + // basically + { + //INFO_LOG(AUDIO,"Playback progress <%i of %i>\n", ms_cur, ms_len); + } + + // Because cur never go all the way to len we can't use a == comparison. Insted of this + // we could also check if the location is the same as right before. + if(ms_cur > ms_len - 1000 && !GlobalPause) // avoid restarting in cases where we just pressed pause + { + INFO_LOG(AUDIO,"Restart <%s>\n", CurrentlyPlayingFile.c_str()); + Player_Play((char *)CurrentlyPlayingFile.c_str()); + } + } + // -------------- + + //INFO_LOG(AUDIO,"Make new time\n"); + MakeTime(); // Make a new one +} + +void MakeTime() +{ + timeSetEvent( + 2000, // Interval in ms + 0, + #ifdef _M_X64 + (LPTIMECALLBACK) Update, // The function + #else + Update, + #endif + 0, + 0); +} +/////////////////////////// + + + +//////////////////////////////////////////////////////////////////////////////////////////// +// Start the timer +// ŻŻŻŻŻŻŻŻŻŻŻŻŻŻŻ +int MainTimer() +{ + MakeTime(); + + //while( g_Stop == 0 ) + //{ + // cout << "."; + //} + + //INFO_LOG(AUDIO,"MakeTime\n"); + + return 0; +} +/////////////////////////// diff --git a/Externals/MusicMod/Player/Src/Unicode.cpp b/Externals/MusicMod/Player/Src/Unicode.cpp new file mode 100644 index 0000000000..820895d93e --- /dev/null +++ b/Externals/MusicMod/Player/Src/Unicode.cpp @@ -0,0 +1,77 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Unicode.h" + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ToAnsi( char * szDest, wchar_t * szSource, int iLen ) +{ + char * const szBytesource = ( char * )szSource; + for( int i = 0; i < iLen; i++ ) + { + szDest[ i ] = szBytesource[ 2 * i + 1 ]; + } +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +/* +void ToUnicode( wchar_t * szDest, char * szSource, int iLen ) +{ + for( int i = 0; i < iLen; i++ ) + { + szDest[ i ] = ( wchar_t )szSource[ i ]; + } +} +*/ + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ToTchar( TCHAR * szDest, wchar_t * szSource, int iLen ) +{ +#ifdef PA_UNICODE + memcpy( szDest, szSource, 2 * iLen ); +#else + char * const stByteSource = ( TCHAR * )szSource; + for( int i = 0; i < iLen; i++ ) + { + szDest[ i ] = stByteSource[ 2 * i + 1 ]; + } +#endif +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void ToTchar( TCHAR * szDest, char * szSource, int iLen ) +{ +#ifdef PA_UNICODE + for( int i = 0; i < iLen; i++ ) + { + szDest[ i ] = szSource[ 2 * i + 1 ]; + } +#else + memcpy( szDest, szSource, iLen ); +#endif +} diff --git a/Externals/MusicMod/Player/Src/Unicode.h b/Externals/MusicMod/Player/Src/Unicode.h new file mode 100644 index 0000000000..10a53e873c --- /dev/null +++ b/Externals/MusicMod/Player/Src/Unicode.h @@ -0,0 +1,29 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_UNICODE_H +#define PA_UNICODE_H + + + +#include "Global.h" + + + +void ToAnsi( char * szDest, wchar_t * szSource, int iLen ); +// void ToUnicode( wchar_t * szDest, char * szSource, int iLen ) +void ToTchar( TCHAR * szDest, wchar_t * szSource, int iLen ); +void ToTchar( TCHAR * szDest, char * szSource, int iLen ); + + + +#endif // PA_UNICODE_H diff --git a/Externals/MusicMod/Player/Src/Util.cpp b/Externals/MusicMod/Player/Src/Util.cpp new file mode 100644 index 0000000000..26c5deb3dc --- /dev/null +++ b/Externals/MusicMod/Player/Src/Util.cpp @@ -0,0 +1,48 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Util.h" + + + +#ifndef ICC_STANDARD_CLASSES +# define ICC_STANDARD_CLASSES 0x00004000 +#endif + + + +bool bLoaded = false; +bool bAvailable = false; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool LoadCommonControls() +{ + if( bLoaded ) return bAvailable; + + INITCOMMONCONTROLSEX icce = { + sizeof( INITCOMMONCONTROLSEX ), + ICC_BAR_CLASSES | // Statusbar, trackbar, toolbar + ICC_COOL_CLASSES | // Rebar + ICC_LISTVIEW_CLASSES | // Listview + ICC_STANDARD_CLASSES | // + ICC_TREEVIEW_CLASSES // Treeview + }; + + bLoaded = true; + bAvailable = ( InitCommonControlsEx( &icce ) == TRUE ); + + return bAvailable; +} diff --git a/Externals/MusicMod/Player/Src/Util.h b/Externals/MusicMod/Player/Src/Util.h new file mode 100644 index 0000000000..5bee2a9d2c --- /dev/null +++ b/Externals/MusicMod/Player/Src/Util.h @@ -0,0 +1,26 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_UTIL_H +#define PA_UTIL_H + + + +#include "Global.h" + + + +bool LoadCommonControls(); + + + +#endif // PA_UTIL_H diff --git a/Externals/MusicMod/Player/Src/VisCache.cpp b/Externals/MusicMod/Player/Src/VisCache.cpp new file mode 100644 index 0000000000..37d5d1497b --- /dev/null +++ b/Externals/MusicMod/Player/Src/VisCache.cpp @@ -0,0 +1,303 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "VisCache.h" +#include "Console.h" +#include "malloc.h" + + + +unsigned char * SpecCacheLeft; +unsigned char * SpecCacheRight; +unsigned char * WaveCacheLeft; +unsigned char * WaveCacheRight; + +int iWritePos; +int iWriteOffset; // == iWritePos * 576 + +int iVisLatency; +int iDataFps; +int iCacheLen; + +bool bReady = false; + +int iReadTimeMs = 0; +int iWriteTimeMs = 0; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache_Resize( int iLatency, int iFps ) +{ + // if( !bReady ) return; + + const int iNewCacheLen = ( iFps * iLatency ) / 1000 + 1; +/* +TCHAR szBuffer[ 5000 ]; +_stprintf( szBuffer, TEXT( "RESIZE ( %i * %i ) / 1000 + 1 === %i" ), iFps, iLatency, iNewCacheLen ); +Console::Append( szBuffer ); +*/ + const int iByteNewCacheLen = iNewCacheLen * 576; + if( !iCacheLen ) + { + // First time + SpecCacheLeft = ( unsigned char * )malloc( iByteNewCacheLen ); +// memset( SpecCacheLeft, 0, iByteNewCacheLen ); + SpecCacheRight = ( unsigned char * )malloc( iByteNewCacheLen ); +// memset( SpecCacheRight, 0, iByteNewCacheLen ); + WaveCacheLeft = ( unsigned char * )malloc( iByteNewCacheLen ); +// memset( WaveCacheLeft, 0, iByteNewCacheLen ); + WaveCacheRight = ( unsigned char * )malloc( iByteNewCacheLen ); +// memset( WaveCacheRight, 0, iByteNewCacheLen ); + } + else if( iNewCacheLen > iCacheLen ) + { + // Grow + const int iByteCacheLen = iCacheLen * 576; + const int iByteClearLen = ( iNewCacheLen - iCacheLen ) * 576; + + SpecCacheLeft = ( unsigned char * )realloc( SpecCacheLeft, iByteNewCacheLen ); +// memset( SpecCacheLeft + iByteCacheLen, 0, iByteClearLen ); + SpecCacheRight = ( unsigned char * )realloc( SpecCacheRight, iByteNewCacheLen ); +// memset( SpecCacheRight + iByteCacheLen, 0, iByteClearLen ); + WaveCacheLeft = ( unsigned char * )realloc( WaveCacheLeft, iByteNewCacheLen ); +// memset( WaveCacheLeft + iByteCacheLen, 0, iByteClearLen ); + WaveCacheRight = ( unsigned char * )realloc( WaveCacheRight, iByteNewCacheLen ); +// memset( WaveCacheRight + iByteCacheLen, 0, iByteClearLen ); + } + + iCacheLen = iNewCacheLen; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::Create() +{ + if( bReady ) return; + + iWritePos = 0; + iWriteOffset = 0; + + iVisLatency = 50; + iDataFps = 40; + iCacheLen = 0; + + bReady = true; + + + VisCache_Resize( iVisLatency, iDataFps ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::Destroy() +{ + if( !bReady ) return; + + if( SpecCacheLeft ) free( SpecCacheLeft ); + if( SpecCacheRight ) free( SpecCacheRight ); + if( WaveCacheLeft ) free( WaveCacheLeft ); + if( WaveCacheRight ) free( WaveCacheRight ); + + bReady = false; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::EnsureLatency( int iLatency ) +{ + if( !bReady ) return; + + if( iLatency <= iVisLatency ) return; + + VisCache_Resize( + iLatency, + iDataFps + ); + iVisLatency = iLatency; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::EnsureDataFps( int iFps ) +{ + if( !bReady ) return; + if( iFps <= iDataFps ) return; + + VisCache_Resize( + iVisLatency, + iFps + ); + iDataFps = iFps; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::Clean() +{ + if( !bReady ) return; + + const int iByteCacheLen = iCacheLen * 576; + memset( SpecCacheLeft, 0, iByteCacheLen ); + memset( SpecCacheRight, 0, iByteCacheLen ); + memset( WaveCacheLeft, 0, iByteCacheLen ); + memset( WaveCacheRight, 0, iByteCacheLen ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::SetReadTime( int ms ) +{ + iReadTimeMs = ms; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::SetWriteTime( int ms ) +{ + iWriteTimeMs = ms; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int VisCache::LatencyToOffset( int iLatency ) +{ + int iFrame = iWritePos - 1 - ( ( iWriteTimeMs - iReadTimeMs - iLatency ) * iDataFps ) / 1000; + if( iFrame < 0 ) iFrame += iCacheLen; + return iFrame * 576; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::NextFrame() +{ + iWritePos++; + if( iWritePos >= iCacheLen ) iWritePos = 0; + iWriteOffset = iWritePos * 576; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::PutSpecLeft( unsigned char * data ) +{ + if( !bReady ) return; + memcpy( SpecCacheLeft + iWriteOffset, data, 576 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::PutSpecRight( unsigned char * data ) +{ + if( !bReady ) return; + memcpy( SpecCacheRight + iWriteOffset, data, 576 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::PutWaveLeft( unsigned char * data ) +{ + if( !bReady ) return; + memcpy( WaveCacheLeft + iWriteOffset, data, 576 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::PutWaveRight( unsigned char * data ) +{ + if( !bReady ) return; + memcpy( WaveCacheRight + iWriteOffset, data, 576 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::GetSpecLeft( unsigned char * dest, int iOffset ) +{ + if( !bReady ) return; + memcpy( dest, SpecCacheLeft + iOffset, 576 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::GetSpecRight( unsigned char * dest, int iOffset ) +{ + if( !bReady ) return; + memcpy( dest, SpecCacheRight + iOffset, 576 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::GetWaveLeft( unsigned char * dest, int iOffset ) +{ + if( !bReady ) return; + memcpy( dest, WaveCacheLeft + iOffset, 576 ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void VisCache::GetWaveRight( unsigned char * dest, int iOffset ) +{ + if( !bReady ) return; + memcpy( dest, WaveCacheRight + iOffset, 576 ); +} diff --git a/Externals/MusicMod/Player/Src/VisCache.h b/Externals/MusicMod/Player/Src/VisCache.h new file mode 100644 index 0000000000..f093d5d089 --- /dev/null +++ b/Externals/MusicMod/Player/Src/VisCache.h @@ -0,0 +1,50 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_VIS_CACHE_H +#define PA_VIS_CACHE_H + + +#include "Global.h" + + + +namespace VisCache +{ + void Create(); + void Destroy(); + + void EnsureLatency( int iLatency ); // TODO + void EnsureDataFps( int iFps ); // TODO + + void Clean(); + + void SetReadTime( int ms ); + void SetWriteTime( int ms ); + + int LatencyToOffset( int iLatency ); + void NextFrame(); + + void PutSpecLeft( unsigned char * data ); + void PutSpecRight( unsigned char * data ); + void PutWaveLeft( unsigned char * data ); + void PutWaveRight( unsigned char * data ); + + void GetSpecLeft( unsigned char * dest, int iOffset ); + void GetSpecRight( unsigned char * dest, int iOffset ); + void GetWaveLeft( unsigned char * dest, int iOffset ); + void GetWaveRight( unsigned char * dest, int iOffset ); +}; + + + +#endif // PA_VIS_CACHE_H diff --git a/Externals/MusicMod/Player/Src/VisModule.cpp b/Externals/MusicMod/Player/Src/VisModule.cpp new file mode 100644 index 0000000000..b171ed7b3f --- /dev/null +++ b/Externals/MusicMod/Player/Src/VisModule.cpp @@ -0,0 +1,358 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "VisModule.h" +#include "Console.h" +#include "Unicode.h" +#include "Playback.h" +#include "VisCache.h" +#include "PluginManager.h" +#include + + +VisModule ** active_vis_mods = NULL; // extern +int active_vis_count = 0; // extern + + +/* +BOOL CALLBACK EnumThreadWndProc( HWND hwnd, LPARAM lp ) +{ +// MessageBox( 0, "EnumThreadWndProc", "", 0 ); + DestroyWindow( hwnd ); +} +*/ + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +void PlugThread( PVOID pvoid ) +{ + // TODO: cleanup!!! + + Console::Append( TEXT( "Visualization thread born" ) ); + Console::Append( " " ); + + VisModule * mod = ( VisModule * )pvoid; + if( !mod ) return; + if( mod->mod->Init( mod->mod ) != 0 ) return; + + + VisCache::Create(); + + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + active_vis_count++; + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + + bool bKeepItGoing = true; + + + bool bQuitCalled = false; + + int iLast = GetTickCount(); + + // Message loop + MSG msg; + msg.message = WM_QUIT + 1; // Must be != WM_QUIT + do + { + if( PeekMessage( &msg, NULL, 0, 0, PM_REMOVE ) ) + { + if( msg.message == WM_QUIT ) + { +//////////////////////////////////////////////////////////////////////////////// + // Stop + if( !bQuitCalled ) + { + mod->mod->Quit( mod->mod ); + bQuitCalled = true; + } +//////////////////////////////////////////////////////////////////////////////// + + break; + } + + TranslateMessage( &msg ); + DispatchMessage( &msg ); + + if( msg.message == WM_CLOSE || ( ( msg.message == WM_SYSCOMMAND ) && ( msg.wParam == SC_CLOSE ) ) ) + { +//////////////////////////////////////////////////////////////////////////////// + // Stop + if( !bQuitCalled ) + { + mod->mod->Quit( mod->mod ); + bQuitCalled = true; + } +//////////////////////////////////////////////////////////////////////////////// + } + } + + if( bKeepItGoing ) + { + // Variant A + const int iNow = GetTickCount(); + if( iNow - iLast > mod->mod->delayMs ) + { + if( Playback::IsPlaying() ) + { + if( mod->bAllowRender ) + { + mod->bAllowRender = false; + + + const int iOffset = VisCache::LatencyToOffset( mod->mod->latencyMs ); + + switch( mod->mod->spectrumNch ) + { + case 2: + VisCache::GetSpecRight( mod->mod->spectrumData[ 1 ], iOffset ); + case 1: + VisCache::GetSpecLeft( mod->mod->spectrumData[ 0 ], iOffset ); + } + + switch( mod->mod->waveformNch ) + { + case 2: + VisCache::GetWaveRight( mod->mod->waveformData[ 1 ], iOffset ); + case 1: + VisCache::GetWaveLeft( mod->mod->waveformData[ 0 ], iOffset ); + } + + if( mod->mod->Render( mod->mod ) != 0 ) + { +//////////////////////////////////////////////////////////////////////////////// + // Stop + if( !bQuitCalled ) + { + // TODO: milkdrop doesn#t save window position + // when quit using manual plugin stop + + mod->mod->Quit( mod->mod ); + bQuitCalled = true; + } +//////////////////////////////////////////////////////////////////////////////// +/* + // Destroy all windows belonging to this thread + // This will lead to WM_QUIT automatically + EnumThreadWindows( GetCurrentThreadId(), EnumThreadWndProc, 0 ); + bKeepItGoing = false; +*/ + } + + + iLast = iNow; + } + } + else + { + if( mod->mod->Render( mod->mod ) != 0 ) + { +//////////////////////////////////////////////////////////////////////////////// + // Stop + if( !bQuitCalled ) + { + mod->mod->Quit( mod->mod ); + bQuitCalled = true; + } +//////////////////////////////////////////////////////////////////////////////// +/* + // Destroy all windows belonging to this thread + // This will lead to WM_QUIT automatically + EnumThreadWindows( GetCurrentThreadId(), EnumThreadWndProc, 0 ); + bKeepItGoing = false; +*/ + } + + iLast = iNow; + } + } + +//////////////////////////////////////////////////////////////////////////////// + bool bVisLockLeft = false; + VisLock.Enter(); +//////////////////////////////////////////////////////////////////////////////// + + if( mod->bShouldQuit ) + { + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Leave(); + bVisLockLeft = true; +//////////////////////////////////////////////////////////////////////////////// + + +//////////////////////////////////////////////////////////////////////////////// + // Stop + if( !bQuitCalled ) + { + mod->mod->Quit( mod->mod ); + bQuitCalled = true; + } +//////////////////////////////////////////////////////////////////////////////// +/* + // Destroy all windows belonging to this thread + // This will lead to WM_QUIT automatically + EnumThreadWindows( GetCurrentThreadId(), EnumThreadWndProc, 0 ); + bKeepItGoing = false; +*/ + } + +//////////////////////////////////////////////////////////////////////////////// + if( !bVisLockLeft ) + { + VisLock.Leave(); + } +//////////////////////////////////////////////////////////////////////////////// + } + + Sleep( 1 ); + } + while( msg.message != WM_QUIT ); + + mod->bShouldQuit = false; + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Enter(); + if( ( active_vis_count > 1 ) && ( mod->iArrayIndex < active_vis_count - 1 ) ) + { + active_vis_mods[ mod->iArrayIndex ] = active_vis_mods[ active_vis_count - 1 ]; + active_vis_mods[ mod->iArrayIndex ]->iArrayIndex = mod->iArrayIndex; + } + active_vis_count--; + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + mod->iArrayIndex = -1; + +/* + // Stop + mod->mod->Quit( mod->mod ); +*/ + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Enter(); + mod->bActive = false; + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + UpdatePluginStatus( mod->plugin, true, mod->plugin->IsActive() ); + + Console::Append( TEXT( "Visualization thread dead" ) ); + Console::Append( " " ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +VisModule::VisModule( char * szName, int iIndex, winampVisModule * mod, VisPlugin * plugin ) +{ + iArrayIndex = -1; + bActive = false; + bShouldQuit = false; + bAllowRender = false; + + iNameLen = ( int )strlen( szName ); + this->szName = new TCHAR[ iNameLen + 1 ]; + ToTchar( this->szName, szName, iNameLen ); + this->szName[ iNameLen ] = TEXT( '\0' ); + + this->iIndex = iIndex; + this->mod = mod; + this->plugin = plugin; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool VisModule::Start() +{ + if( !mod ) return false; + if( bActive ) return false; + if( plugin->IsActive() ) return false; + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Enter(); + if( !active_vis_count ) + { + active_vis_mods = new VisModule * [ 1 ]; + active_vis_mods[ 0 ] = this; + } + else + { + VisModule ** new_active_vis_mods = new VisModule * [ active_vis_count + 1 ]; + memcpy( new_active_vis_mods, active_vis_mods, active_vis_count * sizeof( VisModule * ) ); + new_active_vis_mods[ active_vis_count ] = this; + delete [] active_vis_mods; + active_vis_mods = new_active_vis_mods; + } + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + iArrayIndex = active_vis_count; + + + // Start + _beginthread( PlugThread, 1024 * 1024, ( PVOID )this ); + + bActive = true; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool VisModule::Stop() +{ + if( !mod ) return false; + if( !bActive ) return false; + if( !plugin->IsActive() ) return false; + +//////////////////////////////////////////////////////////////////////////////// + VisLock.Enter(); + bShouldQuit = true; + VisLock.Leave(); +//////////////////////////////////////////////////////////////////////////////// + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool VisModule::Config() +{ + if( !mod ) return false; + if( !mod->Config ) return false; + + mod->Config( mod ); + + return true; +} diff --git a/Externals/MusicMod/Player/Src/VisModule.h b/Externals/MusicMod/Player/Src/VisModule.h new file mode 100644 index 0000000000..23e13e7450 --- /dev/null +++ b/Externals/MusicMod/Player/Src/VisModule.h @@ -0,0 +1,71 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_VIS_MODULE_H +#define PA_VIS_MODULE_H + + + +#include "Global.h" +#include "VisPlugin.h" +#include + + + +class VisModule; +class VisPlugin; + +extern VisModule ** active_vis_mods; +extern int active_vis_count; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp visualization module wrapper +//////////////////////////////////////////////////////////////////////////////// +class VisModule +{ +public: + VisModule( char * szName, int iIndex, winampVisModule * mod, VisPlugin * plugin ); + + bool Start(); + bool Config(); + bool Stop(); + + inline bool IsActive() { return bActive; } + + inline TCHAR * GetName() { return szName; } + inline int GetNameLen() { return iNameLen; } + +private: + int iArrayIndex; + bool bActive; + bool bShouldQuit; + bool bAllowRender; + + TCHAR * szName; + int iNameLen; + + int iIndex; + winampVisModule * mod; + VisPlugin * plugin; + + + friend void PlugThread( PVOID pvoid ); + friend void VSAAdd( void * data, int timestamp ); + friend void VSAAddPCMData( void * PCMData, int nch, int bps, int timestamp ); + friend int VSAGetMode( int * specNch, int * waveNch ); +}; + + + +#endif // PA_VIS_MODULE_H diff --git a/Externals/MusicMod/Player/Src/VisPlugin.cpp b/Externals/MusicMod/Player/Src/VisPlugin.cpp new file mode 100644 index 0000000000..c25262e715 --- /dev/null +++ b/Externals/MusicMod/Player/Src/VisPlugin.cpp @@ -0,0 +1,189 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "VisPlugin.h" +#include "Unicode.h" +#include "Console.h" +#include "Main.h" + + + +vector vis_plugins; // extern + +Lock VisLock = Lock( TEXT( "PLAINAMP_VIS_LOCK" ) ); // extern + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +VisPlugin::VisPlugin( TCHAR * szDllpath, bool bKeepLoaded ) : Plugin( szDllpath ) +{ + header = NULL; + + if( !Load() ) + { + return; + } + + if( !bKeepLoaded ) + { + Unload(); + } + + vis_plugins.push_back( this ); +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool VisPlugin::Load() +{ + // (1) Load DLL + hDLL = LoadLibrary( GetFullpath() ); + if( !hDLL ) return false; + + // (2) Find export + WINAMP_VIS_GETTER winampGetVisHeader = + ( WINAMP_VIS_GETTER )GetProcAddress( hDLL, "winampVisGetHeader" ); + if( winampGetVisHeader == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + + // (3) Get header + header = winampGetVisHeader(); + if( header == NULL ) + { + FreeLibrary( hDLL ); + hDLL = NULL; + return false; + } + +//////////////////////////////////////////////////////////////////////////////// + // Forget old modules or we get them twice + if( !modules.empty() ) + { + modules.clear(); + } +//////////////////////////////////////////////////////////////////////////////// + + if( !szName ) + { + // Note: The prefix is not removed to hide their + // origin at Nullsoft! It just reads easier. + if( !strnicmp( header->description, "nullsoft ", 9 ) ) + { + header->description += 9; + } + iNameLen = ( int )strlen( header->description ); + szName = new TCHAR[ iNameLen + 1 ]; + ToTchar( szName, header->description, iNameLen ); + szName[ iNameLen ] = TEXT( '\0' ); + } + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Loading <%s>, %s" ), GetFilename(), szName ); + Console::Append( szBuffer ); + + // (4) Get modules + winampVisModule * mod; + int iFound = 0; + while( true ) + { + mod = header->getModule( iFound ); + if( !mod ) break; + + // (4a) Modify module + mod->hDllInstance = hDLL; + mod->hwndParent = WindowMain; + mod->sRate = 44100; // TODO + mod->nCh = 2; // TODO + + // (4b) Add module to list + VisModule * vismod = new VisModule( + mod->description, // char * szName + iFound, // UINT uIndex + mod, // winampVisModule * mod + this // VisPlugin * plugin + ); + modules.push_back( vismod ); + + iFound++; + + _stprintf( szBuffer, TEXT( " %s" ), vismod->GetName() ); + Console::Append( szBuffer ); + } + + Console::Append( TEXT( " " ) ); + + if( !iFound ) return false; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool VisPlugin::Unload() +{ + if( !IsLoaded() ) return true; + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Unloading <%s>" ), GetFilename() ); + Console::Append( szBuffer ); + Console::Append( TEXT( " " ) ); + printf( ">>>Unloading <%s>\n" , GetFilename() ); + + header = NULL; + + /* + TODO + VisModule * walk; + vector ::iterator iter = modules.begin(); + while( iter != modules.end() ) + { + walk = *iter; + delete [] walk->szName; + delete walk; + + iter++; + } + */ + + FreeLibrary( hDLL ); + hDLL = NULL; + + return true; +} + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +bool VisPlugin::IsActive() +{ + vector ::iterator iter = modules.begin(); + while( iter != modules.end() ) + { + if( ( *iter )->IsActive() ) return true; + iter++; + } + return false; +} diff --git a/Externals/MusicMod/Player/Src/VisPlugin.h b/Externals/MusicMod/Player/Src/VisPlugin.h new file mode 100644 index 0000000000..d28b1a5f8e --- /dev/null +++ b/Externals/MusicMod/Player/Src/VisPlugin.h @@ -0,0 +1,68 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_VIS_PLUGIN_H +#define PA_VIS_PLUGIN_H + + + +#include "Global.h" +#include "Plugin.h" +#include "Winamp/Vis.h" +#include "VisModule.h" +#include "Lock.h" +#include + +using namespace std; + + + +typedef winampVisHeader * ( * WINAMP_VIS_GETTER )( void ); + + + +class VisModule; +class VisPlugin; + +extern vector vis_plugins; +extern Lock VisLock; + + + +//////////////////////////////////////////////////////////////////////////////// +/// Winamp visualization plugin wrapper +//////////////////////////////////////////////////////////////////////////////// +class VisPlugin : public Plugin +{ +public: + VisPlugin( TCHAR * szDllpath, bool bKeepLoaded ); + + bool Load(); + bool Unload(); + + TCHAR * GetTypeString() { return TEXT( "Visual" ); } + int GetTypeStringLen() { return 6; } + PluginType GetType() { return PLUGIN_TYPE_VIS; } + + bool IsActive(); + +private: + winampVisHeader * header; + vector modules; + + + friend void ContextMenuVis( VisPlugin * vis, POINT * p ); +}; + + + +#endif // PA_VIS_PLUGIN_H diff --git a/Externals/MusicMod/Player/Src/Winamp.cpp b/Externals/MusicMod/Player/Src/Winamp.cpp new file mode 100644 index 0000000000..6e123b6355 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp.cpp @@ -0,0 +1,996 @@ +// ======================================================================================= +// WndprocWinamp is called repeatedly when the cursor is moved over the main window +// ======================================================================================= + +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Winamp.h" +#include "Playback.h" +#include "Playlist.h" +#include "Console.h" +#include "Main.h" +#include "Winamp/wa_ipc.h" +#include "Winamp/wa_msgids.h" +#include "AddDirectory.h" +#include "AddFiles.h" +#include "Prefs.h" +#include "PluginManager.h" +#include "Embed.h" +#include "Unicode.h" +#include "zlib/zlib.h" +#include "Rebar.h" + + + +int IPC_GENHOTKEYS_ADD = 0; +int ID_DYNAMICLIBRARY = 0; +int IPC_GETPLAINBARTARGET = 0; + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +LRESULT CALLBACK WndprocWinamp( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + //Console::Append( TEXT( "Winamp.cc:WndprocWinamp was called" ) ); + + switch( message ) + { + case WM_COMMAND: +/* + { + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "WM_COMMAND <%i> <%i>" ), wp, lp ); + Console::Append( szBuffer ); + } +*/ + switch( LOWORD( wp ) ) + { + case WINAMP_FILE_QUIT: + PostMessage( WindowMain, WM_SYSCOMMAND, SC_CLOSE, 0 ); + break; + + case WINAMP_OPTIONS_PREFS: + Prefs::Show(); + break; +/* + case WINAMP_OPTIONS_AOT: break; + case WINAMP_FILE_REPEAT: break; + case WINAMP_FILE_SHUFFLE: break; + case WINAMP_HIGH_PRIORITY: break; +*/ + case WINAMP_FILE_PLAY: + AddFiles(); + break; +/* + case WINAMP_OPTIONS_EQ: break; + case WINAMP_OPTIONS_ELAPSED: break; + case WINAMP_OPTIONS_REMAINING: break; + case WINAMP_OPTIONS_PLEDIT: break; +*/ + case WINAMP_HELP_ABOUT: + About( hwnd ); + break; + + case WINAMP_MAINMENU: + { + POINT p; + GetCursorPos( &p ); + + CheckMenuItem( + main_context_menu, + PLAINAMP_TOGGLE_CONSOLE, + IsWindowVisible( WindowConsole ) ? MF_CHECKED : MF_UNCHECKED + ); + CheckMenuItem( + main_context_menu, + PLAINAMP_TOGGLE_MANAGER, + IsWindowVisible( WindowManager ) ? MF_CHECKED : MF_UNCHECKED + ); + + TrackPopupMenu( + main_context_menu, + TPM_LEFTALIGN | + TPM_TOPALIGN | + TPM_RIGHTBUTTON, + p.x, + p.y, + 0, + hwnd, + NULL + ); + + break; + } + + case WINAMP_BUTTON1: + Playback::Prev(); + Playback::UpdateSeek(); + break; + + case WINAMP_BUTTON2: + Playback::Play(); + Playback::UpdateSeek(); + break; + + case WINAMP_BUTTON3: + Playback::Pause(); + Playback::UpdateSeek(); + break; + + case WINAMP_BUTTON4: + Playback::Stop(); + Playback::UpdateSeek(); + break; + + case WINAMP_BUTTON5: + Playback::Next(); + Playback::UpdateSeek(); + break; + +/* + case WINAMP_BUTTON1_SHIFT: break; + case WINAMP_BUTTON2_SHIFT: break; + case WINAMP_BUTTON3_SHIFT: break; + case WINAMP_BUTTON4_SHIFT: break; + case WINAMP_BUTTON5_SHIFT: break; + case WINAMP_BUTTON1_CTRL: break; + case WINAMP_BUTTON2_CTRL: break; + case WINAMP_BUTTON3_CTRL: break; + case WINAMP_BUTTON4_CTRL: break; + case WINAMP_BUTTON5_CTRL: break; +*/ + + case WINAMP_VOLUMEUP: + Playback::Volume::Up(); + // TODO Update slider + break; + + case WINAMP_VOLUMEDOWN: + Playback::Volume::Down(); + // TODO Update slider + break; + + case WINAMP_FFWD5S: + Playback::Forward(); + Playback::UpdateSeek(); + break; + + case WINAMP_REW5S: + Playback::Rewind(); + Playback::UpdateSeek(); + break; +/* + case WINAMP_NEXT_WINDOW: break; + case WINAMP_OPTIONS_WINDOWSHADE: break; + case WINAMP_OPTIONS_DSIZE: break; + case IDC_SORT_FILENAME: break; + case IDC_SORT_FILETITLE: break; + case IDC_SORT_ENTIREFILENAME: break; + case IDC_SELECTALL: break; + case IDC_SELECTNONE: break; + case IDC_SELECTINV: break; + case IDM_EQ_LOADPRE: break; + case IDM_EQ_LOADMP3: break; + case IDM_EQ_LOADDEFAULT: break; + case IDM_EQ_SAVEPRE: break; + case IDM_EQ_SAVEMP3: break; + case IDM_EQ_SAVEDEFAULT: break; + case IDM_EQ_DELPRE: break; + case IDM_EQ_DELMP3: break; + case IDC_PLAYLIST_PLAY: break; + case WINAMP_FILE_LOC: break; + case WINAMP_OPTIONS_EASYMOVE: break; +*/ + case WINAMP_FILE_DIR: + AddDirectory(); + break; +/* + case WINAMP_EDIT_ID3: break; + case WINAMP_TOGGLE_AUTOSCROLL: break; + case WINAMP_VISSETUP: break; + case WINAMP_PLGSETUP: break; + case WINAMP_VISPLUGIN: break; + case WINAMP_JUMP: break; + case WINAMP_JUMPFILE: break; + case WINAMP_JUMP10FWD: break; + case WINAMP_JUMP10BACK: break; + case WINAMP_PREVSONG: break; + case WINAMP_OPTIONS_EXTRAHQ: break; +*/ + case ID_PE_NEW: + playlist->RemoveAll(); + break; + + case ID_PE_OPEN: + Playlist::DialogOpen(); + break; +/* + case ID_PE_SAVE: break; +*/ + case ID_PE_SAVEAS: + Playlist::DialogSaveAs(); + break; + + case ID_PE_SELECTALL: + playlist->SelectAll( true ); + break; + + case ID_PE_INVERT: + playlist->SelectInvert(); + break; + + case ID_PE_NONE: + playlist->SelectAll( false ); + break; +/* + case ID_PE_ID3: break; + case ID_PE_S_TITLE: break; + case ID_PE_S_FILENAME: break; + case ID_PE_S_PATH: break; + case ID_PE_S_RANDOM: break; + case ID_PE_S_REV: break; +*/ + case ID_PE_CLEAR: + playlist->RemoveAll(); + break; +/* + case ID_PE_MOVEUP: break; + case ID_PE_MOVEDOWN: break; + case WINAMP_SELSKIN: break; + case WINAMP_VISCONF: break; + case ID_PE_NONEXIST: break; + case ID_PE_DELETEFROMDISK: break; + case ID_PE_CLOSE: break; + case WINAMP_VIS_SETOSC: break; + case WINAMP_VIS_SETANA: break; + case WINAMP_VIS_SETOFF: break; + case WINAMP_VIS_DOTSCOPE: break; + case WINAMP_VIS_LINESCOPE: break; + case WINAMP_VIS_SOLIDSCOPE: break; + case WINAMP_VIS_NORMANA: break; + case WINAMP_VIS_FIREANA: break; + case WINAMP_VIS_LINEANA: break; + case WINAMP_VIS_NORMVU: break; + case WINAMP_VIS_SMOOTHVU: break; + case WINAMP_VIS_FULLREF: break; + case WINAMP_VIS_FULLREF2: break; + case WINAMP_VIS_FULLREF3: break; + case WINAMP_VIS_FULLREF4: break; + case WINAMP_OPTIONS_TOGTIME: break; + case EQ_ENABLE: break; + case EQ_AUTO: break; + case EQ_PRESETS: break; + case WINAMP_VIS_FALLOFF0: break; + case WINAMP_VIS_FALLOFF1: break; + case WINAMP_VIS_FALLOFF2: break; + case WINAMP_VIS_FALLOFF3: break; + case WINAMP_VIS_FALLOFF4: break; + case WINAMP_VIS_PEAKS: break; + case ID_LOAD_EQF: break; + case ID_SAVE_EQF: break; + case ID_PE_ENTRY: break; + case ID_PE_SCROLLUP: break; + case ID_PE_SCROLLDOWN: break; + case WINAMP_MAIN_WINDOW: break; + case WINAMP_VIS_PFALLOFF0: break; + case WINAMP_VIS_PFALLOFF1: break; + case WINAMP_VIS_PFALLOFF2: break; + case WINAMP_VIS_PFALLOFF3: break; + case WINAMP_VIS_PFALLOFF4: break; + case ID_PE_TOP: break; + case ID_PE_BOTTOM: break; + case WINAMP_OPTIONS_WINDOWSHADE_PL: break; + case EQ_INC1: break; + case EQ_INC2: break; + case EQ_INC3: break; + case EQ_INC4: break; + case EQ_INC5: break; + case EQ_INC6: break; + case EQ_INC7: break; + case EQ_INC8: break; + case EQ_INC9: break; + case EQ_INC10: break; + case EQ_INCPRE: break; + case EQ_DECPRE: break; + case EQ_DEC1: break; + case EQ_DEC2: break; + case EQ_DEC3: break; + case EQ_DEC4: break; + case EQ_DEC5: break; + case EQ_DEC6: break; + case EQ_DEC7: break; + case EQ_DEC8: break; + case EQ_DEC9: break; + case EQ_DEC10: break; + case ID_PE_SCUP: break; + case ID_PE_SCDOWN: break; + case WINAMP_REFRESHSKIN: break; + case ID_PE_PRINT: break; + case ID_PE_EXTINFO: break; + case WINAMP_PLAYLIST_ADVANCE: break; + case WINAMP_VIS_LIN: break; + case WINAMP_VIS_BAR: break; + case WINAMP_OPTIONS_MINIBROWSER: break; + case MB_FWD: break; + case MB_BACK: break; + case MB_RELOAD: break; + case MB_OPENMENU: break; + case MB_OPENLOC: break; + case WINAMP_NEW_INSTANCE: break; + case MB_UPDATE: break; + case WINAMP_OPTIONS_WINDOWSHADE_EQ: break; + case EQ_PANLEFT: break; + case EQ_PANRIGHT: break; + case WINAMP_GETMORESKINS: break; + case WINAMP_VIS_OPTIONS: break; + case WINAMP_PE_SEARCH: break; + case ID_PE_BOOKMARK: break; + case WINAMP_EDIT_BOOKMARKS: break; + case WINAMP_MAKECURBOOKMARK: break; + case ID_MAIN_PLAY_BOOKMARK_NONE: break; + case ID_MAIN_PLAY_AUDIOCD: break; + case ID_MAIN_PLAY_AUDIOCD2: break; + case ID_MAIN_PLAY_AUDIOCD3: break; + case ID_MAIN_PLAY_AUDIOCD4: break; + case WINAMP_OPTIONS_VIDEO: break; + case ID_VIDEOWND_ZOOMFULLSCREEN: break; + case ID_VIDEOWND_ZOOM100: break; + case ID_VIDEOWND_ZOOM200: break; + case ID_VIDEOWND_ZOOM50: break; + case ID_VIDEOWND_VIDEOOPTIONS: break; + case WINAMP_MINIMIZE: break; + case ID_PE_FONTBIGGER: break; + case ID_PE_FONTSMALLER: break; + case WINAMP_VIDEO_TOGGLE_FS: break; + case WINAMP_VIDEO_TVBUTTON: break; + case WINAMP_LIGHTNING_CLICK: break; + case ID_FILE_ADDTOLIBRARY: break; + case ID_HELP_HELPTOPICS: break; + case ID_HELP_GETTINGSTARTED: break; + case ID_HELP_WINAMPFORUMS: break; +*/ + case ID_PLAY_VOLUMEUP: + Playback::Volume::Up(); + // TODO Update slider + break; + + case ID_PLAY_VOLUMEDOWN: + Playback::Volume::Down(); + // TODO Update slider + break; +/* + case ID_PEFILE_OPENPLAYLISTFROMLIBRARY_NOPLAYLISTSINLIBRARY: break; + case ID_PEFILE_ADDFROMLIBRARY: break; + case ID_PEFILE_CLOSEPLAYLISTEDITOR: break; + case ID_PEPLAYLIST_PLAYLISTPREFERENCES: break; + case ID_MLFILE_NEWPLAYLIST: break; + case ID_MLFILE_LOADPLAYLIST: break; + case ID_MLFILE_SAVEPLAYLIST: break; + case ID_MLFILE_ADDMEDIATOLIBRARY: break; + case ID_MLFILE_CLOSEMEDIALIBRARY: break; + case ID_MLVIEW_NOWPLAYING: break; + case ID_MLVIEW_LOCALMEDIA_ALLMEDIA: break; + case ID_MLVIEW_LOCALMEDIA_AUDIO: break; + case ID_MLVIEW_LOCALMEDIA_VIDEO: break; + case ID_MLVIEW_PLAYLISTS_NOPLAYLISTINLIBRARY: break; + case ID_MLVIEW_INTERNETRADIO: break; + case ID_MLVIEW_INTERNETTV: break; + case ID_MLVIEW_LIBRARYPREFERENCES: break; + case ID_MLVIEW_DEVICES_NOAVAILABLEDEVICE: break; + case ID_MLFILE_IMPORTCURRENTPLAYLIST: break; + case ID_MLVIEW_MEDIA: break; + case ID_MLVIEW_PLAYLISTS: break; + case ID_MLVIEW_MEDIA_ALLMEDIA: break; + case ID_MLVIEW_DEVICES: break; + case ID_FILE_SHOWLIBRARY: break; + case ID_FILE_CLOSELIBRARY: break; + case ID_POST_PLAY_PLAYLIST: break; + case ID_VIS_NEXT: break; + case ID_VIS_PREV: break; + case ID_VIS_RANDOM: break; + case ID_MANAGEPLAYLISTS: break; + case ID_PREFS_SKIN_SWITCHTOSKIN: break; + case ID_PREFS_SKIN_DELETESKIN: break; + case ID_PREFS_SKIN_RENAMESKIN: break; + case ID_VIS_FS: break; + case ID_VIS_CFG: break; + case ID_VIS_MENU: break; + case ID_VIS_SET_FS_FLAG: break; + case ID_PE_SHOWPLAYING: break; + case ID_HELP_REGISTERWINAMPPRO: break; + case ID_PE_MANUAL_ADVANCE: break; + case WA_SONG_5_STAR_RATING: break; + case WA_SONG_4_STAR_RATING: break; + case WA_SONG_3_STAR_RATING: break; + case WA_SONG_2_STAR_RATING: break; + case WA_SONG_1_STAR_RATING: break; + case WA_SONG_NO_RATING: break; + case PL_SONG_5_STAR_RATING: break; + case PL_SONG_4_STAR_RATING: break; + case PL_SONG_3_STAR_RATING: break; + case PL_SONG_2_STAR_RATING: break; + case PL_SONG_1_STAR_RATING: break; + case PL_SONG_NO_RATING: break; + case AUDIO_TRACK_ONE: break; + case VIDEO_TRACK_ONE: break; + case ID_SWITCH_COLOURTHEME: break; + case ID_GENFF_LIMIT: break; + +*/ + case PLAINAMP_TOGGLE_CONSOLE: + ShowWindow( WindowConsole, IsWindowVisible( WindowConsole ) ? SW_HIDE : SW_SHOW ); + break; + + case PLAINAMP_TOGGLE_MANAGER: + ShowWindow( WindowManager, IsWindowVisible( WindowManager ) ? SW_HIDE : SW_SHOW ); + break; + + case PLAINAMP_PL_REM_SEL: + playlist->RemoveSelected( true ); + break; + + case PLAINAMP_PL_REM_CROP: + playlist->RemoveSelected( false ); + break; + + default: + { + /* + if( wp == ID_DYNAMICLIBRARY ) + { + // Stupid dnylib workaround + PostMessage( hwnd, WM_COMMAND, ID_DYNAMICLIBRARY | ( 1 << 16 ), 0 ); + } + */ + + if( LOWORD( wp ) < 40001 ) break; + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "WM_COMMAND <%i> <%i>" ), wp, lp ); + Console::Append( szBuffer ); + Console::Append( TEXT( "NOT handled" ) ); + Console::Append( " " ); + } + + } + break; + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + + case WM_WA_IPC: +/* + { + TCHAR szBuffer[ 5000 ]; + if( lp != 2006 ) + { + _stprintf( szBuffer, TEXT( "WM_WA_IPC <%i> <%i>" ), wp, lp ); + } + else + { + // Hotkey sent as strings! + // Idea: make stl map + _stprintf( szBuffer, TEXT( "WM_WA_IPC <%s> <%i>" ), ( char * )wp, lp ); + } + Console::Append( szBuffer ); + } +*/ + switch( lp ) + { + case IPC_GETVERSION: + return 0x5010; // 5.10 +/* + case IPC_GETREGISTEREDVERSION: break; + case IPC_PLAYFILE: break; + case IPC_ENQUEUEFILE: break; +*/ + case IPC_DELETE: + playlist->RemoveAll(); + break; +/* + case IPC_DELETE_INT: break; +*/ + case IPC_STARTPLAY: + Playback::Play(); + Playback::UpdateSeek(); + break; +/* + case IPC_STARTPLAY_INT: break; + case IPC_CHDIR: break; +*/ + case IPC_ISPLAYING: // untested + return ( Playback::IsPlaying() ? ( Playback::IsPaused() ? 3 : 1 ) : 0 ); +/* + case IPC_GETOUTPUTTIME: break; + case IPC_JUMPTOTIME: break; + case IPC_GETMODULENAME: break; + case IPC_EX_ISRIGHTEXE: break; + case IPC_WRITEPLAYLIST: break; +*/ + case IPC_SETPLAYLISTPOS: + playlist->SetCurIndex( ( int )wp ); + break; + + case IPC_SETVOLUME: break; + Playback::Volume::Set( ( int )wp ); + // TODO Update slider + break; + + case IPC_SETPANNING: break; + Playback::Pan::Set( ( int )wp ); + // TODO Update slider + break; + + case IPC_GETLISTLENGTH: + return playlist->GetSize(); + + case IPC_GETLISTPOS: + return playlist->GetCurIndex(); +/* + case IPC_GETINFO: break; + case IPC_GETEQDATA: break; + // TODO + case IPC_SETEQDATA: break; + // TODO + case IPC_ADDBOOKMARK: break; + case IPC_INSTALLPLUGIN: break; + case IPC_RESTARTWINAMP: break; + case IPC_ISFULLSTOP: break; + case IPC_INETAVAILABLE: break; + case IPC_UPDTITLE: break; + case IPC_REFRESHPLCACHE: break; + case IPC_GET_SHUFFLE: break; + case IPC_GET_REPEAT: break; + case IPC_SET_SHUFFLE: break; + case IPC_SET_REPEAT: break; + case IPC_ENABLEDISABLE_ALL_WINDOWS: break; +*/ + case IPC_GETWND: + { + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "IPC_GETWND <%i>" ), wp ); + Console::Append( szBuffer ); + Console::Append( " " ); + } + + switch( wp ) + { + case IPC_GETWND_EQ: break; + case IPC_GETWND_PE: return ( LRESULT )WindowMain; + case IPC_GETWND_MB: break; + case IPC_GETWND_VIDEO: break; + } + + return ( LRESULT )NULL; + + case IPC_ISWNDVISIBLE: + { + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "IPC_ISWNDVISIBLE <%i>" ), wp ); + Console::Append( szBuffer ); + Console::Append( " " ); + } + + switch( wp ) + { + case IPC_GETWND_EQ: break; + case IPC_GETWND_PE: return 1; + case IPC_GETWND_MB: break; + case IPC_GETWND_VIDEO: break; + } + + return 0; +/* + case IPC_SETSKIN: break; + case IPC_GETSKIN: break; + case IPC_EXECPLUG: break; +*/ + case IPC_GETPLAYLISTFILE: + { + static char szAnsiFilename[ 2000 ] = "\0"; + Playlist::GetFilename( ( int )wp, szAnsiFilename, 1999 ); + return ( LRESULT )szAnsiFilename; + } + + case IPC_GETPLAYLISTTITLE: + { + static char szAnsiTitle[ 2000 ] = "\0"; + Playlist::GetTitle( ( int )wp, szAnsiTitle, 1999 ); + return ( LRESULT )szAnsiTitle; + } +/* + case IPC_GETHTTPGETTER: break; + case IPC_MBOPEN: break; + case IPC_CHANGECURRENTFILE: break; + case IPC_GETMBURL: break; + case IPC_MBBLOCK: break; + case IPC_MBOPENREAL: break; + case IPC_ADJUST_OPTIONSMENUPOS: break; +*/ + case IPC_GET_HMENU: + { + switch( wp ) + { + case 0: + return ( LRESULT )main_context_menu; +/* + case 1: break; + case 2: break; + case 3: break; + case 4: break; +*/ + } + return ( LRESULT )NULL; + } + + case IPC_GET_EXTENDED_FILE_INFO: + Console::Append( "IPC_GET_EXTENDED_FILE_INFO" ); + Console::Append( TEXT( "NOT handled" ) ); + Console::Append( " " ); + break; +/* + case IPC_GET_EXTENDED_FILE_INFO_HOOKABLE: break; + case IPC_GET_BASIC_FILE_INFO: break; +*/ + case IPC_GET_EXTLIST: + // TODO + return ( LRESULT )GlobalAlloc( GMEM_ZEROINIT, 2 ); // "\0\0" +/* + case IPC_INFOBOX: break; + case IPC_SET_EXTENDED_FILE_INFO: break; + case IPC_WRITE_EXTENDED_FILE_INFO: break; + case IPC_FORMAT_TITLE: break; +*/ +// ======================================================================================= +// Let's remove this +/* + case IPC_GETUNCOMPRESSINTERFACE: + if( wp == 0x10100000 ) + { + Console::Append( "IPC_GETUNCOMPRESSINTERFACE @ wa_inflate_struct" ); + Console::Append( TEXT( "NOT handled" ) ); + Console::Append( " " ); + } + else + { + Console::Append( "IPC_GETUNCOMPRESSINTERFACE @ zlib" ); + Console::Append( " " ); + return ( LRESULT )uncompress; + } + break; +*/ +// ======================================================================================= + + case IPC_ADD_PREFS_DLG: + Prefs::AddPage( ( prefsDlgRec * )wp ); + break; +/* + case IPC_REMOVE_PREFS_DLG: break; +*/ + case IPC_OPENPREFSTOPAGE: + Prefs::Show( ( prefsDlgRec * )wp ); + break; + + case IPC_GETINIFILE: + { + static char szWinampInipath[ MAX_PATH ] = ""; + if( *szWinampInipath == '\0' ) + { + GetModuleFileNameA( NULL, szWinampInipath, MAX_PATH - 1 ); + char * szWalk = szWinampInipath + strlen( szWinampInipath ) - 1; + while( ( szWalk > szWinampInipath ) && ( *szWalk != '.' ) ) szWalk--; + szWalk++; + strcpy( szWalk, "ini" ); + } + return ( LRESULT )szWinampInipath; + } + + case IPC_GETINIDIRECTORY: + { + // TODO: trailing slash or not??? + static char szPluginInipath[ MAX_PATH ] = ""; + if( *szPluginInipath == '\0' ) + { + GetModuleFileNameA( NULL, szPluginInipath, MAX_PATH - 1 ); + char * szWalk = szPluginInipath + strlen( szPluginInipath ) - 1; + while( ( szWalk > szPluginInipath ) && ( *szWalk != '\\' ) ) szWalk--; + szWalk++; + strcpy( szWalk, TEXT( "Plugins" ) ); + } + return ( LRESULT )szPluginInipath; + } +/* + case IPC_SPAWNBUTTONPOPUP: break; + case IPC_OPENURLBOX: break; +*/ + case IPC_OPENFILEBOX: + AddFiles(); + break; + + case IPC_OPENDIRBOX: + AddDirectory(); + break; + + case IPC_GET_GENSKINBITMAP: + { + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "IPC_GET_GENSKINBITMAP <%i>" ), wp ); + Console::Append( szBuffer ); + + switch( wp ) + { + case 0: + { + // TODO: make it work + + // Gen.bmp? + static HBITMAP hBitmap = NULL; + static BYTE * image = NULL; + if( !hBitmap ) + { + const int iWidth = 200; // 194; + const int iHeight = 300; // 109; + const int bpp = 24; + int bytes_per_row = iWidth * ( bpp / 8 ); + const int diff = bytes_per_row % 4; + if( diff ) bytes_per_row += ( 4 - diff ); + + const int size_in_bytes = bytes_per_row * iHeight; + image = new BYTE[ size_in_bytes ]; + + hBitmap = CreateBitmap( + iWidth, // int nWidth + iHeight, // int nHeight + 1, // UINT cPlanes + bpp, // UINT cBitsPerPel + image // CONST VOID * lpvBits + ); + + memset( image, 255, size_in_bytes ); + } + + return ( LRESULT )hBitmap; + } + + default: + { + Console::Append( TEXT( "NOT handled" ) ); + } + + } + Console::Append( TEXT( " " ) ); + + break; + } + + case IPC_GET_EMBEDIF: + // TODO + if( !wp ) + { + return ( LRESULT )Embed::Embed; + } + else + { + return ( LRESULT )Embed::Embed( ( embedWindowState * )wp ); + } + break; +/* + case IPC_EMBED_ENUM: break; + case IPC_EMBED_ISVALID: break; + case IPC_CONVERTFILE: break; + case IPC_CONVERTFILE_END: break; + case IPC_CONVERT_CONFIG: break; + case IPC_CONVERT_CONFIG_END: break; + case IPC_GETSADATAFUNC: break; + case IPC_ISMAINWNDVISIBLE: break; + case IPC_SETPLEDITCOLORS: break; + case IPC_SPAWNEQPRESETMENU: break; + case IPC_SPAWNFILEMENU: break; + case IPC_SPAWNOPTIONSMENU: break; + case IPC_SPAWNWINDOWSMENU: break; + case IPC_SPAWNHELPMENU: break; + case IPC_SPAWNPLAYMENU: break; + case IPC_SPAWNPEFILEMENU: break; + case IPC_SPAWNPEPLAYLISTMENU: break; + case IPC_SPAWNPESORTMENU: break; + case IPC_SPAWNPEHELPMENU: break; + case IPC_SPAWNMLFILEMENU: break; + case IPC_SPAWNMLVIEWMENU: break; + case IPC_SPAWNMLHELPMENU: break; + case IPC_SPAWNPELISTOFPLAYLISTS: break; + case WM_WA_SYSTRAY: break; + case IPC_IS_PLAYING_VIDEO: break; + case IPC_GET_IVIDEOOUTPUT: break; +*/ + case IPC_CB_ONSHOWWND: + case IPC_CB_ONHIDEWND: + case IPC_CB_GETTOOLTIP: + case IPC_CB_MISC: + case IPC_CB_CONVERT_STATUS: + case IPC_CB_CONVERT_DONE: + break; +/* + case IPC_ADJUST_FFWINDOWSMENUPOS: break; + case IPC_ISDOUBLESIZE: break; + case IPC_ADJUST_FFOPTIONSMENUPOS: break; +*/ + case IPC_GETTIMEDISPLAYMODE: + return 0; // == elapsed time +/* + case IPC_SETVISWND: break; + case IPC_GETVISWND: break; + case IPC_ISVISRUNNING: break; +*/ + case IPC_CB_VISRANDOM: break; +/* + case IPC_SETIDEALVIDEOSIZE: break; + case IPC_GETSTOPONVIDEOCLOSE: break; + case IPC_SETSTOPONVIDEOCLOSE: break; +*/ + case IPC_TRANSLATEACCELERATOR: + Console::Append( TEXT( "IPC_TRANSLATEACCELERATOR" ) ); + Console::Append( TEXT( "NOT handled" ) ); + Console::Append( TEXT( " " ) ); + break; + + case IPC_CB_ONTOGGLEAOT: break; +/* + case IPC_GETPREFSWND: break; + case IPC_SET_PE_WIDTHHEIGHT: break; + case IPC_GETLANGUAGEPACKINSTANCE: break; +*/ + case IPC_CB_PEINFOTEXT: + case IPC_CB_OUTPUTCHANGED: + break; +/* + case IPC_GETOUTPUTPLUGIN: break; + case IPC_SETDRAWBORDERS: break; + case IPC_DISABLESKINCURSORS: break; +*/ + case IPC_CB_RESETFONT: break; +/* + case IPC_IS_FULLSCREEN: break; + case IPC_SET_VIS_FS_FLAG: break; + case IPC_SHOW_NOTIFICATION: break; + case IPC_GETSKININFO: break; + case IPC_GET_MANUALPLADVANCE: break; + case IPC_SET_MANUALPLADVANCE: break; + case IPC_GET_NEXT_PLITEM: break; + case IPC_GET_PREVIOUS_PLITEM: break; + case IPC_IS_WNDSHADE: break; + case IPC_SETRATING: break; + case IPC_GETRATING: break; + case IPC_GETNUMAUDIOTRACKS: break; + case IPC_GETNUMVIDEOTRACKS: break; + case IPC_GETAUDIOTRACK: break; + case IPC_GETVIDEOTRACK: break; + case IPC_SETAUDIOTRACK: break; + case IPC_SETVIDEOTRACK: break; + case IPC_PUSH_DISABLE_EXIT: break; + case IPC_POP_DISABLE_EXIT: break; + case IPC_IS_EXIT_ENABLED: break; + case IPC_IS_AOT: break; + case IPC_PLCMD: break; + case IPC_MBCMD: break; + case IPC_VIDCMD: break; + case IPC_MBURL: break; + case IPC_MBGETCURURL: break; + case IPC_MBGETDESC: break; + case IPC_MBCHECKLOCFILE: break; + case IPC_MBREFRESH: break; + case IPC_MBGETDEFURL: break; + case IPC_STATS_LIBRARY_ITEMCNT: break; + case IPC_FF_FIRST: break; + case IPC_FF_LAST: break; + case IPC_GETDROPTARGET: break; + case IPC_PLAYLIST_MODIFIED: break; + case IPC_PLAYING_FILE: break; + case IPC_FILE_TAG_MAY_HAVE_UPDATED: break; + case IPC_ALLOW_PLAYTRACKING: break; +*/ + case IPC_HOOK_OKTOQUIT: + return 1; // Okay +/* + case IPC_WRITECONFIG: break; +*/ + case IPC_REGISTER_WINAMP_IPCMESSAGE: + { + // TODO convert to TCHAR???? + UINT res = RegisterWindowMessage( ( LPCTSTR )wp ); + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Message \"%s\" registered as #%i" ), wp, res ); + Console::Append( szBuffer ); + Console::Append( " " ); + + if( !stricmp( ( char * )wp, "GenHotkeysAdd" ) ) + { + IPC_GENHOTKEYS_ADD = res; + } + else if( !stricmp( ( char * )wp, "Dynamic Library" ) ) + { + ID_DYNAMICLIBRARY = res; + } + else if( !stricmp( ( char * )wp, "IPC_GETPLAINBARTARGET" ) ) + { + IPC_GETPLAINBARTARGET = res; + } + + return res; + } + + case 2006: // undocumented, name IPC_CB_HOTKEY or so later and ask to add this to sdk + { + // Hotkey sent as strings! + // Idea: make stl map + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "Hotkey \"%s\" detected" ), wp ); + Console::Append( szBuffer ); + Console::Append( " " ); + + return 1; // Accept??? + } + + default: + { + if( lp == IPC_GENHOTKEYS_ADD ) + { + break; + } + else if( lp == IPC_GETPLAINBARTARGET ) + { + return ( LRESULT )( IsWindowVisible( WindowVis ) ? WindowVis : NULL ); + } + + TCHAR szBuffer[ 5000 ]; + _stprintf( szBuffer, TEXT( "WM_WA_IPC <%i> <%i>" ), wp, lp ); + Console::Append( szBuffer ); + Console::Append( TEXT( "NOT handled" ) ); + Console::Append( " " ); + } + } + break; + + case WM_WA_MPEG_EOF: + Playback::NotifyTrackEnd(); + Playback::Next(); + break; + + case WM_COPYDATA: + { + if( !lp ) return FALSE; + + COPYDATASTRUCT * cds = ( COPYDATASTRUCT * )lp; + switch( cds->dwData ) + { + case IPC_PLAYFILE: + { + const int iLen = cds->cbData; + if( !iLen ) return FALSE; + TCHAR * szKeep = new TCHAR[ iLen + 1 ]; + ToTchar( szKeep, ( char * )cds->lpData, iLen ); + szKeep[ iLen ] = TEXT( '\0' ); + + playlist->PushBack( szKeep ); + return TRUE; + } + } + return FALSE; + } + + } + return DefWindowProc( hwnd, message, wp, lp ); +} diff --git a/Externals/MusicMod/Player/Src/Winamp.h b/Externals/MusicMod/Player/Src/Winamp.h new file mode 100644 index 0000000000..7ef4344c69 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp.h @@ -0,0 +1,26 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#ifndef PA_WINAMP_H +#define PA_WINAMP_H + + + +#include "Global.h" + + + +LRESULT CALLBACK WndprocWinamp( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + + +#endif // PA_WINAMP_H diff --git a/Externals/MusicMod/Player/Src/Winamp/Dsp.h b/Externals/MusicMod/Player/Src/Winamp/Dsp.h new file mode 100644 index 0000000000..0c62886b8a --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp/Dsp.h @@ -0,0 +1,47 @@ +#ifndef WINAMP_DSP_H +#define WINAMP_DSP_H + + +#include + +// DSP plugin interface + +// notes: +// any window that remains in foreground should optimally pass unused +// keystrokes to the parent (winamp's) window, so that the user +// can still control it. As for storing configuration, +// Configuration data should be stored in \plugin.ini +// (look at the vis plugin for configuration code) + +typedef struct winampDSPModule { + char *description; // description + HWND hwndParent; // parent window (filled in by calling app) + HINSTANCE hDllInstance; // instance handle to this DLL (filled in by calling app) + + void (*Config)(struct winampDSPModule *this_mod); // configuration dialog (if needed) + int (*Init)(struct winampDSPModule *this_mod); // 0 on success, creates window, etc (if needed) + + // modify waveform samples: returns number of samples to actually write + // (typically numsamples, but no more than twice numsamples, and no less than half numsamples) + // numsamples should always be at least 128. should, but I'm not sure + int (*ModifySamples)(struct winampDSPModule *this_mod, short int *samples, int numsamples, int bps, int nch, int srate); + + void (*Quit)(struct winampDSPModule *this_mod); // called when unloading + + void *userData; // user data, optional +} winampDSPModule; + +typedef struct { + int version; // DSP_HDRVER + char *description; // description of library + winampDSPModule* (*getModule)(int); // module retrieval function +} winampDSPHeader; + +// exported symbols +typedef winampDSPHeader* (*winampDSPGetHeaderType)(); + +// header version: 0x20 == 0.20 == winamp 2.0 +#define DSP_HDRVER 0x20 + + +#endif // WINAMP_DSP_H diff --git a/Externals/MusicMod/Player/Src/Winamp/Gen.h b/Externals/MusicMod/Player/Src/Winamp/Gen.h new file mode 100644 index 0000000000..7e3c5a2368 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp/Gen.h @@ -0,0 +1,28 @@ +#ifndef WINAMP_GEN_H +#define WINAMP_GEN_H + + + +#define GPPHDR_VER 0x10 + + + +typedef struct { + int version; + char *description; + int (*init)(); + void (*config)(); + void (*quit)(); + HWND hwndParent; + HINSTANCE hDllInstance; +} winampGeneralPurposePlugin; + +typedef winampGeneralPurposePlugin * ( * winampGeneralPurposePluginGetter )(); + + + +// extern winampGeneralPurposePlugin * gen_plugins[ 256 ]; + + + +#endif // WINAMP_GEN_H diff --git a/Externals/MusicMod/Player/Src/Winamp/In2.h b/Externals/MusicMod/Player/Src/Winamp/In2.h new file mode 100644 index 0000000000..5f990eb2f3 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp/In2.h @@ -0,0 +1,119 @@ +#ifndef WINAMP_IN_H +#define WINAMP_IN_H + + +#include +#include "out.h" + +// note: exported symbol is now winampGetInModule2. + +#define IN_VER 0x100 + + +// Changed +// void (*VSASetInfo)(int nch, int srate); +// to +// void (*VSASetInfo)(int srate, int nch); +// since the old one is wrong! +// +// Otherwise in_mp3 is the problem!?... + + +typedef struct +{ + int version; // module type (IN_VER) + char *description; // description of module, with version string + + HWND hMainWindow; // winamp's main window (filled in by winamp) + HINSTANCE hDllInstance; // DLL instance handle (Also filled in by winamp) + + char *FileExtensions; // "mp3\0Layer 3 MPEG\0mp2\0Layer 2 MPEG\0mpg\0Layer 1 MPEG\0" + // May be altered from Config, so the user can select what they want + + int is_seekable; // is this stream seekable? + int UsesOutputPlug; // does this plug-in use the output plug-ins? (musn't ever change, ever :) + + void (*Config)(HWND hwndParent); // configuration dialog + void (*About)(HWND hwndParent); // about dialog + + void (*Init)(); // called at program init + void (*Quit)(); // called at program quit + + void (*GetFileInfo)(char *file, char *title, int *length_in_ms); // if file == NULL, current playing is used + int (*InfoBox)(char *file, HWND hwndParent); + + int (*IsOurFile)(char *fn); // called before extension checks, to allow detection of mms://, etc + // playback stuff + int (*Play)(char *fn); // return zero on success, -1 on file-not-found, some other value on other (stopping winamp) error + void (*Pause)(); // pause stream + void (*UnPause)(); // unpause stream + int (*IsPaused)(); // ispaused? return 1 if paused, 0 if not + void (*Stop)(); // stop (unload) stream + + // time stuff + int (*GetLength)(); // get length in ms + int (*GetOutputTime)(); // returns current output time in ms. (usually returns outMod->GetOutputTime() + void (*SetOutputTime)(int time_in_ms); // seeks to point in stream (in ms). Usually you signal yoru thread to seek, which seeks and calls outMod->Flush().. + + // volume stuff + void (*SetVolume)(int volume); // from 0 to 255.. usually just call outMod->SetVolume + void (*SetPan)(int pan); // from -127 to 127.. usually just call outMod->SetPan + + // in-window builtin vis stuff + + void (*SAVSAInit)(int maxlatency_in_ms, int srate); // call once in Play(). maxlatency_in_ms should be the value returned from outMod->Open() + // call after opening audio device with max latency in ms and samplerate + void (*SAVSADeInit)(); // call in Stop() + + + // simple vis supplying mode + void (*SAAddPCMData)(void *PCMData, int nch, int bps, int timestamp); + // sets the spec data directly from PCM data + // quick and easy way to get vis working :) + // needs at least 576 samples :) + + // advanced vis supplying mode, only use if you're cool. Use SAAddPCMData for most stuff. + int (*SAGetMode)(); // gets csa (the current type (4=ws,2=osc,1=spec)) + // use when calling SAAdd() + void (*SAAdd)(void *data, int timestamp, int csa); // sets the spec data, filled in by winamp + + + // vis stuff (plug-in) + // simple vis supplying mode + void (*VSAAddPCMData)(void *PCMData, int nch, int bps, int timestamp); // sets the vis data directly from PCM data + // quick and easy way to get vis working :) + // needs at least 576 samples :) + + // advanced vis supplying mode, only use if you're cool. Use VSAAddPCMData for most stuff. + int (*VSAGetMode)(int *specNch, int *waveNch); // use to figure out what to give to VSAAdd + void (*VSAAdd)(void *data, int timestamp); // filled in by winamp, called by plug-in + + + // call this in Play() to tell the vis plug-ins the current output params. + void (*VSASetInfo)(int srate, int nch); + + + // dsp plug-in processing: + // (filled in by winamp, called by input plug) + + // returns 1 if active (which means that the number of samples returned by dsp_dosamples + // could be greater than went in.. Use it to estimate if you'll have enough room in the + // output buffer + int (*dsp_isactive)(); + + // returns number of samples to output. This can be as much as twice numsamples. + // be sure to allocate enough buffer for samples, then. + int (*dsp_dosamples)(short int *samples, int numsamples, int bps, int nch, int srate); + + + // eq stuff + void (*EQSet)(int on, char data[10], int preamp); // 0-64 each, 31 is +0, 0 is +12, 63 is -12. Do nothing to ignore. + + // info setting (filled in by winamp) + void (*SetInfo)(int bitrate, int srate, int stereo, int synched); // if -1, changes ignored? :) + + Out_Module *outMod; // filled in by winamp, optionally used :) +} In_Module; + + +#endif // WINAMP_IN_H diff --git a/Externals/MusicMod/Player/Src/Winamp/Out.h b/Externals/MusicMod/Player/Src/Winamp/Out.h new file mode 100644 index 0000000000..7653bcbd96 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp/Out.h @@ -0,0 +1,69 @@ +#ifndef WINAMP_OUT_H +#define WINAMP_OUT_H + + +#include + + +// [ids] +// waveout 32 +// disk 33 +// dsound 38 +// xfade 63 +// gapless 64 +// null 65 +// mm2 69 + +#define OUT_VER 0x10 + +typedef struct +{ + int version; // module version (OUT_VER) + char *description; // description of module, with version string + int id; // module id. each input module gets its own. non-nullsoft modules should + // be >= 65536. + + HWND hMainWindow; // winamp's main window (filled in by winamp) + HINSTANCE hDllInstance; // DLL instance handle (filled in by winamp) + + void (*Config)(HWND hwndParent); // configuration dialog + void (*About)(HWND hwndParent); // about dialog + + void (*Init)(); // called when loaded + void (*Quit)(); // called when unloaded + + int (*Open)(int samplerate, int numchannels, int bitspersamp, int bufferlenms, int prebufferms); + // returns >=0 on success, <0 on failure + // NOTENOTENOTE: bufferlenms and prebufferms are ignored in most if not all output plug-ins. + // ... so don't expect the max latency returned to be what you asked for. + // returns max latency in ms (0 for diskwriters, etc) + // bufferlenms and prebufferms must be in ms. 0 to use defaults. + // prebufferms must be <= bufferlenms + + void (*Close)(); // close the ol' output device. + + int (*Write)(char *buf, int len); + // 0 on success. Len == bytes to write (<= 8192 always). buf is straight audio data. + // 1 returns not able to write (yet). Non-blocking, always. + + int (*CanWrite)(); // returns number of bytes possible to write at a given time. + // Never will decrease unless you call Write (or Close, heh) + + int (*IsPlaying)(); // non0 if output is still going or if data in buffers waiting to be + // written (i.e. closing while IsPlaying() returns 1 would truncate the song + + int (*Pause)(int pause); // returns previous pause state + + void (*SetVolume)(int volume); // volume is 0-255 + void (*SetPan)(int pan); // pan is -128 to 128 + + void (*Flush)(int t); // flushes buffers and restarts output at time t (in ms) + // (used for seeking) + + int (*GetOutputTime)(); // returns played time in MS + int (*GetWrittenTime)(); // returns time written in MS (used for synching up vis stuff) + +} Out_Module; + + +#endif // WINAMP_OUT_H diff --git a/Externals/MusicMod/Player/Src/Winamp/Vis.h b/Externals/MusicMod/Player/Src/Winamp/Vis.h new file mode 100644 index 0000000000..280830d7cd --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp/Vis.h @@ -0,0 +1,57 @@ +#ifndef WINAMP_VIS_H +#define WINAMP_VIS_H + + +// notes: +// any window that remains in foreground should optimally pass +// keystrokes to the parent (winamp's) window, so that the user +// can still control it. unless escape is hit, or some option +// key specific to the vis is hit. As for storing configuration, +// Configuration data should be stored in \plugin.ini +// Look at the example plugin for a framework. + +// ints are 32 bits, and structure members are aligned on the default 8 byte boundaries +// tested with VC++ 4.2 and 5.0 + + +#include + + +typedef struct winampVisModule { + char *description; // description of module + HWND hwndParent; // parent window (filled in by calling app) + HINSTANCE hDllInstance; // instance handle to this DLL (filled in by calling app) + int sRate; // sample rate (filled in by calling app) + int nCh; // number of channels (filled in...) + int latencyMs; // latency from call of RenderFrame to actual drawing + // (calling app looks at this value when getting data) + int delayMs; // delay between calls in ms + + // the data is filled in according to the respective Nch entry + int spectrumNch; + int waveformNch; + unsigned char spectrumData[2][576]; + unsigned char waveformData[2][576]; + + void (*Config)(struct winampVisModule *this_mod); // configuration dialog + int (*Init)(struct winampVisModule *this_mod); // 0 on success, creates window, etc + int (*Render)(struct winampVisModule *this_mod); // returns 0 if successful, 1 if vis should end + void (*Quit)(struct winampVisModule *this_mod); // call when done + + void *userData; // user data, optional +} winampVisModule; + +typedef struct { + int version; // VID_HDRVER + char *description; // description of library + winampVisModule* (*getModule)(int); +} winampVisHeader; + +// exported symbols +typedef winampVisHeader* (*winampVisGetHeaderType)(); + +// version of current module (0x101 == 1.01) +#define VIS_HDRVER 0x101 + + +#endif // WINAMP_VIS_H diff --git a/Externals/MusicMod/Player/Src/Winamp/wa_ipc.h b/Externals/MusicMod/Player/Src/Winamp/wa_ipc.h new file mode 100644 index 0000000000..75689b9eea --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp/wa_ipc.h @@ -0,0 +1,1103 @@ +/* +** Copyright (C) 2003 Nullsoft, Inc. +** +** This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held +** liable for any damages arising from the use of this software. +** +** Permission is granted to anyone to use this software for any purpose, including commercial applications, and to +** alter it and redistribute it freely, subject to the following restrictions: +** +** 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. +** If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. +** +** 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. +** +** 3. This notice may not be removed or altered from any source distribution. +** +*/ + +#ifndef _WA_IPC_H_ +#define _WA_IPC_H_ + +/* +** This is the modern replacement for the classic 'frontend.h'. Most of these +** updates are designed for in-process use, i.e. from a plugin. +** +*/ + +/* message used to sent many messages to winamp's main window. +** most all of the IPC_* messages involve sending the message in the form of: +** result = SendMessage(hwnd_winamp,WM_WA_IPC,(parameter),IPC_*); +*/ +#define WM_WA_IPC WM_USER +/* but some of them use WM_COPYDATA. be afraid. +*/ + +#define IPC_GETVERSION 0 +/* int version = SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETVERSION); +** +** Version will be 0x20yx for winamp 2.yx. versions previous to Winamp 2.0 +** typically (but not always) use 0x1zyx for 1.zx versions. Weird, I know. +*/ + +#define IPC_GETREGISTEREDVERSION 770 + + +typedef struct { + char *filename; + char *title; + int length; +} enqueueFileWithMetaStruct; // send this to a IPC_PLAYFILE in a non WM_COPYDATA, +// and you get the nice desired result. if title is NULL, it is treated as a "thing", +// otherwise it's assumed to be a file (for speed) + +#define IPC_PLAYFILE 100 // dont be fooled, this is really the same as enqueufile +#define IPC_ENQUEUEFILE 100 +/* sent as a WM_COPYDATA, with IPC_PLAYFILE as the dwData, and the string to play +** as the lpData. Just enqueues, does not clear the playlist or change the playback +** state. +*/ + + +#define IPC_DELETE 101 +#define IPC_DELETE_INT 1101 // don't use this, it's used internally by winamp when + // dealing with some lame explorer issues. +/* SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_DELETE); +** Use IPC_DELETE to clear Winamp's internal playlist. +*/ + + +#define IPC_STARTPLAY 102 // starts playback. almost like hitting play in Winamp. +#define IPC_STARTPLAY_INT 1102 // used internally, don't bother using it (won't be any fun) + + +#define IPC_CHDIR 103 +/* sent as a WM_COPYDATA, with IPC_CHDIR as the dwData, and the directory to change to +** as the lpData. +*/ + + +#define IPC_ISPLAYING 104 +/* int res = SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_ISPLAYING); +** If it returns 1, it is playing. if it returns 3, it is paused, +** if it returns 0, it is not playing. +*/ + + +#define IPC_GETOUTPUTTIME 105 +/* int res = SendMessage(hwnd_winamp,WM_WA_IPC,mode,IPC_GETOUTPUTTIME); +** returns the position in milliseconds of the current track (mode = 0), +** or the track length, in seconds (mode = 1). Returns -1 if not playing or error. +*/ + + +#define IPC_JUMPTOTIME 106 +/* (requires Winamp 1.60+) +** SendMessage(hwnd_winamp,WM_WA_IPC,ms,IPC_JUMPTOTIME); +** IPC_JUMPTOTIME sets the position in milliseconds of the +** current song (approximately). +** Returns -1 if not playing, 1 on eof, or 0 if successful +*/ + +#define IPC_GETMODULENAME 109 +#define IPC_EX_ISRIGHTEXE 666 +/* usually shouldnt bother using these, but here goes: +** send a WM_COPYDATA with IPC_GETMODULENAME, and an internal +** flag gets set, which if you send a normal WM_WA_IPC message with +** IPC_EX_ISRIGHTEXE, it returns whether or not that filename +** matches. lame, I know. +*/ + +#define IPC_WRITEPLAYLIST 120 +/* (requires Winamp 1.666+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_WRITEPLAYLIST); +** +** IPC_WRITEPLAYLIST writes the current playlist to \\Winamp.m3u, +** and returns the current playlist position. +** Kinda obsoleted by some of the 2.x new stuff, but still good for when +** using a front-end (instead of a plug-in) +*/ + + +#define IPC_SETPLAYLISTPOS 121 +/* (requires Winamp 2.0+) +** SendMessage(hwnd_winamp,WM_WA_IPC,position,IPC_SETPLAYLISTPOS) +** IPC_SETPLAYLISTPOS sets the playlist position to 'position'. It +** does not change playback or anything, it just sets position, and +** updates the view if necessary +*/ + + +#define IPC_SETVOLUME 122 +/* (requires Winamp 2.0+) +** SendMessage(hwnd_winamp,WM_WA_IPC,volume,IPC_SETVOLUME); +** IPC_SETVOLUME sets the volume of Winamp (from 0-255). +*/ + + +#define IPC_SETPANNING 123 +/* (requires Winamp 2.0+) +** SendMessage(hwnd_winamp,WM_WA_IPC,panning,IPC_SETPANNING); +** IPC_SETPANNING sets the panning of Winamp (from 0 (left) to 255 (right)). +*/ + + +#define IPC_GETLISTLENGTH 124 +/* (requires Winamp 2.0+) +** int length = SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETLISTLENGTH); +** IPC_GETLISTLENGTH returns the length of the current playlist, in +** tracks. +*/ + + +#define IPC_GETLISTPOS 125 +/* (requires Winamp 2.05+) +** int pos=SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETLISTPOS); +** IPC_GETLISTPOS returns the playlist position. A lot like IPC_WRITEPLAYLIST +** only faster since it doesn't have to write out the list. Heh, silly me. +*/ + + +#define IPC_GETINFO 126 +/* (requires Winamp 2.05+) +** int inf=SendMessage(hwnd_winamp,WM_WA_IPC,mode,IPC_GETINFO); +** IPC_GETINFO returns info about the current playing song. The value +** it returns depends on the value of 'mode'. +** Mode Meaning +** ------------------ +** 0 Samplerate (i.e. 44100) +** 1 Bitrate (i.e. 128) +** 2 Channels (i.e. 2) +** 3 (5+) Video LOWORD=w HIWORD=h +** 4 (5+) > 65536, string (video description) +*/ + + +#define IPC_GETEQDATA 127 +/* (requires Winamp 2.05+) +** int data=SendMessage(hwnd_winamp,WM_WA_IPC,pos,IPC_GETEQDATA); +** IPC_GETEQDATA queries the status of the EQ. +** The value returned depends on what 'pos' is set to: +** Value Meaning +** ------------------ +** 0-9 The 10 bands of EQ data. 0-63 (+20db - -20db) +** 10 The preamp value. 0-63 (+20db - -20db) +** 11 Enabled. zero if disabled, nonzero if enabled. +** 12 Autoload. zero if disabled, nonzero if enabled. +*/ + + +#define IPC_SETEQDATA 128 +/* (requires Winamp 2.05+) +** SendMessage(hwnd_winamp,WM_WA_IPC,pos,IPC_GETEQDATA); +** SendMessage(hwnd_winamp,WM_WA_IPC,value,IPC_SETEQDATA); +** IPC_SETEQDATA sets the value of the last position retrieved +** by IPC_GETEQDATA. This is pretty lame, and we should provide +** an extended version that lets you do a MAKELPARAM(pos,value). +** someday... + + new (2.92+): + if the high byte is set to 0xDB, then the third byte specifies + which band, and the bottom word specifies the value. +*/ + +#define IPC_ADDBOOKMARK 129 +/* (requires Winamp 2.4+) +** Sent as a WM_COPYDATA, using IPC_ADDBOOKMARK, adds the specified +** file/url to the Winamp bookmark list. +*/ +/* +In winamp 5+, we use this as a normal WM_WA_IPC and the string: + + "filename\0title\0" + + to notify the library/bookmark editor that a bookmark +was added. Note that using this message in this context does not +actually add the bookmark. +do not use :) +*/ + + +#define IPC_INSTALLPLUGIN 130 +/* not implemented, but if it was you could do a WM_COPYDATA with +** a path to a .wpz, and it would install it. +*/ + + +#define IPC_RESTARTWINAMP 135 +/* (requires Winamp 2.2+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_RESTARTWINAMP); +** IPC_RESTARTWINAMP will restart Winamp (isn't that obvious ? :) +*/ + + +#define IPC_ISFULLSTOP 400 +/* (requires winamp 2.7+ I think) +** ret=SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_ISFULLSTOP); +** useful for when you're an output plugin, and you want to see +** if the stop/close is a full stop, or just between tracks. +** returns nonzero if it's full, zero if it's just a new track. +*/ + + +#define IPC_INETAVAILABLE 242 +/* (requires Winamp 2.05+) +** val=SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_INETAVAILABLE); +** IPC_INETAVAILABLE will return 1 if the Internet connection is available for Winamp. +*/ + + +#define IPC_UPDTITLE 243 +/* (requires Winamp 2.2+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_UPDTITLE); +** IPC_UPDTITLE will ask Winamp to update the informations about the current title. +*/ + + +#define IPC_REFRESHPLCACHE 247 +/* (requires Winamp 2.2+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_REFRESHPLCACHE); +** IPC_REFRESHPLCACHE will flush the playlist cache buffer. +** (send this if you want it to go refetch titles for tracks) +*/ + + +#define IPC_GET_SHUFFLE 250 +/* (requires Winamp 2.4+) +** val=SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GET_SHUFFLE); +** +** IPC_GET_SHUFFLE returns the status of the Shuffle option (1 if set) +*/ + + +#define IPC_GET_REPEAT 251 +/* (requires Winamp 2.4+) +** val=SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GET_REPEAT); +** +** IPC_GET_REPEAT returns the status of the Repeat option (1 if set) +*/ + + +#define IPC_SET_SHUFFLE 252 +/* (requires Winamp 2.4+) +** SendMessage(hwnd_winamp,WM_WA_IPC,value,IPC_SET_SHUFFLE); +** +** IPC_SET_SHUFFLE sets the status of the Shuffle option (1 to turn it on) +*/ + + +#define IPC_SET_REPEAT 253 +/* (requires Winamp 2.4+) +** SendMessage(hwnd_winamp,WM_WA_IPC,value,IPC_SET_REPEAT); +** +** IPC_SET_REPEAT sets the status of the Repeat option (1 to turn it on) +*/ + + +#define IPC_ENABLEDISABLE_ALL_WINDOWS 259 // 0xdeadbeef to disable +/* (requires Winamp 2.9+) +** SendMessage(hwnd_winamp,WM_WA_IPC,enable?0:0xdeadbeef,IPC_MBOPENREAL); +** sending with 0xdeadbeef as the param disables all winamp windows, +** any other values will enable all winamp windows. +*/ + + +#define IPC_GETWND 260 +/* (requires Winamp 2.9+) +** HWND h=SendMessage(hwnd_winamp,WM_WA_IPC,IPC_GETWND_xxx,IPC_GETWND); +** returns the HWND of the window specified. +*/ + #define IPC_GETWND_EQ 0 // use one of these for the param + #define IPC_GETWND_PE 1 + #define IPC_GETWND_MB 2 + #define IPC_GETWND_VIDEO 3 +#define IPC_ISWNDVISIBLE 261 // same param as IPC_GETWND + + + + +/************************************************************************ +***************** in-process only (WE LOVE PLUGINS) +************************************************************************/ + + +#define IPC_SETSKIN 200 +/* (requires Winamp 2.04+, only usable from plug-ins (not external apps)) +** SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)"skinname",IPC_SETSKIN); +** IPC_SETSKIN sets the current skin to "skinname". Note that skinname +** can be the name of a skin, a skin .zip file, with or without path. +** If path isn't specified, the default search path is the winamp skins +** directory. +*/ + + +#define IPC_GETSKIN 201 +/* (requires Winamp 2.04+, only usable from plug-ins (not external apps)) +** SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)skinname_buffer,IPC_GETSKIN); +** IPC_GETSKIN puts the directory where skin bitmaps can be found +** into skinname_buffer. +** skinname_buffer must be MAX_PATH characters in length. +** When using a .zip'd skin file, it'll return a temporary directory +** where the ZIP was decompressed. +*/ + + +#define IPC_EXECPLUG 202 +/* (requires Winamp 2.04+, only usable from plug-ins (not external apps)) +** SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)"vis_file.dll",IPC_EXECPLUG); +** IPC_EXECPLUG executes a visualization plug-in pointed to by WPARAM. +** the format of this string can be: +** "vis_whatever.dll" +** "vis_whatever.dll,0" // (first mod, file in winamp plug-in dir) +** "C:\\dir\\vis_whatever.dll,1" +*/ + + +#define IPC_GETPLAYLISTFILE 211 +/* (requires Winamp 2.04+, only usable from plug-ins (not external apps)) +** char *name=SendMessage(hwnd_winamp,WM_WA_IPC,index,IPC_GETPLAYLISTFILE); +** IPC_GETPLAYLISTFILE gets the filename of the playlist entry [index]. +** returns a pointer to it. returns NULL on error. +*/ + + +#define IPC_GETPLAYLISTTITLE 212 +/* (requires Winamp 2.04+, only usable from plug-ins (not external apps)) +** char *name=SendMessage(hwnd_winamp,WM_WA_IPC,index,IPC_GETPLAYLISTTITLE); +** +** IPC_GETPLAYLISTTITLE gets the title of the playlist entry [index]. +** returns a pointer to it. returns NULL on error. +*/ + + +#define IPC_GETHTTPGETTER 240 +/* retrieves a function pointer to a HTTP retrieval function. +** if this is unsupported, returns 1 or 0. +** the function should be: +** int (*httpRetrieveFile)(HWND hwnd, char *url, char *file, char *dlgtitle); +** if you call this function, with a parent window, a URL, an output file, and a dialog title, +** it will return 0 on successful download, 1 on error. +*/ + + +#define IPC_MBOPEN 241 +/* (requires Winamp 2.05+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_MBOPEN); +** SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)url,IPC_MBOPEN); +** IPC_MBOPEN will open a new URL in the minibrowser. if url is NULL, it will open the Minibrowser window. +*/ + + + +#define IPC_CHANGECURRENTFILE 245 +/* (requires Winamp 2.05+) +** SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)file,IPC_CHANGECURRENTFILE); +** IPC_CHANGECURRENTFILE will set the current playlist item. +*/ + + +#define IPC_GETMBURL 246 +/* (requires Winamp 2.2+) +** char buffer[4096]; // Urls can be VERY long +** SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)buffer,IPC_GETMBURL); +** IPC_GETMBURL will retrieve the current Minibrowser URL into buffer. +** buffer must be at least 4096 bytes long. +*/ + + +#define IPC_MBBLOCK 248 +/* (requires Winamp 2.4+) +** SendMessage(hwnd_winamp,WM_WA_IPC,value,IPC_MBBLOCK); +** +** IPC_MBBLOCK will block the Minibrowser from updates if value is set to 1 +*/ + +#define IPC_MBOPENREAL 249 +/* (requires Winamp 2.4+) +** SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)url,IPC_MBOPENREAL); +** +** IPC_MBOPENREAL works the same as IPC_MBOPEN except that it will works even if +** IPC_MBBLOCK has been set to 1 +*/ + +#define IPC_ADJUST_OPTIONSMENUPOS 280 +/* (requires Winamp 2.9+) +** int newpos=SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)adjust_offset,IPC_ADJUST_OPTIONSMENUPOS); +** moves where winamp expects the Options menu in the main menu. Useful if you wish to insert a +** menu item above the options/skins/vis menus. +*/ + +#define IPC_GET_HMENU 281 +/* (requires Winamp 2.9+) +** HMENU hMenu=SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)0,IPC_GET_HMENU); +** values for data: +** 0 : main popup menu +** 1 : main menubar file menu +** 2 : main menubar options menu +** 3 : main menubar windows menu +** 4 : main menubar help menu +** other values will return NULL. +*/ + +#define IPC_GET_EXTENDED_FILE_INFO 290 //pass a pointer to the following struct in wParam +#define IPC_GET_EXTENDED_FILE_INFO_HOOKABLE 296 +/* (requires Winamp 2.9+) +** to use, create an extendedFileInfoStruct, point the values filename and metadata to the +** filename and metadata field you wish to query, and ret to a buffer, with retlen to the +** length of that buffer, and then SendMessage(hwnd_winamp,WM_WA_IPC,&struct,IPC_GET_EXTENDED_FILE_INFO); +** the results should be in the buffer pointed to by ret. +** returns 1 if the decoder supports a getExtendedFileInfo method +*/ +typedef struct { + char *filename; + char *metadata; + char *ret; + int retlen; +} extendedFileInfoStruct; + +#define IPC_GET_BASIC_FILE_INFO 291 //pass a pointer to the following struct in wParam +typedef struct { + char *filename; + + int quickCheck; // set to 0 to always get, 1 for quick, 2 for default (if 2, quickCheck will be set to 0 if quick wasnot used) + + // filled in by winamp + int length; + char *title; + int titlelen; +} basicFileInfoStruct; + +#define IPC_GET_EXTLIST 292 //returns doublenull delimited. GlobalFree() it when done. if data is 0, returns raw extlist, if 1, returns something suitable for getopenfilename + +#define IPC_INFOBOX 293 +typedef struct { + HWND parent; + char *filename; +} infoBoxParam; + +#define IPC_SET_EXTENDED_FILE_INFO 294 //pass a pointer to the a extendedFileInfoStruct in wParam +/* (requires Winamp 2.9+) +** to use, create an extendedFileInfoStruct, point the values filename and metadata to the +** filename and metadata field you wish to write in ret. (retlen is not used). and then +** SendMessage(hwnd_winamp,WM_WA_IPC,&struct,IPC_SET_EXTENDED_FILE_INFO); +** returns 1 if the metadata is supported +** Call IPC_WRITE_EXTENDED_FILE_INFO once you're done setting all the metadata you want to update +*/ + +#define IPC_WRITE_EXTENDED_FILE_INFO 295 +/* (requires Winamp 2.9+) +** writes all the metadata set thru IPC_SET_EXTENDED_FILE_INFO to the file +** returns 1 if the file has been successfully updated, 0 if error +*/ + +#define IPC_FORMAT_TITLE 297 +typedef struct +{ + char *spec; // NULL=default winamp spec + void *p; + + char *out; + int out_len; + + char * (*TAGFUNC)(char * tag, void * p); //return 0 if not found + void (*TAGFREEFUNC)(char * tag,void * p); +} waFormatTitle; + +#define IPC_GETUNCOMPRESSINTERFACE 331 +/* returns a function pointer to uncompress(). +** int (*uncompress)(unsigned char *dest, unsigned long *destLen, const unsigned char *source, unsigned long sourceLen); +** right out of zlib, useful for decompressing zlibbed data. +** if you pass the parm of 0x10100000, it will return a wa_inflate_struct * to an inflate API. +*/ + +typedef struct { + int (*inflateReset)(void *strm); + int (*inflateInit_)(void *strm,const char *version, int stream_size); + int (*inflate)(void *strm, int flush); + int (*inflateEnd)(void *strm); + unsigned long (*crc32)(unsigned long crc, const unsigned char *buf, unsigned int len); +} wa_inflate_struct; + + +#define IPC_ADD_PREFS_DLG 332 +#define IPC_REMOVE_PREFS_DLG 333 +/* (requires Winamp 2.9+) +** to use, allocate a prefsDlgRec structure (either on the heap or some global +** data, but NOT on the stack), initialze the members: +** hInst to the DLL instance where the resource is located +** dlgID to the ID of the dialog, +** proc to the window procedure for the dialog +** name to the name of the prefs page in the prefs. +** where to 0 (eventually we may add more options) +** then, SendMessage(hwnd_winamp,WM_WA_IPC,&prefsRec,IPC_ADD_PREFS_DLG); +** +** you can also IPC_REMOVE_PREFS_DLG with the address of the same prefsRec, +** but you shouldn't really ever have to. +** +*/ +#define IPC_OPENPREFSTOPAGE 380 // pass an id of a builtin page, or a &prefsDlgRec of prefs page to open + +typedef struct _prefsDlgRec { + HINSTANCE hInst; + int dlgID; + void *proc; + + char *name; + int where; // 0 for options, 1 for plugins, 2 for skins, 3 for bookmarks, 4 for prefs + + + int _id; + struct _prefsDlgRec *next; +} prefsDlgRec; + + +#define IPC_GETINIFILE 334 // returns a pointer to winamp.ini +#define IPC_GETINIDIRECTORY 335 // returns a pointer to the directory to put config files in (if you dont want to use winamp.ini) + +#define IPC_SPAWNBUTTONPOPUP 361 // param = +// 0 = eject +// 1 = previous +// 2 = next +// 3 = pause +// 4 = play +// 5 = stop + +#define IPC_OPENURLBOX 360 // pass a HWND to a parent, returns a HGLOBAL that needs to be freed with GlobalFree(), if successful +#define IPC_OPENFILEBOX 362 // pass a HWND to a parent +#define IPC_OPENDIRBOX 363 // pass a HWND to a parent + +// pass an HWND to a parent. call this if you take over the whole UI so that the dialogs are not appearing on the +// bottom right of the screen since the main winamp window is at 3000x3000, call again with NULL to reset +#define IPC_SETDIALOGBOXPARENT 364 + + + +// pass 0 for a copy of the skin HBITMAP +// pass 1 for name of font to use for playlist editor likeness +// pass 2 for font charset +// pass 3 for font size +#define IPC_GET_GENSKINBITMAP 503 + + +#define IPC_GET_EMBEDIF 505 // pass an embedWindowState +// returns an HWND embedWindow(embedWindowState *); if the data is NULL, otherwise returns the HWND directly +typedef struct +{ + HWND me; //hwnd of the window + + int flags; + + RECT r; + + void *user_ptr; // for application use + + int extra_data[64]; // for internal winamp use +} embedWindowState; + +#define EMBED_FLAGS_NORESIZE 1 // set this bit in embedWindowState.flags to keep window from being resizable +#define EMBED_FLAGS_NOTRANSPARENCY 2 // set this bit in embedWindowState.flags to make gen_ff turn transparency off for this wnd + + +#define IPC_EMBED_ENUM 532 +typedef struct embedEnumStruct +{ + int (*enumProc)(embedWindowState *ws, struct embedEnumStruct *param); // return 1 to abort + int user_data; // or more :) +} embedEnumStruct; + // pass + +#define IPC_EMBED_ISVALID 533 + +#define IPC_CONVERTFILE 506 +/* (requires Winamp 2.92+) +** Converts a given file to a different format (PCM, MP3, etc...) +** To use, pass a pointer to a waFileConvertStruct struct +** This struct can be either on the heap or some global +** data, but NOT on the stack. At least, until the conversion is done. +** +** eg: SendMessage(hwnd_winamp,WM_WA_IPC,&myConvertStruct,IPC_CONVERTFILE); +** +** Return value: +** 0: Can't start the conversion. Look at myConvertStruct->error for details. +** 1: Conversion started. Status messages will be sent to the specified callbackhwnd. +** Be sure to call IPC_CONVERTFILE_END when your callback window receives the +** IPC_CB_CONVERT_DONE message. +*/ +typedef struct +{ + char *sourcefile; // "c:\\source.mp3" + char *destfile; // "c:\\dest.pcm" + int destformat[8]; // like 'PCM ',srate,nch,bps + HWND callbackhwnd; // window that will receive the IPC_CB_CONVERT notification messages + + //filled in by winamp.exe + char *error; //if IPC_CONVERTFILE returns 0, the reason will be here + + int bytes_done; //you can look at both of these values for speed statistics + int bytes_total; + int bytes_out; + + int killswitch; // don't set it manually, use IPC_CONVERTFILE_END + int extra_data[64]; // for internal winamp use +} convertFileStruct; + +#define IPC_CONVERTFILE_END 507 +/* (requires Winamp 2.92+) +** Stop/ends a convert process started from IPC_CONVERTFILE +** You need to call this when you receive the IPC_CB_CONVERTDONE message or when you +** want to abort a conversion process +** +** eg: SendMessage(hwnd_winamp,WM_WA_IPC,&myConvertStruct,IPC_CONVERTFILE_END); +** +** No return value +*/ + +typedef struct { + HWND hwndParent; + int format; + + //filled in by winamp.exe + HWND hwndConfig; + int extra_data[8]; +} convertConfigStruct; +#define IPC_CONVERT_CONFIG 508 +#define IPC_CONVERT_CONFIG_END 509 + +typedef struct +{ + void (*enumProc)(int user_data, const char *desc, int fourcc); + int user_data; +} converterEnumFmtStruct; +#define IPC_CONVERT_CONFIG_ENUMFMTS 510 +/* (requires Winamp 2.92+) +*/ + + +typedef struct +{ + char cdletter; + char *playlist_file; + HWND callback_hwnd; + + //filled in by winamp.exe + char *error; +} burnCDStruct; +#define IPC_BURN_CD 511 +/* (requires Winamp 5.0+) +*/ + +typedef struct +{ + convertFileStruct *cfs; + int priority; +} convertSetPriority; +#define IPC_CONVERT_SET_PRIORITY 512 + +typedef struct +{ + char *filename; + char *title; // 2048 bytes + int length; + int force_useformatting; // can set this to 1 if you want to force a url to use title formatting shit +} waHookTitleStruct; +// return TRUE if you hook this +#define IPC_HOOK_TITLES 850 + +#define IPC_GETSADATAFUNC 800 +// 0: returns a char *export_sa_get() that returns 150 bytes of data +// 1: returns a export_sa_setreq(int want); + +#define IPC_ISMAINWNDVISIBLE 900 + + +#define IPC_SETPLEDITCOLORS 920 +typedef struct +{ + int numElems; + int *elems; + HBITMAP bm; // set if you want to override +} waSetPlColorsStruct; + + +// the following IPC use waSpawnMenuParms as parameter +#define IPC_SPAWNEQPRESETMENU 933 +#define IPC_SPAWNFILEMENU 934 //menubar +#define IPC_SPAWNOPTIONSMENU 935 //menubar +#define IPC_SPAWNWINDOWSMENU 936 //menubar +#define IPC_SPAWNHELPMENU 937 //menubar +#define IPC_SPAWNPLAYMENU 938 //menubar +#define IPC_SPAWNPEFILEMENU 939 //menubar +#define IPC_SPAWNPEPLAYLISTMENU 940 //menubar +#define IPC_SPAWNPESORTMENU 941 //menubar +#define IPC_SPAWNPEHELPMENU 942 //menubar +#define IPC_SPAWNMLFILEMENU 943 //menubar +#define IPC_SPAWNMLVIEWMENU 944 //menubar +#define IPC_SPAWNMLHELPMENU 945 //menubar +#define IPC_SPAWNPELISTOFPLAYLISTS 946 + +typedef struct +{ + HWND wnd; + int xpos; // in screen coordinates + int ypos; +} waSpawnMenuParms; + +// waSpawnMenuParms2 is used by the menubar submenus +typedef struct +{ + HWND wnd; + int xpos; // in screen coordinates + int ypos; + int width; + int height; +} waSpawnMenuParms2; + + +// system tray sends this (you might want to simulate it) +#define WM_WA_SYSTRAY WM_USER+1 + +// input plugins send this when they are done playing back +#define WM_WA_MPEG_EOF WM_USER+2 + + + +//// video stuff + +#define IPC_IS_PLAYING_VIDEO 501 // returns >1 if playing, 0 if not, 1 if old version (so who knows):) +#define IPC_GET_IVIDEOOUTPUT 500 // see below for IVideoOutput interface +#define VIDEO_MAKETYPE(A,B,C,D) ((A) | ((B)<<8) | ((C)<<16) | ((D)<<24)) +#define VIDUSER_SET_INFOSTRING 0x1000 +#define VIDUSER_GET_VIDEOHWND 0x1001 +#define VIDUSER_SET_VFLIP 0x1002 +#define VIDUSER_SET_TRACKSELINTERFACE 0x1003 // give your ITrackSelector interface as param2 + +#ifndef NO_IVIDEO_DECLARE +#ifdef __cplusplus + +class VideoOutput; +class SubsItem; + +typedef struct { + unsigned char* baseAddr; + long rowBytes; +} YV12_PLANE; + +typedef struct { + YV12_PLANE y; + YV12_PLANE u; + YV12_PLANE v; +} YV12_PLANES; + +class IVideoOutput +{ + public: + virtual ~IVideoOutput() { } + virtual int open(int w, int h, int vflip, double aspectratio, unsigned int fmt)=0; + virtual void setcallback(LRESULT (*msgcallback)(void *token, HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam), void *token) { } + virtual void close()=0; + virtual void draw(void *frame)=0; + virtual void drawSubtitle(SubsItem *item) { } + virtual void showStatusMsg(const char *text) { } + virtual int get_latency() { return 0; } + virtual void notifyBufferState(int bufferstate) { } /* 0-255*/ + + virtual int extended(int param1, int param2, int param3) { return 0; } // Dispatchable, eat this! +}; + +class ITrackSelector +{ + public: + virtual int getNumAudioTracks()=0; + virtual void enumAudioTrackName(int n, const char *buf, int size)=0; + virtual int getCurAudioTrack()=0; + virtual int getNumVideoTracks()=0; + virtual void enumVideoTrackName(int n, const char *buf, int size)=0; + virtual int getCurVideoTrack()=0; + + virtual void setAudioTrack(int n)=0; + virtual void setVideoTrack(int n)=0; +}; + +#endif //cplusplus +#endif//NO_IVIDEO_DECLARE + +// these messages are callbacks that you can grab by subclassing the winamp window + +// wParam = +#define IPC_CB_WND_EQ 0 // use one of these for the param +#define IPC_CB_WND_PE 1 +#define IPC_CB_WND_MB 2 +#define IPC_CB_WND_VIDEO 3 +#define IPC_CB_WND_MAIN 4 + +#define IPC_CB_ONSHOWWND 600 +#define IPC_CB_ONHIDEWND 601 + +#define IPC_CB_GETTOOLTIP 602 + +#define IPC_CB_MISC 603 + #define IPC_CB_MISC_TITLE 0 + #define IPC_CB_MISC_VOLUME 1 // volume/pan + #define IPC_CB_MISC_STATUS 2 + #define IPC_CB_MISC_EQ 3 + #define IPC_CB_MISC_INFO 4 + #define IPC_CB_MISC_VIDEOINFO 5 + +#define IPC_CB_CONVERT_STATUS 604 // param value goes from 0 to 100 (percent) +#define IPC_CB_CONVERT_DONE 605 + +#define IPC_ADJUST_FFWINDOWSMENUPOS 606 +/* (requires Winamp 2.9+) +** int newpos=SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)adjust_offset,IPC_ADJUST_FFWINDOWSMENUPOS); +** moves where winamp expects the freeform windows in the menubar windows main menu. Useful if you wish to insert a +** menu item above extra freeform windows. +*/ + +#define IPC_ISDOUBLESIZE 608 + +#define IPC_ADJUST_FFOPTIONSMENUPOS 609 +/* (requires Winamp 2.9+) +** int newpos=SendMessage(hwnd_winamp,WM_WA_IPC,(WPARAM)adjust_offset,IPC_ADJUST_FFOPTIONSMENUPOS); +** moves where winamp expects the freeform preferences item in the menubar windows main menu. Useful if you wish to insert a +** menu item above preferences item. +*/ + +#define IPC_GETTIMEDISPLAYMODE 610 // returns 0 if displaying elapsed time or 1 if displaying remaining time + +#define IPC_SETVISWND 611 // param is hwnd, setting this allows you to receive ID_VIS_NEXT/PREVOUS/RANDOM/FS wm_commands +#define ID_VIS_NEXT 40382 +#define ID_VIS_PREV 40383 +#define ID_VIS_RANDOM 40384 +#define ID_VIS_FS 40389 +#define ID_VIS_CFG 40390 +#define ID_VIS_MENU 40391 + +#define IPC_GETVISWND 612 // returns the vis cmd handler hwnd +#define IPC_ISVISRUNNING 613 +#define IPC_CB_VISRANDOM 628 // param is status of random + +#define IPC_SETIDEALVIDEOSIZE 614 // sent by winamp to winamp, trap it if you need it. width=HIWORD(param), height=LOWORD(param) + +#define IPC_GETSTOPONVIDEOCLOSE 615 +#define IPC_SETSTOPONVIDEOCLOSE 616 + +typedef struct { + HWND hwnd; + int uMsg; + int wParam; + int lParam; +} transAccelStruct; + +#define IPC_TRANSLATEACCELERATOR 617 + +typedef struct { + int cmd; + int x; + int y; + int align; +} windowCommand; // send this as param to an IPC_PLCMD, IPC_MBCMD, IPC_VIDCMD + +#define IPC_CB_ONTOGGLEAOT 618 + +#define IPC_GETPREFSWND 619 + +#define IPC_SET_PE_WIDTHHEIGHT 620 // data is a pointer to a POINT structure that holds width & height + +#define IPC_GETLANGUAGEPACKINSTANCE 621 + +#define IPC_CB_PEINFOTEXT 622 // data is a string, ie: "04:21/45:02" + +#define IPC_CB_OUTPUTCHANGED 623 // output plugin was changed in config + +#define IPC_GETOUTPUTPLUGIN 625 + +#define IPC_SETDRAWBORDERS 626 +#define IPC_DISABLESKINCURSORS 627 +#define IPC_CB_RESETFONT 629 + +#define IPC_IS_FULLSCREEN 630 // returns 1 if video or vis is in fullscreen mode +#define IPC_SET_VIS_FS_FLAG 631 // a vis should send this message with 1/as param to notify winamp that it has gone to or has come back from fullscreen mode + +#define IPC_SHOW_NOTIFICATION 632 + +#define IPC_GETSKININFO 633 + +#define IPC_GET_MANUALPLADVANCE 634 +/* (requires Winamp 5.03+) +** val=SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GET_MANUALPLADVANCE); +** +** IPC_GET_MANUALPLADVANCE returns the status of the Manual Playlist Advance (1 if set) +*/ + +#define IPC_SET_MANUALPLADVANCE 635 +/* (requires Winamp 5.03+) +** SendMessage(hwnd_winamp,WM_WA_IPC,value,IPC_SET_MANUALPLADVANCE); +** +** IPC_SET_MANUALPLADVANCE sets the status of the Manual Playlist Advance option (1 to turn it on) +*/ + +#define IPC_GET_NEXT_PLITEM 636 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_EOF_GET_NEXT_PLITEM); +** +** Sent to Winamp's main window when an item has just finished playback or the next button has been pressed and +** requesting the new playlist item number to go to. +** Mainly used by gen_jumpex. Subclass this message in your application to return the new item number. +** -1 for normal winamp operation (default) or the new item number in the playlist to play. +*/ + +#define IPC_GET_PREVIOUS_PLITEM 637 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_EOF_GET_PREVIOUS_PLITEM); +** +** Sent to Winamp's main window when the previous button has been pressed and Winamp is requesting the new playlist item number to go to. +** Mainly used by gen_jumpex. Subclass this message in your application to return the new item number. +** -1 for normal winamp operation (default) or the new item number in the playlist to play. +*/ + +#define IPC_IS_WNDSHADE 638 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,wnd,IPC_IS_WNDSHADE); +** +** 'wnd' is window id as defined for IPC_GETWND, or -1 for main window +** Returns 1 if wnd is set to winshade mode, or 0 if it is not +*/ + +#define IPC_SETRATING 639 +/* (requires Winamp 5.04+ with ML) +** SendMessage(hwnd_winamp,WM_WA_IPC,rating,IPC_SETRATING); +** 'rating' is an int value from 0 (no rating) to 5 +*/ + +#define IPC_GETRATING 640 +/* (requires Winamp 5.04+ with ML) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETRATING); +** returns the current item's rating +*/ + +#define IPC_GETNUMAUDIOTRACKS 641 +/* (requires Winamp 5.04+) +** int n = SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETNUMAUDIOTRACKS); +** returns the number of audio tracks for the currently playing item +*/ + +#define IPC_GETNUMVIDEOTRACKS 642 +/* (requires Winamp 5.04+) +** int n = SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETNUMVIDEOTRACKS); +** returns the number of video tracks for the currently playing item +*/ + +#define IPC_GETAUDIOTRACK 643 +/* (requires Winamp 5.04+) +** int cur = SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETAUDIOTRACK); +** returns the id of the current audio track for the currently playing item +*/ + +#define IPC_GETVIDEOTRACK 644 +/* (requires Winamp 5.04+) +** int cur = SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_GETVIDEOTRACK); +** returns the id of the current video track for the currently playing item +*/ + +#define IPC_SETAUDIOTRACK 645 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,track,IPC_SETAUDIOTRACK); +** switch the currently playing item to a new audio track +*/ + +#define IPC_SETVIDEOTRACK 646 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,track,IPC_SETVIDEOTRACK); +** switch the currently playing item to a new video track +*/ + +#define IPC_PUSH_DISABLE_EXIT 647 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_PUSH_DISABLE_EXIT ); +** lets you disable or re-enable the UI exit functions (close button, +** context menu, alt-f4). +** call IPC_POP_DISABLE_EXIT when you are done doing whatever required +** preventing exit +*/ + +#define IPC_POP_DISABLE_EXIT 648 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_POP_DISABLE_EXIT ); +** see IPC_PUSH_DISABLE_EXIT +*/ + +#define IPC_IS_EXIT_ENABLED 649 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_IS_EXIT_ENABLED); +** returns 0 if exit is disabled, 1 otherwise +*/ + +#define IPC_IS_AOT 650 +/* (requires Winamp 5.04+) +** SendMessage(hwnd_winamp,WM_WA_IPC,0,IPC_IS_AOT); +** returns status of always on top flag. note: this may not match the actual +** TOPMOST window flag while another fullscreen application is focused +*/ + + +// >>>>>>>>>>> Next is 651 + +#define IPC_PLCMD 1000 + +#define PLCMD_ADD 0 +#define PLCMD_REM 1 +#define PLCMD_SEL 2 +#define PLCMD_MISC 3 +#define PLCMD_LIST 4 + +#define IPC_MBCMD 1001 + +#define MBCMD_BACK 0 +#define MBCMD_FORWARD 1 +#define MBCMD_STOP 2 +#define MBCMD_RELOAD 3 +#define MBCMD_MISC 4 + +#define IPC_VIDCMD 1002 + +#define VIDCMD_FULLSCREEN 0 +#define VIDCMD_1X 1 +#define VIDCMD_2X 2 +#define VIDCMD_LIB 3 +#define VIDPOPUP_MISC 4 + +#define IPC_MBURL 1003 //sets the URL +#define IPC_MBGETCURURL 1004 //copies the current URL into wParam (have a 4096 buffer ready) +#define IPC_MBGETDESC 1005 //copies the current URL description into wParam (have a 4096 buffer ready) +#define IPC_MBCHECKLOCFILE 1006 //checks that the link file is up to date (otherwise updates it). wParam=parent HWND +#define IPC_MBREFRESH 1007 //refreshes the "now playing" view in the library +#define IPC_MBGETDEFURL 1008 //copies the default URL into wParam (have a 4096 buffer ready) + +#define IPC_STATS_LIBRARY_ITEMCNT 1300 // updates library count status + +// IPC 2000-3000 reserved for freeform messages, see gen_ff/ff_ipc.h +#define IPC_FF_FIRST 2000 +#define IPC_FF_LAST 3000 + +#define IPC_GETDROPTARGET 3001 + +#define IPC_PLAYLIST_MODIFIED 3002 // sent to main wnd whenever the playlist is modified + +#define IPC_PLAYING_FILE 3003 // sent to main wnd with the file as parm whenever a file is played +#define IPC_FILE_TAG_MAY_HAVE_UPDATED 3004 // sent to main wnd with the file as parm whenever a file tag might be updated + + +#define IPC_ALLOW_PLAYTRACKING 3007 +// send nonzero to allow, zero to disallow + +#define IPC_HOOK_OKTOQUIT 3010 // return 0 to abort a quit, nonzero if quit is OK + +#define IPC_WRITECONFIG 3011 // pass 2 to write all, 1 to write playlist + common, 0 to write common+less common + +// pass a string to be the name to register, and returns a value > 65536, which is a unique value you can use +// for custom WM_WA_IPC messages. +#define IPC_REGISTER_WINAMP_IPCMESSAGE 65536 + + + +#endif //_WA_IPC_H_ diff --git a/Externals/MusicMod/Player/Src/Winamp/wa_msgids.h b/Externals/MusicMod/Player/Src/Winamp/wa_msgids.h new file mode 100644 index 0000000000..ae5ce4fe7b --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winamp/wa_msgids.h @@ -0,0 +1,297 @@ +/* +** wa_msgids.h (created 14/04/2004 12:23:19 PM) +** Created from wa_ipc.h and resource.h from the language sdk +** by Darren Owen aka DrO +** +** This a simple header file which defines the message ids to allow you to control +** Winamp in keeping with the old frontend.h (R.I.P.) +** +** +** Version History: +** +** v1.0 :: intial version with ids for Winamp 5.02+ +** v1.0a :: fixed the file to work on compile +** v1.1 :: added the msg id for 'Manual Playlist Advance' +** v1.2 :: added in song rating menu items +** +** +** How to use: +** +** To send these, use the format: +** +** SendMessage(hwnd_winamp, WM_COMMAND,command_name,0); +** +** For other languages such as Visual Basic, Pascal, etc you will need to use +** the equivalent calling SendMessage(..) calling convention +** +** +** Notes: +** +** IDs 42000 to 45000 are reserved for gen_ff +** IDs from 45000 to 57000 are reserved for library +** +*/ + +#ifndef _WA_MSGIDS_H_ +#define _WA_MSGIDS_H_ + +#define WINAMP_FILE_QUIT 40001 +#define WINAMP_OPTIONS_PREFS 40012 // pops up the preferences +#define WINAMP_OPTIONS_AOT 40019 // toggles always on top +#define WINAMP_FILE_REPEAT 40022 +#define WINAMP_FILE_SHUFFLE 40023 +#define WINAMP_HIGH_PRIORITY 40025 +#define WINAMP_FILE_PLAY 40029 // pops up the load file(s) box +#define WINAMP_OPTIONS_EQ 40036 // toggles the EQ window +#define WINAMP_OPTIONS_ELAPSED 40037 +#define WINAMP_OPTIONS_REMAINING 40038 +#define WINAMP_OPTIONS_PLEDIT 40040 // toggles the playlist window +#define WINAMP_HELP_ABOUT 40041 // pops up the about box :) +#define WINAMP_MAINMENU 40043 + + +/* the following are the five main control buttons, with optionally shift +** or control pressed +** (for the exact functions of each, just try it out) +*/ +#define WINAMP_BUTTON1 40044 +#define WINAMP_BUTTON2 40045 +#define WINAMP_BUTTON3 40046 +#define WINAMP_BUTTON4 40047 +#define WINAMP_BUTTON5 40048 +#define WINAMP_BUTTON1_SHIFT 40144 +#define WINAMP_BUTTON2_SHIFT 40145 +#define WINAMP_BUTTON3_SHIFT 40146 +#define WINAMP_BUTTON4_SHIFT 40147 +#define WINAMP_BUTTON5_SHIFT 40148 +#define WINAMP_BUTTON1_CTRL 40154 +#define WINAMP_BUTTON2_CTRL 40155 +#define WINAMP_BUTTON3_CTRL 40156 +#define WINAMP_BUTTON4_CTRL 40157 +#define WINAMP_BUTTON5_CTRL 40158 + +#define WINAMP_VOLUMEUP 40058 // turns the volume up a little +#define WINAMP_VOLUMEDOWN 40059 // turns the volume down a little +#define WINAMP_FFWD5S 40060 // fast forwards 5 seconds +#define WINAMP_REW5S 40061 // rewinds 5 seconds +#define WINAMP_NEXT_WINDOW 40063 +#define WINAMP_OPTIONS_WINDOWSHADE 40064 +#define WINAMP_OPTIONS_DSIZE 40165 +#define IDC_SORT_FILENAME 40166 +#define IDC_SORT_FILETITLE 40167 +#define IDC_SORT_ENTIREFILENAME 40168 +#define IDC_SELECTALL 40169 +#define IDC_SELECTNONE 40170 +#define IDC_SELECTINV 40171 +#define IDM_EQ_LOADPRE 40172 +#define IDM_EQ_LOADMP3 40173 +#define IDM_EQ_LOADDEFAULT 40174 +#define IDM_EQ_SAVEPRE 40175 +#define IDM_EQ_SAVEMP3 40176 +#define IDM_EQ_SAVEDEFAULT 40177 +#define IDM_EQ_DELPRE 40178 +#define IDM_EQ_DELMP3 40180 +#define IDC_PLAYLIST_PLAY 40184 +#define WINAMP_FILE_LOC 40185 +#define WINAMP_OPTIONS_EASYMOVE 40186 +#define WINAMP_FILE_DIR 40187 // pops up the load directory box +#define WINAMP_EDIT_ID3 40188 +#define WINAMP_TOGGLE_AUTOSCROLL 40189 +#define WINAMP_VISSETUP 40190 +#define WINAMP_PLGSETUP 40191 +#define WINAMP_VISPLUGIN 40192 +#define WINAMP_JUMP 40193 +#define WINAMP_JUMPFILE 40194 +#define WINAMP_JUMP10FWD 40195 +#define WINAMP_JUMP10BACK 40197 +#define WINAMP_PREVSONG 40198 +#define WINAMP_OPTIONS_EXTRAHQ 40200 +#define ID_PE_NEW 40201 +#define ID_PE_OPEN 40202 +#define ID_PE_SAVE 40203 +#define ID_PE_SAVEAS 40204 +#define ID_PE_SELECTALL 40205 +#define ID_PE_INVERT 40206 +#define ID_PE_NONE 40207 +#define ID_PE_ID3 40208 +#define ID_PE_S_TITLE 40209 +#define ID_PE_S_FILENAME 40210 +#define ID_PE_S_PATH 40211 +#define ID_PE_S_RANDOM 40212 +#define ID_PE_S_REV 40213 +#define ID_PE_CLEAR 40214 +#define ID_PE_MOVEUP 40215 +#define ID_PE_MOVEDOWN 40216 +#define WINAMP_SELSKIN 40219 +#define WINAMP_VISCONF 40221 +#define ID_PE_NONEXIST 40222 +#define ID_PE_DELETEFROMDISK 40223 +#define ID_PE_CLOSE 40224 +#define WINAMP_VIS_SETOSC 40226 +#define WINAMP_VIS_SETANA 40227 +#define WINAMP_VIS_SETOFF 40228 +#define WINAMP_VIS_DOTSCOPE 40229 +#define WINAMP_VIS_LINESCOPE 40230 +#define WINAMP_VIS_SOLIDSCOPE 40231 +#define WINAMP_VIS_NORMANA 40233 +#define WINAMP_VIS_FIREANA 40234 +#define WINAMP_VIS_LINEANA 40235 +#define WINAMP_VIS_NORMVU 40236 +#define WINAMP_VIS_SMOOTHVU 40237 +#define WINAMP_VIS_FULLREF 40238 +#define WINAMP_VIS_FULLREF2 40239 +#define WINAMP_VIS_FULLREF3 40240 +#define WINAMP_VIS_FULLREF4 40241 +#define WINAMP_OPTIONS_TOGTIME 40242 +#define EQ_ENABLE 40244 +#define EQ_AUTO 40245 +#define EQ_PRESETS 40246 +#define WINAMP_VIS_FALLOFF0 40247 +#define WINAMP_VIS_FALLOFF1 40248 +#define WINAMP_VIS_FALLOFF2 40249 +#define WINAMP_VIS_FALLOFF3 40250 +#define WINAMP_VIS_FALLOFF4 40251 +#define WINAMP_VIS_PEAKS 40252 +#define ID_LOAD_EQF 40253 +#define ID_SAVE_EQF 40254 +#define ID_PE_ENTRY 40255 +#define ID_PE_SCROLLUP 40256 +#define ID_PE_SCROLLDOWN 40257 +#define WINAMP_MAIN_WINDOW 40258 +#define WINAMP_VIS_PFALLOFF0 40259 +#define WINAMP_VIS_PFALLOFF1 40260 +#define WINAMP_VIS_PFALLOFF2 40261 +#define WINAMP_VIS_PFALLOFF3 40262 +#define WINAMP_VIS_PFALLOFF4 40263 +#define ID_PE_TOP 40264 +#define ID_PE_BOTTOM 40265 +#define WINAMP_OPTIONS_WINDOWSHADE_PL 40266 +#define EQ_INC1 40267 +#define EQ_INC2 40268 +#define EQ_INC3 40269 +#define EQ_INC4 40270 +#define EQ_INC5 40271 +#define EQ_INC6 40272 +#define EQ_INC7 40273 +#define EQ_INC8 40274 +#define EQ_INC9 40275 +#define EQ_INC10 40276 +#define EQ_INCPRE 40277 +#define EQ_DECPRE 40278 +#define EQ_DEC1 40279 +#define EQ_DEC2 40280 +#define EQ_DEC3 40281 +#define EQ_DEC4 40282 +#define EQ_DEC5 40283 +#define EQ_DEC6 40284 +#define EQ_DEC7 40285 +#define EQ_DEC8 40286 +#define EQ_DEC9 40287 +#define EQ_DEC10 40288 +#define ID_PE_SCUP 40289 +#define ID_PE_SCDOWN 40290 +#define WINAMP_REFRESHSKIN 40291 +#define ID_PE_PRINT 40292 +#define ID_PE_EXTINFO 40293 +#define WINAMP_PLAYLIST_ADVANCE 40294 +#define WINAMP_VIS_LIN 40295 +#define WINAMP_VIS_BAR 40296 +#define WINAMP_OPTIONS_MINIBROWSER 40298 +#define MB_FWD 40299 +#define MB_BACK 40300 +#define MB_RELOAD 40301 +#define MB_OPENMENU 40302 +#define MB_OPENLOC 40303 +#define WINAMP_NEW_INSTANCE 40305 +#define MB_UPDATE 40309 +#define WINAMP_OPTIONS_WINDOWSHADE_EQ 40310 +#define EQ_PANLEFT 40313 +#define EQ_PANRIGHT 40314 +#define WINAMP_GETMORESKINS 40316 +#define WINAMP_VIS_OPTIONS 40317 +#define WINAMP_PE_SEARCH 40318 +#define ID_PE_BOOKMARK 40319 +#define WINAMP_EDIT_BOOKMARKS 40320 +#define WINAMP_MAKECURBOOKMARK 40321 +#define ID_MAIN_PLAY_BOOKMARK_NONE 40322 +#define ID_MAIN_PLAY_AUDIOCD 40323 // starts playing the audio CD in the first CD reader +#define ID_MAIN_PLAY_AUDIOCD2 40324 // plays the 2nd +#define ID_MAIN_PLAY_AUDIOCD3 40325 // plays the 3rd +#define ID_MAIN_PLAY_AUDIOCD4 40326 // plays the 4th +#define WINAMP_OPTIONS_VIDEO 40328 +#define ID_VIDEOWND_ZOOMFULLSCREEN 40329 +#define ID_VIDEOWND_ZOOM100 40330 +#define ID_VIDEOWND_ZOOM200 40331 +#define ID_VIDEOWND_ZOOM50 40332 +#define ID_VIDEOWND_VIDEOOPTIONS 40333 +#define WINAMP_MINIMIZE 40334 +#define ID_PE_FONTBIGGER 40335 +#define ID_PE_FONTSMALLER 40336 +#define WINAMP_VIDEO_TOGGLE_FS 40337 +#define WINAMP_VIDEO_TVBUTTON 40338 +#define WINAMP_LIGHTNING_CLICK 40339 +#define ID_FILE_ADDTOLIBRARY 40344 +#define ID_HELP_HELPTOPICS 40347 +#define ID_HELP_GETTINGSTARTED 40348 +#define ID_HELP_WINAMPFORUMS 40349 +#define ID_PLAY_VOLUMEUP 40351 +#define ID_PLAY_VOLUMEDOWN 40352 +#define ID_PEFILE_OPENPLAYLISTFROMLIBRARY_NOPLAYLISTSINLIBRARY 40355 +#define ID_PEFILE_ADDFROMLIBRARY 40356 +#define ID_PEFILE_CLOSEPLAYLISTEDITOR 40357 +#define ID_PEPLAYLIST_PLAYLISTPREFERENCES 40358 +#define ID_MLFILE_NEWPLAYLIST 40359 +#define ID_MLFILE_LOADPLAYLIST 40360 +#define ID_MLFILE_SAVEPLAYLIST 40361 +#define ID_MLFILE_ADDMEDIATOLIBRARY 40362 +#define ID_MLFILE_CLOSEMEDIALIBRARY 40363 +#define ID_MLVIEW_NOWPLAYING 40364 +#define ID_MLVIEW_LOCALMEDIA_ALLMEDIA 40366 +#define ID_MLVIEW_LOCALMEDIA_AUDIO 40367 +#define ID_MLVIEW_LOCALMEDIA_VIDEO 40368 +#define ID_MLVIEW_PLAYLISTS_NOPLAYLISTINLIBRARY 40369 +#define ID_MLVIEW_INTERNETRADIO 40370 +#define ID_MLVIEW_INTERNETTV 40371 +#define ID_MLVIEW_LIBRARYPREFERENCES 40372 +#define ID_MLVIEW_DEVICES_NOAVAILABLEDEVICE 40373 +#define ID_MLFILE_IMPORTCURRENTPLAYLIST 40374 +#define ID_MLVIEW_MEDIA 40376 +#define ID_MLVIEW_PLAYLISTS 40377 +#define ID_MLVIEW_MEDIA_ALLMEDIA 40377 +#define ID_MLVIEW_DEVICES 40378 +#define ID_FILE_SHOWLIBRARY 40379 +#define ID_FILE_CLOSELIBRARY 40380 +#define ID_POST_PLAY_PLAYLIST 40381 +#define ID_VIS_NEXT 40382 +#define ID_VIS_PREV 40383 +#define ID_VIS_RANDOM 40384 +#define ID_MANAGEPLAYLISTS 40385 +#define ID_PREFS_SKIN_SWITCHTOSKIN 40386 +#define ID_PREFS_SKIN_DELETESKIN 40387 +#define ID_PREFS_SKIN_RENAMESKIN 40388 +#define ID_VIS_FS 40389 +#define ID_VIS_CFG 40390 +#define ID_VIS_MENU 40391 +#define ID_VIS_SET_FS_FLAG 40392 +#define ID_PE_SHOWPLAYING 40393 +#define ID_HELP_REGISTERWINAMPPRO 40394 +#define ID_PE_MANUAL_ADVANCE 40395 +#define WA_SONG_5_STAR_RATING 40396 +#define WA_SONG_4_STAR_RATING 40397 +#define WA_SONG_3_STAR_RATING 40398 +#define WA_SONG_2_STAR_RATING 40399 +#define WA_SONG_1_STAR_RATING 40400 +#define WA_SONG_NO_RATING 40401 +#define PL_SONG_5_STAR_RATING 40402 +#define PL_SONG_4_STAR_RATING 40403 +#define PL_SONG_3_STAR_RATING 40404 +#define PL_SONG_2_STAR_RATING 40405 +#define PL_SONG_1_STAR_RATING 40406 +#define PL_SONG_NO_RATING 40407 +#define AUDIO_TRACK_ONE 40408 +#define VIDEO_TRACK_ONE 40424 + +#define ID_SWITCH_COLOURTHEME 44500 +#define ID_GENFF_LIMIT 45000 + +#endif diff --git a/Externals/MusicMod/Player/Src/Winmain.cpp b/Externals/MusicMod/Player/Src/Winmain.cpp new file mode 100644 index 0000000000..bf9362a991 --- /dev/null +++ b/Externals/MusicMod/Player/Src/Winmain.cpp @@ -0,0 +1,236 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp, Open source Winamp core +// +// Copyright İ 2005 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#include "Global.h" +#include "Font.h" +#include "InputPlugin.h" +#include "OutputPlugin.h" +#include "VisPlugin.h" +#include "DspPlugin.h" +#include "GenPlugin.h" +#include "Main.h" +#include "Rebar.h" +#include "Playlist.h" +#include "Status.h" +#include "PluginManager.h" +#include "Prefs.h" +#include "Config.h" + +#include "Emabox/Emabox.h" + + + +#define PLUS_ALT ( FVIRTKEY | FALT ) +#define PLUS_CONTROL ( FVIRTKEY | FCONTROL ) +#define PLUS_CONTROL_ALT ( FVIRTKEY | FCONTROL | FALT ) +#define PLUS_CONTROL_SHIFT ( FVIRTKEY | FCONTROL | FSHIFT ) +#define PLUS_SHIFT ( FVIRTKEY | FSHIFT ) + + + +HINSTANCE g_hInstance = NULL; // extern + +TCHAR * szHomeDir = NULL; // extern +int iHomeDirLen = 0; // extern + +TCHAR * szPluginDir = NULL; // extern +int iPluginDirLen = 0; // extern + + + +TCHAR szCurDir[ MAX_PATH + 1 ] = TEXT( "" ); +ConfCurDir ccdCurDir( szCurDir, TEXT( "CurDir" ) ); + + +bool bWarnPluginsMissing; +ConfBool cbWarnPluginsMissing( &bWarnPluginsMissing, TEXT( "WarnPluginsMissing" ), CONF_MODE_PUBLIC, true ); + + + +//////////////////////////////////////////////////////////////////////////////// +/// +//////////////////////////////////////////////////////////////////////////////// +int WINAPI WinMain( HINSTANCE hInstance, HINSTANCE, LPSTR lpCmdLine, int nCmdShow ) +{ + g_hInstance = hInstance; + + + // Load full config + Conf::Init( hInstance ); + + + // Get home dir + szHomeDir = new TCHAR[ MAX_PATH + 1 ]; + iHomeDirLen = GetModuleFileName( NULL, szHomeDir, MAX_PATH ); + if( !iHomeDirLen ) return 1; + + TCHAR * walk = szHomeDir + iHomeDirLen - 1; + while( ( walk > szHomeDir ) && ( *walk != TEXT( '\\' ) ) ) walk--; + walk++; + *walk = TEXT( '\0' ); + iHomeDirLen = walk - szHomeDir; + + + // Get plugins dir + szPluginDir = new TCHAR[ MAX_PATH + 1 ]; + memcpy( szPluginDir, szHomeDir, iHomeDirLen * sizeof( TCHAR ) ); + memcpy( szPluginDir + iHomeDirLen, TEXT( "Plugins" ), 7 * sizeof( TCHAR ) ); + szPluginDir[ iHomeDirLen + 7 ] = TEXT( '\0' ); + + + Font::Create(); + BuildMainWindow(); + Prefs::Create(); + + + // Show window + ShowWindow( WindowMain, SW_SHOW ); + SetForegroundWindow( WindowMain ); + SetFocus( WindowMain ); + + Plugin::FindAll ( szPluginDir, TEXT( "in_*.dll" ), true ); + Plugin::FindAll( szPluginDir, TEXT( "out_*.dll" ), false ); + Plugin::FindAll ( szPluginDir, TEXT( "vis_*.dll" ), false ); + Plugin::FindAll ( szPluginDir, TEXT( "dsp_*.dll" ), false ); + Plugin::FindAll ( szPluginDir, TEXT( "gen_*.dll" ), true ); + + PluginManager::Fill(); + + + // Check plugin presence + // One msgbox maximum + if( bWarnPluginsMissing ) + { + + + if( input_plugins.empty() ) + { + // No input plugins + TCHAR szBuffer[ 5000 ]; + _stprintf( + szBuffer, + TEXT( + "No input plugins were found.\n" + "\n" + "Please install at least one Winamp input plugin to \n" + "%s " + ), + szPluginDir + ); + + int iNeverAgain = bWarnPluginsMissing ? 0 : 1; + EmaBox( 0, szBuffer, TEXT( "Input plugins missing" ), MB_ICONEXCLAMATION | MB_CHECKNEVERAGAIN, &iNeverAgain ); + bWarnPluginsMissing = ( iNeverAgain == 0 ); + } + else if( output_plugins.empty() ) + { + // No output plugins + TCHAR szBuffer[ 5000 ]; + _stprintf( + szBuffer, + TEXT( + "No output plugins were found.\n" + "\n" + "Please install at least one Winamp output plugin to \n" + "%s " + ), + szPluginDir + ); + + int iNeverAgain = bWarnPluginsMissing ? 0 : 1; + EmaBox( 0, szBuffer, TEXT( "Output plugins missing" ), MB_ICONEXCLAMATION | MB_CHECKNEVERAGAIN, &iNeverAgain ); + bWarnPluginsMissing = ( iNeverAgain == 0 ); + } + } + + + // Todo: all the rest... + ACCEL accels[] = { + { PLUS_CONTROL, 'A', ID_PE_SELECTALL }, // [Ctrl] + [A] + { PLUS_CONTROL, 'I', ID_PE_INVERT }, // [Ctrl] + [I] + { PLUS_CONTROL, 'N', ID_PE_NEW }, // [Ctrl] + [N] + { PLUS_CONTROL, 'O', ID_PE_OPEN }, // [Ctrl] + [O] + { PLUS_CONTROL, 'P', WINAMP_OPTIONS_PREFS }, // [Ctrl] + [P] + { PLUS_CONTROL, 'S', ID_PE_SAVEAS }, // [Ctrl] + [S] + { PLUS_CONTROL, VK_F1, WINAMP_HELP_ABOUT }, // [Ctrl] + [F1] + { PLUS_CONTROL_SHIFT, 'A', ID_PE_NONE }, // [Ctrl] + [Shift] + [A] + { PLUS_CONTROL_SHIFT, VK_DELETE, ID_PE_CLEAR }, // [Ctrl] + [Shift] + [Del] + { PLUS_ALT, 'F', WINAMP_MAINMENU }, // [Alt] + [F] + { PLUS_ALT, VK_F4, WINAMP_FILE_QUIT } // [Alt] + [F4] + }; + + + HACCEL hAccel = CreateAcceleratorTable( accels, sizeof( accels ) / sizeof( ACCEL ) ); + if( !hAccel ) + { + MessageBox( 0, TEXT( "Accelerator table error" ), TEXT( "" ), 0 ); + } + + // Message loop + MSG msg; + while( GetMessage( &msg, NULL, 0, 0 ) > 0 ) + { + // Note: Keys without [Alt] or [Ctrl] not everywhere! + if( ( ( msg.hwnd == WindowMain ) || IsChild( WindowMain, msg.hwnd ) ) && + TranslateAccelerator( WindowMain, hAccel, &msg ) ) + { + // MessageBox( 0, TEXT( "Trans" ), TEXT( "" ), 0 ); + } + + TranslateMessage( &msg ); + DispatchMessage( &msg ); + } + + DestroyAcceleratorTable( hAccel ); + + + // Input + vector ::iterator iter_input = input_plugins.begin(); + while( iter_input != input_plugins.end() ) + { + ( *iter_input )->Unload(); + iter_input++; + } + + // Output + vector ::iterator iter_output = output_plugins.begin(); + while( iter_output != output_plugins.end() ) + { + ( *iter_output )->Unload(); + iter_output++; + } + + // General + vector ::iterator iter_gen = gen_plugins.begin(); + while( iter_gen != gen_plugins.end() ) + { + ( *iter_gen )->Unload(); + iter_gen++; + } + + + // TODO: create main::destroy + // UnregisterClass( PA_CLASSNAME, g_hInstance ); + + Prefs::Destroy(); + + Font::Destroy(); + +/* + delete [] szPluginDir; + delete [] szHomeDir; +*/ + + Conf::Write(); + + return 0; +} diff --git a/Externals/MusicMod/Player/Src/afxres.h b/Externals/MusicMod/Player/Src/afxres.h new file mode 100644 index 0000000000..7eb4900177 --- /dev/null +++ b/Externals/MusicMod/Player/Src/afxres.h @@ -0,0 +1,823 @@ +// This is a part of the Microsoft Foundation Classes C++ library. +// Copyright (C) Microsoft Corporation +// All rights reserved. +// +// This source code is only intended as a supplement to the +// Microsoft Foundation Classes Reference and related +// electronic documentation provided with the library. +// See these sources for detailed information regarding the +// Microsoft Foundation Classes product. + +#ifndef __AFXRES_H__ +#define __AFXRES_H__ + +#ifdef RC_INVOKED +#ifndef _INC_WINDOWS +#define _INC_WINDOWS + #include "winres.h" // extract from windows header +#endif +#endif + +#ifdef _AFX_MINREBUILD +#pragma component(minrebuild, off) +#endif + +#ifdef RC_INVOKED + #if defined(_VC80_UPGRADE) && (_VC80_UPGRADE >= 0x0700) && (_VC80_UPGRADE < 0x0800) && defined(IDR_MANIFEST) + // Handle project upgrade from VC7/VC7.1 for projects with manifest + #define IDR_MANIFEST 1024 + #endif +#endif // RC_INVOKED + +#ifdef APSTUDIO_INVOKED +#define APSTUDIO_HIDDEN_SYMBOLS +#endif + +///////////////////////////////////////////////////////////////////////////// +// MFC resource types (see Technical note TN024 for implementation details) + +#ifndef RC_INVOKED +#define RT_DLGINIT MAKEINTRESOURCE(240) +#define RT_TOOLBAR MAKEINTRESOURCE(241) +#endif + +///////////////////////////////////////////////////////////////////////////// + +#ifdef APSTUDIO_INVOKED +#undef APSTUDIO_HIDDEN_SYMBOLS +#endif + +///////////////////////////////////////////////////////////////////////////// +// General style bits etc + +// ControlBar styles +#define CBRS_ALIGN_LEFT 0x1000L +#define CBRS_ALIGN_TOP 0x2000L +#define CBRS_ALIGN_RIGHT 0x4000L +#define CBRS_ALIGN_BOTTOM 0x8000L +#define CBRS_ALIGN_ANY 0xF000L + +#define CBRS_BORDER_LEFT 0x0100L +#define CBRS_BORDER_TOP 0x0200L +#define CBRS_BORDER_RIGHT 0x0400L +#define CBRS_BORDER_BOTTOM 0x0800L +#define CBRS_BORDER_ANY 0x0F00L + +#define CBRS_TOOLTIPS 0x0010L +#define CBRS_FLYBY 0x0020L +#define CBRS_FLOAT_MULTI 0x0040L +#define CBRS_BORDER_3D 0x0080L +#define CBRS_HIDE_INPLACE 0x0008L +#define CBRS_SIZE_DYNAMIC 0x0004L +#define CBRS_SIZE_FIXED 0x0002L +#define CBRS_FLOATING 0x0001L + +#define CBRS_GRIPPER 0x00400000L + +#define CBRS_ORIENT_HORZ (CBRS_ALIGN_TOP|CBRS_ALIGN_BOTTOM) +#define CBRS_ORIENT_VERT (CBRS_ALIGN_LEFT|CBRS_ALIGN_RIGHT) +#define CBRS_ORIENT_ANY (CBRS_ORIENT_HORZ|CBRS_ORIENT_VERT) + +#define CBRS_ALL 0x0040FFFFL + +// the CBRS_ style is made up of an alignment style and a draw border style +// the alignment styles are mutually exclusive +// the draw border styles may be combined +#define CBRS_NOALIGN 0x00000000L +#define CBRS_LEFT (CBRS_ALIGN_LEFT|CBRS_BORDER_RIGHT) +#define CBRS_TOP (CBRS_ALIGN_TOP|CBRS_BORDER_BOTTOM) +#define CBRS_RIGHT (CBRS_ALIGN_RIGHT|CBRS_BORDER_LEFT) +#define CBRS_BOTTOM (CBRS_ALIGN_BOTTOM|CBRS_BORDER_TOP) + +///////////////////////////////////////////////////////////////////////////// +// Manifest Resource ID of manifest containing Localized DLL information +#define ID_MFCLOC_MANIFEST 1000 + +///////////////////////////////////////////////////////////////////////////// +// Standard window components + +// Mode indicators in status bar - these are routed like commands +#define ID_INDICATOR_EXT 0xE700 // extended selection indicator +#define ID_INDICATOR_CAPS 0xE701 // cap lock indicator +#define ID_INDICATOR_NUM 0xE702 // num lock indicator +#define ID_INDICATOR_SCRL 0xE703 // scroll lock indicator +#define ID_INDICATOR_OVR 0xE704 // overtype mode indicator +#define ID_INDICATOR_REC 0xE705 // record mode indicator +#define ID_INDICATOR_KANA 0xE706 // kana lock indicator + +#define ID_SEPARATOR 0 // special separator value + +#ifndef RC_INVOKED // code only +// Standard control bars (IDW = window ID) +#define AFX_IDW_CONTROLBAR_FIRST 0xE800 +#define AFX_IDW_CONTROLBAR_LAST 0xE8FF + +#define AFX_IDW_TOOLBAR 0xE800 // main Toolbar for window +#define AFX_IDW_STATUS_BAR 0xE801 // Status bar window +#define AFX_IDW_PREVIEW_BAR 0xE802 // PrintPreview Dialog Bar +#define AFX_IDW_RESIZE_BAR 0xE803 // OLE in-place resize bar +#define AFX_IDW_REBAR 0xE804 // COMCTL32 "rebar" Bar +#define AFX_IDW_DIALOGBAR 0xE805 // CDialogBar + +// Note: If your application supports docking toolbars, you should +// not use the following IDs for your own toolbars. The IDs chosen +// are at the top of the first 32 such that the bars will be hidden +// while in print preview mode, and are not likely to conflict with +// IDs your application may have used succesfully in the past. + +#define AFX_IDW_DOCKBAR_TOP 0xE81B +#define AFX_IDW_DOCKBAR_LEFT 0xE81C +#define AFX_IDW_DOCKBAR_RIGHT 0xE81D +#define AFX_IDW_DOCKBAR_BOTTOM 0xE81E +#define AFX_IDW_DOCKBAR_FLOAT 0xE81F + +// Macro for mapping standard control bars to bitmask (limit of 32) +#define AFX_CONTROLBAR_MASK(nIDC) (1L << (nIDC - AFX_IDW_CONTROLBAR_FIRST)) + +// parts of Main Frame +#define AFX_IDW_PANE_FIRST 0xE900 // first pane (256 max) +#define AFX_IDW_PANE_LAST 0xE9ff +#define AFX_IDW_HSCROLL_FIRST 0xEA00 // first Horz scrollbar (16 max) +#define AFX_IDW_VSCROLL_FIRST 0xEA10 // first Vert scrollbar (16 max) + +#define AFX_IDW_SIZE_BOX 0xEA20 // size box for splitters +#define AFX_IDW_PANE_SAVE 0xEA21 // to shift AFX_IDW_PANE_FIRST +#endif //!RC_INVOKED + +#ifndef APSTUDIO_INVOKED + +// common style for form views +#define AFX_WS_DEFAULT_VIEW (WS_CHILD | WS_VISIBLE | WS_BORDER) + +#endif //!APSTUDIO_INVOKED + +///////////////////////////////////////////////////////////////////////////// +// Standard app configurable strings + +// for application title (defaults to EXE name or name in constructor) +#define AFX_IDS_APP_TITLE 0xE000 +// idle message bar line +#define AFX_IDS_IDLEMESSAGE 0xE001 +// message bar line when in shift-F1 help mode +#define AFX_IDS_HELPMODEMESSAGE 0xE002 +// document title when editing OLE embedding +#define AFX_IDS_APP_TITLE_EMBEDDING 0xE003 +// company name +#define AFX_IDS_COMPANY_NAME 0xE004 +// object name when server is inplace +#define AFX_IDS_OBJ_TITLE_INPLACE 0xE005 + +///////////////////////////////////////////////////////////////////////////// +// Standard Commands + +// File commands +#define ID_FILE_NEW 0xE100 +#define ID_FILE_OPEN 0xE101 +#define ID_FILE_CLOSE 0xE102 +#define ID_FILE_SAVE 0xE103 +#define ID_FILE_SAVE_AS 0xE104 +#define ID_FILE_PAGE_SETUP 0xE105 +#define ID_FILE_PRINT_SETUP 0xE106 +#define ID_FILE_PRINT 0xE107 +#define ID_FILE_PRINT_DIRECT 0xE108 +#define ID_FILE_PRINT_PREVIEW 0xE109 +#define ID_FILE_UPDATE 0xE10A +#define ID_FILE_SAVE_COPY_AS 0xE10B +#define ID_FILE_SEND_MAIL 0xE10C +#define ID_FILE_NEW_FRAME 0xE10D + +#define ID_FILE_MRU_FIRST 0xE110 +#define ID_FILE_MRU_FILE1 0xE110 // range - 16 max +#define ID_FILE_MRU_FILE2 0xE111 +#define ID_FILE_MRU_FILE3 0xE112 +#define ID_FILE_MRU_FILE4 0xE113 +#define ID_FILE_MRU_FILE5 0xE114 +#define ID_FILE_MRU_FILE6 0xE115 +#define ID_FILE_MRU_FILE7 0xE116 +#define ID_FILE_MRU_FILE8 0xE117 +#define ID_FILE_MRU_FILE9 0xE118 +#define ID_FILE_MRU_FILE10 0xE119 +#define ID_FILE_MRU_FILE11 0xE11A +#define ID_FILE_MRU_FILE12 0xE11B +#define ID_FILE_MRU_FILE13 0xE11C +#define ID_FILE_MRU_FILE14 0xE11D +#define ID_FILE_MRU_FILE15 0xE11E +#define ID_FILE_MRU_FILE16 0xE11F +#define ID_FILE_MRU_LAST 0xE11F + +// Edit commands +#define ID_EDIT_CLEAR 0xE120 +#define ID_EDIT_CLEAR_ALL 0xE121 +#define ID_EDIT_COPY 0xE122 +#define ID_EDIT_CUT 0xE123 +#define ID_EDIT_FIND 0xE124 +#define ID_EDIT_PASTE 0xE125 +#define ID_EDIT_PASTE_LINK 0xE126 +#define ID_EDIT_PASTE_SPECIAL 0xE127 +#define ID_EDIT_REPEAT 0xE128 +#define ID_EDIT_REPLACE 0xE129 +#define ID_EDIT_SELECT_ALL 0xE12A +#define ID_EDIT_UNDO 0xE12B +#define ID_EDIT_REDO 0xE12C + +// Window commands +#define ID_WINDOW_NEW 0xE130 +#define ID_WINDOW_ARRANGE 0xE131 +#define ID_WINDOW_CASCADE 0xE132 +#define ID_WINDOW_TILE_HORZ 0xE133 +#define ID_WINDOW_TILE_VERT 0xE134 +#define ID_WINDOW_SPLIT 0xE135 +#ifndef RC_INVOKED // code only +#define AFX_IDM_WINDOW_FIRST 0xE130 +#define AFX_IDM_WINDOW_LAST 0xE13F +#define AFX_IDM_FIRST_MDICHILD 0xFF00 // window list starts here +#endif //!RC_INVOKED + +// Help and App commands +#define ID_APP_ABOUT 0xE140 +#define ID_APP_EXIT 0xE141 +#define ID_HELP_INDEX 0xE142 +#define ID_HELP_FINDER 0xE143 +#define ID_HELP_USING 0xE144 +#define ID_CONTEXT_HELP 0xE145 // shift-F1 +// special commands for processing help +#define ID_HELP 0xE146 // first attempt for F1 +#define ID_DEFAULT_HELP 0xE147 // last attempt + +// Misc +#define ID_NEXT_PANE 0xE150 +#define ID_PREV_PANE 0xE151 + +// Format +#define ID_FORMAT_FONT 0xE160 + +// OLE commands +#define ID_OLE_INSERT_NEW 0xE200 +#define ID_OLE_EDIT_LINKS 0xE201 +#define ID_OLE_EDIT_CONVERT 0xE202 +#define ID_OLE_EDIT_CHANGE_ICON 0xE203 +#define ID_OLE_EDIT_PROPERTIES 0xE204 +#define ID_OLE_VERB_FIRST 0xE210 // range - 16 max +#ifndef RC_INVOKED // code only +#define ID_OLE_VERB_LAST 0xE21F +#endif //!RC_INVOKED + +// for print preview dialog bar +#define AFX_ID_PREVIEW_CLOSE 0xE300 +#define AFX_ID_PREVIEW_NUMPAGE 0xE301 // One/Two Page button +#define AFX_ID_PREVIEW_NEXT 0xE302 +#define AFX_ID_PREVIEW_PREV 0xE303 +#define AFX_ID_PREVIEW_PRINT 0xE304 +#define AFX_ID_PREVIEW_ZOOMIN 0xE305 +#define AFX_ID_PREVIEW_ZOOMOUT 0xE306 + +// View commands (same number used as IDW used for control bar) +#define ID_VIEW_TOOLBAR 0xE800 +#define ID_VIEW_STATUS_BAR 0xE801 +#define ID_VIEW_REBAR 0xE804 +#define ID_VIEW_AUTOARRANGE 0xE805 + // E810 -> E81F must be kept in order for RANGE macros +#define ID_VIEW_SMALLICON 0xE810 +#define ID_VIEW_LARGEICON 0xE811 +#define ID_VIEW_LIST 0xE812 +#define ID_VIEW_DETAILS 0xE813 +#define ID_VIEW_LINEUP 0xE814 +#define ID_VIEW_BYNAME 0xE815 +#define AFX_ID_VIEW_MINIMUM ID_VIEW_SMALLICON +#define AFX_ID_VIEW_MAXIMUM ID_VIEW_BYNAME + // E800 -> E8FF reserved for other control bar commands + +// RecordForm commands +#define ID_RECORD_FIRST 0xE900 +#define ID_RECORD_LAST 0xE901 +#define ID_RECORD_NEXT 0xE902 +#define ID_RECORD_PREV 0xE903 + +///////////////////////////////////////////////////////////////////////////// +// Standard control IDs + +#ifdef IDC_STATIC +#undef IDC_STATIC +#endif +#define IDC_STATIC (-1) // all static controls + +///////////////////////////////////////////////////////////////////////////// +// Standard string error/warnings + +#ifndef RC_INVOKED // code only +#define AFX_IDS_SCFIRST 0xEF00 +#endif //!RC_INVOKED + +#define AFX_IDS_SCSIZE 0xEF00 +#define AFX_IDS_SCMOVE 0xEF01 +#define AFX_IDS_SCMINIMIZE 0xEF02 +#define AFX_IDS_SCMAXIMIZE 0xEF03 +#define AFX_IDS_SCNEXTWINDOW 0xEF04 +#define AFX_IDS_SCPREVWINDOW 0xEF05 +#define AFX_IDS_SCCLOSE 0xEF06 +#define AFX_IDS_SCRESTORE 0xEF12 +#define AFX_IDS_SCTASKLIST 0xEF13 + +#define AFX_IDS_MDICHILD 0xEF1F + +#define AFX_IDS_DESKACCESSORY 0xEFDA + +// General strings +#define AFX_IDS_OPENFILE 0xF000 +#define AFX_IDS_SAVEFILE 0xF001 +#define AFX_IDS_ALLFILTER 0xF002 +#define AFX_IDS_UNTITLED 0xF003 +#define AFX_IDS_SAVEFILECOPY 0xF004 +#define AFX_IDS_PREVIEW_CLOSE 0xF005 +#define AFX_IDS_UNNAMED_FILE 0xF006 +#define AFX_IDS_HIDE 0xF011 + +// MFC Standard Exception Error messages +#define AFX_IDP_NO_ERROR_AVAILABLE 0xF020 +#define AFX_IDS_NOT_SUPPORTED_EXCEPTION 0xF021 +#define AFX_IDS_RESOURCE_EXCEPTION 0xF022 +#define AFX_IDS_MEMORY_EXCEPTION 0xF023 +#define AFX_IDS_USER_EXCEPTION 0xF024 +#define AFX_IDS_INVALID_ARG_EXCEPTION 0xF025 + +// Printing and print preview strings +#define AFX_IDS_PRINTONPORT 0xF040 +#define AFX_IDS_ONEPAGE 0xF041 +#define AFX_IDS_TWOPAGE 0xF042 +#define AFX_IDS_PRINTPAGENUM 0xF043 +#define AFX_IDS_PREVIEWPAGEDESC 0xF044 +#define AFX_IDS_PRINTDEFAULTEXT 0xF045 +#define AFX_IDS_PRINTDEFAULT 0xF046 +#define AFX_IDS_PRINTFILTER 0xF047 +#define AFX_IDS_PRINTCAPTION 0xF048 +#define AFX_IDS_PRINTTOFILE 0xF049 + + +// OLE strings +#define AFX_IDS_OBJECT_MENUITEM 0xF080 +#define AFX_IDS_EDIT_VERB 0xF081 +#define AFX_IDS_ACTIVATE_VERB 0xF082 +#define AFX_IDS_CHANGE_LINK 0xF083 +#define AFX_IDS_AUTO 0xF084 +#define AFX_IDS_MANUAL 0xF085 +#define AFX_IDS_FROZEN 0xF086 +#define AFX_IDS_ALL_FILES 0xF087 +// dynamically changing menu items +#define AFX_IDS_SAVE_MENU 0xF088 +#define AFX_IDS_UPDATE_MENU 0xF089 +#define AFX_IDS_SAVE_AS_MENU 0xF08A +#define AFX_IDS_SAVE_COPY_AS_MENU 0xF08B +#define AFX_IDS_EXIT_MENU 0xF08C +#define AFX_IDS_UPDATING_ITEMS 0xF08D +// COlePasteSpecialDialog defines +#define AFX_IDS_METAFILE_FORMAT 0xF08E +#define AFX_IDS_DIB_FORMAT 0xF08F +#define AFX_IDS_BITMAP_FORMAT 0xF090 +#define AFX_IDS_LINKSOURCE_FORMAT 0xF091 +#define AFX_IDS_EMBED_FORMAT 0xF092 +// other OLE utility strings +#define AFX_IDS_PASTELINKEDTYPE 0xF094 +#define AFX_IDS_UNKNOWNTYPE 0xF095 +#define AFX_IDS_RTF_FORMAT 0xF096 +#define AFX_IDS_TEXT_FORMAT 0xF097 +// OLE datatype format error strings +#define AFX_IDS_INVALID_CURRENCY 0xF098 +#define AFX_IDS_INVALID_DATETIME 0xF099 +#define AFX_IDS_INVALID_DATETIMESPAN 0xF09A + +// General error / prompt strings +#define AFX_IDP_INVALID_FILENAME 0xF100 +#define AFX_IDP_FAILED_TO_OPEN_DOC 0xF101 +#define AFX_IDP_FAILED_TO_SAVE_DOC 0xF102 +#define AFX_IDP_ASK_TO_SAVE 0xF103 +#define AFX_IDP_FAILED_TO_CREATE_DOC 0xF104 +#define AFX_IDP_FILE_TOO_LARGE 0xF105 +#define AFX_IDP_FAILED_TO_START_PRINT 0xF106 +#define AFX_IDP_FAILED_TO_LAUNCH_HELP 0xF107 +#define AFX_IDP_INTERNAL_FAILURE 0xF108 // general failure +#define AFX_IDP_COMMAND_FAILURE 0xF109 // command failure +#define AFX_IDP_FAILED_MEMORY_ALLOC 0xF10A +#define AFX_IDP_UNREG_DONE 0xF10B +#define AFX_IDP_UNREG_FAILURE 0xF10C +#define AFX_IDP_DLL_LOAD_FAILED 0xF10D +#define AFX_IDP_DLL_BAD_VERSION 0xF10E + +// DDV parse errors +#define AFX_IDP_PARSE_INT 0xF110 +#define AFX_IDP_PARSE_REAL 0xF111 +#define AFX_IDP_PARSE_INT_RANGE 0xF112 +#define AFX_IDP_PARSE_REAL_RANGE 0xF113 +#define AFX_IDP_PARSE_STRING_SIZE 0xF114 +#define AFX_IDP_PARSE_RADIO_BUTTON 0xF115 +#define AFX_IDP_PARSE_BYTE 0xF116 +#define AFX_IDP_PARSE_UINT 0xF117 +#define AFX_IDP_PARSE_DATETIME 0xF118 +#define AFX_IDP_PARSE_CURRENCY 0xF119 +#define AFX_IDP_PARSE_GUID 0xF11A +#define AFX_IDP_PARSE_TIME 0xF11B +#define AFX_IDP_PARSE_DATE 0xF11C + +// CFile/CArchive error strings for user failure +#define AFX_IDP_FAILED_INVALID_FORMAT 0xF120 +#define AFX_IDP_FAILED_INVALID_PATH 0xF121 +#define AFX_IDP_FAILED_DISK_FULL 0xF122 +#define AFX_IDP_FAILED_ACCESS_READ 0xF123 +#define AFX_IDP_FAILED_ACCESS_WRITE 0xF124 +#define AFX_IDP_FAILED_IO_ERROR_READ 0xF125 +#define AFX_IDP_FAILED_IO_ERROR_WRITE 0xF126 + +// Script errors / prompt strings +#define AFX_IDP_SCRIPT_ERROR 0xF130 +#define AFX_IDP_SCRIPT_DISPATCH_EXCEPTION 0xF131 + +// OLE errors / prompt strings +#define AFX_IDP_STATIC_OBJECT 0xF180 +#define AFX_IDP_FAILED_TO_CONNECT 0xF181 +#define AFX_IDP_SERVER_BUSY 0xF182 +#define AFX_IDP_BAD_VERB 0xF183 +#define AFX_IDS_NOT_DOCOBJECT 0xF184 +#define AFX_IDP_FAILED_TO_NOTIFY 0xF185 +#define AFX_IDP_FAILED_TO_LAUNCH 0xF186 +#define AFX_IDP_ASK_TO_UPDATE 0xF187 +#define AFX_IDP_FAILED_TO_UPDATE 0xF188 +#define AFX_IDP_FAILED_TO_REGISTER 0xF189 +#define AFX_IDP_FAILED_TO_AUTO_REGISTER 0xF18A +#define AFX_IDP_FAILED_TO_CONVERT 0xF18B +#define AFX_IDP_GET_NOT_SUPPORTED 0xF18C +#define AFX_IDP_SET_NOT_SUPPORTED 0xF18D +#define AFX_IDP_ASK_TO_DISCARD 0xF18E +#define AFX_IDP_FAILED_TO_CREATE 0xF18F + +// MAPI errors / prompt strings +#define AFX_IDP_FAILED_MAPI_LOAD 0xF190 +#define AFX_IDP_INVALID_MAPI_DLL 0xF191 +#define AFX_IDP_FAILED_MAPI_SEND 0xF192 + +#define AFX_IDP_FILE_NONE 0xF1A0 +#define AFX_IDP_FILE_GENERIC 0xF1A1 +#define AFX_IDP_FILE_NOT_FOUND 0xF1A2 +#define AFX_IDP_FILE_BAD_PATH 0xF1A3 +#define AFX_IDP_FILE_TOO_MANY_OPEN 0xF1A4 +#define AFX_IDP_FILE_ACCESS_DENIED 0xF1A5 +#define AFX_IDP_FILE_INVALID_FILE 0xF1A6 +#define AFX_IDP_FILE_REMOVE_CURRENT 0xF1A7 +#define AFX_IDP_FILE_DIR_FULL 0xF1A8 +#define AFX_IDP_FILE_BAD_SEEK 0xF1A9 +#define AFX_IDP_FILE_HARD_IO 0xF1AA +#define AFX_IDP_FILE_SHARING 0xF1AB +#define AFX_IDP_FILE_LOCKING 0xF1AC +#define AFX_IDP_FILE_DISKFULL 0xF1AD +#define AFX_IDP_FILE_EOF 0xF1AE + +#define AFX_IDP_ARCH_NONE 0xF1B0 +#define AFX_IDP_ARCH_GENERIC 0xF1B1 +#define AFX_IDP_ARCH_READONLY 0xF1B2 +#define AFX_IDP_ARCH_ENDOFFILE 0xF1B3 +#define AFX_IDP_ARCH_WRITEONLY 0xF1B4 +#define AFX_IDP_ARCH_BADINDEX 0xF1B5 +#define AFX_IDP_ARCH_BADCLASS 0xF1B6 +#define AFX_IDP_ARCH_BADSCHEMA 0xF1B7 + +#define AFX_IDS_OCC_SCALEUNITS_PIXELS 0xF1C0 + +// 0xf200-0xf20f reserved + +// font names and point sizes +#define AFX_IDS_STATUS_FONT 0xF230 +#define AFX_IDS_TOOLTIP_FONT 0xF231 +#define AFX_IDS_UNICODE_FONT 0xF232 +#define AFX_IDS_MINI_FONT 0xF233 + +// ODBC Database errors / prompt strings +#ifndef RC_INVOKED // code only +#define AFX_IDP_SQL_FIRST 0xF280 +#endif //!RC_INVOKED +#define AFX_IDP_SQL_CONNECT_FAIL 0xF281 +#define AFX_IDP_SQL_RECORDSET_FORWARD_ONLY 0xF282 +#define AFX_IDP_SQL_EMPTY_COLUMN_LIST 0xF283 +#define AFX_IDP_SQL_FIELD_SCHEMA_MISMATCH 0xF284 +#define AFX_IDP_SQL_ILLEGAL_MODE 0xF285 +#define AFX_IDP_SQL_MULTIPLE_ROWS_AFFECTED 0xF286 +#define AFX_IDP_SQL_NO_CURRENT_RECORD 0xF287 +#define AFX_IDP_SQL_NO_ROWS_AFFECTED 0xF288 +#define AFX_IDP_SQL_RECORDSET_READONLY 0xF289 +#define AFX_IDP_SQL_SQL_NO_TOTAL 0xF28A +#define AFX_IDP_SQL_ODBC_LOAD_FAILED 0xF28B +#define AFX_IDP_SQL_DYNASET_NOT_SUPPORTED 0xF28C +#define AFX_IDP_SQL_SNAPSHOT_NOT_SUPPORTED 0xF28D +#define AFX_IDP_SQL_API_CONFORMANCE 0xF28E +#define AFX_IDP_SQL_SQL_CONFORMANCE 0xF28F +#define AFX_IDP_SQL_NO_DATA_FOUND 0xF290 +#define AFX_IDP_SQL_ROW_UPDATE_NOT_SUPPORTED 0xF291 +#define AFX_IDP_SQL_ODBC_V2_REQUIRED 0xF292 +#define AFX_IDP_SQL_NO_POSITIONED_UPDATES 0xF293 +#define AFX_IDP_SQL_LOCK_MODE_NOT_SUPPORTED 0xF294 +#define AFX_IDP_SQL_DATA_TRUNCATED 0xF295 +#define AFX_IDP_SQL_ROW_FETCH 0xF296 +#define AFX_IDP_SQL_INCORRECT_ODBC 0xF297 +#define AFX_IDP_SQL_UPDATE_DELETE_FAILED 0xF298 +#define AFX_IDP_SQL_DYNAMIC_CURSOR_NOT_SUPPORTED 0xF299 +#define AFX_IDP_SQL_FIELD_NOT_FOUND 0xF29A +#define AFX_IDP_SQL_BOOKMARKS_NOT_SUPPORTED 0xF29B +#define AFX_IDP_SQL_BOOKMARKS_NOT_ENABLED 0xF29C + +// ODBC Database strings +#define AFX_IDS_DELETED 0xF29D + +// DAO Database errors / prompt strings +#ifndef RC_INVOKED // code only +#define AFX_IDP_DAO_FIRST 0xF2B0 +#endif //!RC_INVOKED +#define AFX_IDP_DAO_ENGINE_INITIALIZATION 0xF2B0 +#define AFX_IDP_DAO_DFX_BIND 0xF2B1 +#define AFX_IDP_DAO_OBJECT_NOT_OPEN 0xF2B2 + +// ICDAORecordset::GetRows Errors +// These are not placed in DAO Errors collection +// and must be handled directly by MFC. +#define AFX_IDP_DAO_ROWTOOSHORT 0xF2B3 +#define AFX_IDP_DAO_BADBINDINFO 0xF2B4 +#define AFX_IDP_DAO_COLUMNUNAVAILABLE 0xF2B5 + +///////////////////////////////////////////////////////////////////////////// +// Strings for ISAPI support + +#define AFX_IDS_HTTP_TITLE 0xF2D1 +#define AFX_IDS_HTTP_NO_TEXT 0xF2D2 +#define AFX_IDS_HTTP_BAD_REQUEST 0xF2D3 +#define AFX_IDS_HTTP_AUTH_REQUIRED 0xF2D4 +#define AFX_IDS_HTTP_FORBIDDEN 0xF2D5 +#define AFX_IDS_HTTP_NOT_FOUND 0xF2D6 +#define AFX_IDS_HTTP_SERVER_ERROR 0xF2D7 +#define AFX_IDS_HTTP_NOT_IMPLEMENTED 0xF2D8 + +///////////////////////////////////////////////////////////////////////////// +// Strings for Accessibility support for CCheckListBox +#define AFX_IDS_CHECKLISTBOX_UNCHECK 0xF2E1 +#define AFX_IDS_CHECKLISTBOX_CHECK 0xF2E2 +#define AFX_IDS_CHECKLISTBOX_MIXED 0xF2E3 + +///////////////////////////////////////////////////////////////////////////// +// AFX implementation - control IDs (AFX_IDC) + +// Parts of dialogs +#define AFX_IDC_LISTBOX 100 +#define AFX_IDC_CHANGE 101 +#define AFX_IDC_BROWSER 102 + +// for print dialog +#define AFX_IDC_PRINT_DOCNAME 201 +#define AFX_IDC_PRINT_PRINTERNAME 202 +#define AFX_IDC_PRINT_PORTNAME 203 +#define AFX_IDC_PRINT_PAGENUM 204 + +// Property Sheet control id's (determined with Spy++) +#define ID_APPLY_NOW 0x3021 +#define ID_WIZBACK 0x3023 +#define ID_WIZNEXT 0x3024 +#define ID_WIZFINISH 0x3025 +#define AFX_IDC_TAB_CONTROL 0x3020 + +///////////////////////////////////////////////////////////////////////////// +// IDRs for standard components + +#ifndef RC_INVOKED // code only +// These are really COMMDLG dialogs, so there usually isn't a resource +// for them, but these IDs are used as help IDs. +#define AFX_IDD_FILEOPEN 28676 +#define AFX_IDD_FILESAVE 28677 +#define AFX_IDD_FONT 28678 +#define AFX_IDD_COLOR 28679 +#define AFX_IDD_PRINT 28680 +#define AFX_IDD_PRINTSETUP 28681 +#define AFX_IDD_FIND 28682 +#define AFX_IDD_REPLACE 28683 +#endif //!RC_INVOKED + +// Standard dialogs app should leave alone (0x7801->) +#define AFX_IDD_NEWTYPEDLG 30721 +#define AFX_IDD_PRINTDLG 30722 +#define AFX_IDD_PREVIEW_TOOLBAR 30723 + +// Dialogs defined for OLE2UI library +#define AFX_IDD_INSERTOBJECT 30724 +#define AFX_IDD_CHANGEICON 30725 +#define AFX_IDD_CONVERT 30726 +#define AFX_IDD_PASTESPECIAL 30727 +#define AFX_IDD_EDITLINKS 30728 +#define AFX_IDD_FILEBROWSE 30729 +#define AFX_IDD_BUSY 30730 + +#define AFX_IDD_OBJECTPROPERTIES 30732 +#define AFX_IDD_CHANGESOURCE 30733 + +//WinForms +#define AFX_IDD_EMPTYDIALOG 30734 + +// Standard cursors (0x7901->) + // AFX_IDC = Cursor resources +#define AFX_IDC_CONTEXTHELP 30977 // context sensitive help +#define AFX_IDC_MAGNIFY 30978 // print preview zoom +#define AFX_IDC_SMALLARROWS 30979 // splitter +#define AFX_IDC_HSPLITBAR 30980 // splitter +#define AFX_IDC_VSPLITBAR 30981 // splitter +#define AFX_IDC_NODROPCRSR 30982 // No Drop Cursor +#define AFX_IDC_TRACKNWSE 30983 // tracker +#define AFX_IDC_TRACKNESW 30984 // tracker +#define AFX_IDC_TRACKNS 30985 // tracker +#define AFX_IDC_TRACKWE 30986 // tracker +#define AFX_IDC_TRACK4WAY 30987 // tracker +#define AFX_IDC_MOVE4WAY 30988 // resize bar (server only) + +// Wheel mouse cursors +// NOTE: values must be in this order! See CScrollView::OnTimer() +#define AFX_IDC_MOUSE_PAN_NW 30998 // pan east +#define AFX_IDC_MOUSE_PAN_N 30999 // pan northeast +#define AFX_IDC_MOUSE_PAN_NE 31000 // pan north +#define AFX_IDC_MOUSE_PAN_W 31001 // pan northwest +#define AFX_IDC_MOUSE_PAN_HV 31002 // pan both axis +#define AFX_IDC_MOUSE_PAN_E 31003 // pan west +#define AFX_IDC_MOUSE_PAN_SW 31004 // pan south-west +#define AFX_IDC_MOUSE_PAN_S 31005 // pan south +#define AFX_IDC_MOUSE_PAN_SE 31006 // pan south-east +#define AFX_IDC_MOUSE_PAN_HORZ 31007 // pan X-axis +#define AFX_IDC_MOUSE_PAN_VERT 31008 // pan Y-axis + +// Wheel mouse bitmaps +#define AFX_IDC_MOUSE_ORG_HORZ 31009 // anchor for horz only +#define AFX_IDC_MOUSE_ORG_VERT 31010 // anchor for vert only +#define AFX_IDC_MOUSE_ORG_HV 31011 // anchor for horz/vert +#define AFX_IDC_MOUSE_MASK 31012 + +// Mini frame window bitmap ID +#define AFX_IDB_MINIFRAME_MENU 30994 + +// CheckListBox checks bitmap ID +#define AFX_IDB_CHECKLISTBOX_95 30996 + +// AFX standard accelerator resources +#define AFX_IDR_PREVIEW_ACCEL 30997 + +// AFX standard ICON IDs (for MFC V1 apps) (0x7A01->) +#define AFX_IDI_STD_MDIFRAME 31233 +#define AFX_IDI_STD_FRAME 31234 + +///////////////////////////////////////////////////////////////////////////// +// AFX OLE control implementation - control IDs (AFX_IDC) + +// Font property page +#define AFX_IDC_FONTPROP 1000 +#define AFX_IDC_FONTNAMES 1001 +#define AFX_IDC_FONTSTYLES 1002 +#define AFX_IDC_FONTSIZES 1003 +#define AFX_IDC_STRIKEOUT 1004 +#define AFX_IDC_UNDERLINE 1005 +#define AFX_IDC_SAMPLEBOX 1006 + +// Color property page +#define AFX_IDC_COLOR_BLACK 1100 +#define AFX_IDC_COLOR_WHITE 1101 +#define AFX_IDC_COLOR_RED 1102 +#define AFX_IDC_COLOR_GREEN 1103 +#define AFX_IDC_COLOR_BLUE 1104 +#define AFX_IDC_COLOR_YELLOW 1105 +#define AFX_IDC_COLOR_MAGENTA 1106 +#define AFX_IDC_COLOR_CYAN 1107 +#define AFX_IDC_COLOR_GRAY 1108 +#define AFX_IDC_COLOR_LIGHTGRAY 1109 +#define AFX_IDC_COLOR_DARKRED 1110 +#define AFX_IDC_COLOR_DARKGREEN 1111 +#define AFX_IDC_COLOR_DARKBLUE 1112 +#define AFX_IDC_COLOR_LIGHTBROWN 1113 +#define AFX_IDC_COLOR_DARKMAGENTA 1114 +#define AFX_IDC_COLOR_DARKCYAN 1115 +#define AFX_IDC_COLORPROP 1116 +#define AFX_IDC_SYSTEMCOLORS 1117 + +// Picture porperty page +#define AFX_IDC_PROPNAME 1201 +#define AFX_IDC_PICTURE 1202 +#define AFX_IDC_BROWSE 1203 +#define AFX_IDC_CLEAR 1204 + +///////////////////////////////////////////////////////////////////////////// +// IDRs for OLE control standard components + +// Standard propery page dialogs app should leave alone (0x7E01->) +#define AFX_IDD_PROPPAGE_COLOR 32257 +#define AFX_IDD_PROPPAGE_FONT 32258 +#define AFX_IDD_PROPPAGE_PICTURE 32259 + +#define AFX_IDB_TRUETYPE 32384 + +///////////////////////////////////////////////////////////////////////////// +// Standard OLE control strings + +// OLE Control page strings +#define AFX_IDS_PROPPAGE_UNKNOWN 0xFE01 +#define AFX_IDS_COLOR_DESKTOP 0xFE04 +#define AFX_IDS_COLOR_APPWORKSPACE 0xFE05 +#define AFX_IDS_COLOR_WNDBACKGND 0xFE06 +#define AFX_IDS_COLOR_WNDTEXT 0xFE07 +#define AFX_IDS_COLOR_MENUBAR 0xFE08 +#define AFX_IDS_COLOR_MENUTEXT 0xFE09 +#define AFX_IDS_COLOR_ACTIVEBAR 0xFE0A +#define AFX_IDS_COLOR_INACTIVEBAR 0xFE0B +#define AFX_IDS_COLOR_ACTIVETEXT 0xFE0C +#define AFX_IDS_COLOR_INACTIVETEXT 0xFE0D +#define AFX_IDS_COLOR_ACTIVEBORDER 0xFE0E +#define AFX_IDS_COLOR_INACTIVEBORDER 0xFE0F +#define AFX_IDS_COLOR_WNDFRAME 0xFE10 +#define AFX_IDS_COLOR_SCROLLBARS 0xFE11 +#define AFX_IDS_COLOR_BTNFACE 0xFE12 +#define AFX_IDS_COLOR_BTNSHADOW 0xFE13 +#define AFX_IDS_COLOR_BTNTEXT 0xFE14 +#define AFX_IDS_COLOR_BTNHIGHLIGHT 0xFE15 +#define AFX_IDS_COLOR_DISABLEDTEXT 0xFE16 +#define AFX_IDS_COLOR_HIGHLIGHT 0xFE17 +#define AFX_IDS_COLOR_HIGHLIGHTTEXT 0xFE18 +#define AFX_IDS_REGULAR 0xFE19 +#define AFX_IDS_BOLD 0xFE1A +#define AFX_IDS_ITALIC 0xFE1B +#define AFX_IDS_BOLDITALIC 0xFE1C +#define AFX_IDS_SAMPLETEXT 0xFE1D +#define AFX_IDS_DISPLAYSTRING_FONT 0xFE1E +#define AFX_IDS_DISPLAYSTRING_COLOR 0xFE1F +#define AFX_IDS_DISPLAYSTRING_PICTURE 0xFE20 +#define AFX_IDS_PICTUREFILTER 0xFE21 +#define AFX_IDS_PICTYPE_UNKNOWN 0xFE22 +#define AFX_IDS_PICTYPE_NONE 0xFE23 +#define AFX_IDS_PICTYPE_BITMAP 0xFE24 +#define AFX_IDS_PICTYPE_METAFILE 0xFE25 +#define AFX_IDS_PICTYPE_ICON 0xFE26 +#define AFX_IDS_COLOR_PPG 0xFE28 +#define AFX_IDS_COLOR_PPG_CAPTION 0xFE29 +#define AFX_IDS_FONT_PPG 0xFE2A +#define AFX_IDS_FONT_PPG_CAPTION 0xFE2B +#define AFX_IDS_PICTURE_PPG 0xFE2C +#define AFX_IDS_PICTURE_PPG_CAPTION 0xFE2D +#define AFX_IDS_PICTUREBROWSETITLE 0xFE30 +#define AFX_IDS_BORDERSTYLE_0 0xFE31 +#define AFX_IDS_BORDERSTYLE_1 0xFE32 + +// OLE Control verb names +#define AFX_IDS_VERB_EDIT 0xFE40 +#define AFX_IDS_VERB_PROPERTIES 0xFE41 + +// OLE Control internal error messages +#define AFX_IDP_PICTURECANTOPEN 0xFE83 +#define AFX_IDP_PICTURECANTLOAD 0xFE84 +#define AFX_IDP_PICTURETOOLARGE 0xFE85 +#define AFX_IDP_PICTUREREADFAILED 0xFE86 + +// Standard OLE Control error strings +#define AFX_IDP_E_ILLEGALFUNCTIONCALL 0xFEA0 +#define AFX_IDP_E_OVERFLOW 0xFEA1 +#define AFX_IDP_E_OUTOFMEMORY 0xFEA2 +#define AFX_IDP_E_DIVISIONBYZERO 0xFEA3 +#define AFX_IDP_E_OUTOFSTRINGSPACE 0xFEA4 +#define AFX_IDP_E_OUTOFSTACKSPACE 0xFEA5 +#define AFX_IDP_E_BADFILENAMEORNUMBER 0xFEA6 +#define AFX_IDP_E_FILENOTFOUND 0xFEA7 +#define AFX_IDP_E_BADFILEMODE 0xFEA8 +#define AFX_IDP_E_FILEALREADYOPEN 0xFEA9 +#define AFX_IDP_E_DEVICEIOERROR 0xFEAA +#define AFX_IDP_E_FILEALREADYEXISTS 0xFEAB +#define AFX_IDP_E_BADRECORDLENGTH 0xFEAC +#define AFX_IDP_E_DISKFULL 0xFEAD +#define AFX_IDP_E_BADRECORDNUMBER 0xFEAE +#define AFX_IDP_E_BADFILENAME 0xFEAF +#define AFX_IDP_E_TOOMANYFILES 0xFEB0 +#define AFX_IDP_E_DEVICEUNAVAILABLE 0xFEB1 +#define AFX_IDP_E_PERMISSIONDENIED 0xFEB2 +#define AFX_IDP_E_DISKNOTREADY 0xFEB3 +#define AFX_IDP_E_PATHFILEACCESSERROR 0xFEB4 +#define AFX_IDP_E_PATHNOTFOUND 0xFEB5 +#define AFX_IDP_E_INVALIDPATTERNSTRING 0xFEB6 +#define AFX_IDP_E_INVALIDUSEOFNULL 0xFEB7 +#define AFX_IDP_E_INVALIDFILEFORMAT 0xFEB8 +#define AFX_IDP_E_INVALIDPROPERTYVALUE 0xFEB9 +#define AFX_IDP_E_INVALIDPROPERTYARRAYINDEX 0xFEBA +#define AFX_IDP_E_SETNOTSUPPORTEDATRUNTIME 0xFEBB +#define AFX_IDP_E_SETNOTSUPPORTED 0xFEBC +#define AFX_IDP_E_NEEDPROPERTYARRAYINDEX 0xFEBD +#define AFX_IDP_E_SETNOTPERMITTED 0xFEBE +#define AFX_IDP_E_GETNOTSUPPORTEDATRUNTIME 0xFEBF +#define AFX_IDP_E_GETNOTSUPPORTED 0xFEC0 +#define AFX_IDP_E_PROPERTYNOTFOUND 0xFEC1 +#define AFX_IDP_E_INVALIDCLIPBOARDFORMAT 0xFEC2 +#define AFX_IDP_E_INVALIDPICTURE 0xFEC3 +#define AFX_IDP_E_PRINTERERROR 0xFEC4 +#define AFX_IDP_E_CANTSAVEFILETOTEMP 0xFEC5 +#define AFX_IDP_E_SEARCHTEXTNOTFOUND 0xFEC6 +#define AFX_IDP_E_REPLACEMENTSTOOLONG 0xFEC7 + +///////////////////////////////////////////////////////////////////////////// + +#ifdef _AFX_MINREBUILD +#pragma component(minrebuild, on) +#endif + +#endif //__AFXRES_H__ + +///////////////////////////////////////////////////////////////////////////// diff --git a/Externals/MusicMod/Player/Src/fftw3/fftw3.h b/Externals/MusicMod/Player/Src/fftw3/fftw3.h new file mode 100644 index 0000000000..64e445d30d --- /dev/null +++ b/Externals/MusicMod/Player/Src/fftw3/fftw3.h @@ -0,0 +1,276 @@ +/* + * Copyright (c) 2003 Matteo Frigo + * Copyright (c) 2003 Massachusetts Institute of Technology + * + * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + */ + +/* header file for fftw3 */ +/* $Id: fftw3.h,v 1.1 2005/10/28 19:01:21 hartwork Exp $ */ + +#ifndef FFTW3_H +#define FFTW3_H + +#if defined(__ICC) || defined(_MSC_VER) +#pragma comment ( lib, "fftw3" ) +#endif + +#include + +#ifdef __cplusplus +extern "C" +{ +#endif /* __cplusplus */ + +/* If is included, use the C99 complex type. Otherwise + define a type bit-compatible with C99 complex */ +#ifdef _Complex_I +# define FFTW_DEFINE_COMPLEX(R, C) typedef R _Complex C +#else +# define FFTW_DEFINE_COMPLEX(R, C) typedef R C[2] +#endif + +#define FFTW_CONCAT(prefix, name) prefix ## name +#define FFTW_MANGLE_DOUBLE(name) FFTW_CONCAT(fftw_, name) +#define FFTW_MANGLE_FLOAT(name) FFTW_CONCAT(fftwf_, name) +#define FFTW_MANGLE_LONG_DOUBLE(name) FFTW_CONCAT(fftwl_, name) + + +enum fftw_r2r_kind_do_not_use_me { + FFTW_R2HC=0, FFTW_HC2R=1, FFTW_DHT=2, + FFTW_REDFT00=3, FFTW_REDFT01=4, FFTW_REDFT10=5, FFTW_REDFT11=6, + FFTW_RODFT00=7, FFTW_RODFT01=8, FFTW_RODFT10=9, FFTW_RODFT11=10 +}; + +struct fftw_iodim_do_not_use_me { + int n; /* dimension size */ + int is; /* input stride */ + int os; /* output stride */ +}; + +/* + huge second-order macro that defines prototypes for all API + functions. We expand this macro for each supported precision + + X: name-mangling macro + R: real data type + C: complex data type +*/ + +#define FFTW_DEFINE_API(X, R, C) \ + \ +FFTW_DEFINE_COMPLEX(R, C); \ + \ +typedef struct X(plan_s) *X(plan); \ + \ +typedef struct fftw_iodim_do_not_use_me X(iodim); \ + \ +typedef enum fftw_r2r_kind_do_not_use_me X(r2r_kind); \ + \ +void X(execute)(const X(plan) p); \ + \ +X(plan) X(plan_dft)(int rank, const int *n, \ + C *in, C *out, int sign, unsigned flags); \ + \ +X(plan) X(plan_dft_1d)(int n, C *in, C *out, int sign, \ + unsigned flags); \ +X(plan) X(plan_dft_2d)(int nx, int ny, \ + C *in, C *out, int sign, unsigned flags); \ +X(plan) X(plan_dft_3d)(int nx, int ny, int nz, \ + C *in, C *out, int sign, unsigned flags); \ + \ +X(plan) X(plan_many_dft)(int rank, const int *n, \ + int howmany, \ + C *in, const int *inembed, \ + int istride, int idist, \ + C *out, const int *onembed, \ + int ostride, int odist, \ + int sign, unsigned flags); \ + \ +X(plan) X(plan_guru_dft)(int rank, const X(iodim) *dims, \ + int howmany_rank, \ + const X(iodim) *howmany_dims, \ + C *in, C *out, \ + int sign, unsigned flags); \ +X(plan) X(plan_guru_split_dft)(int rank, const X(iodim) *dims, \ + int howmany_rank, \ + const X(iodim) *howmany_dims, \ + R *ri, R *ii, R *ro, R *io, \ + unsigned flags); \ + \ +void X(execute_dft)(const X(plan) p, C *in, C *out); \ +void X(execute_split_dft)(const X(plan) p, R *ri, R *ii, R *ro, R *io); \ + \ +X(plan) X(plan_many_dft_r2c)(int rank, const int *n, \ + int howmany, \ + R *in, const int *inembed, \ + int istride, int idist, \ + C *out, const int *onembed, \ + int ostride, int odist, \ + unsigned flags); \ + \ +X(plan) X(plan_dft_r2c)(int rank, const int *n, \ + R *in, C *out, unsigned flags); \ + \ +X(plan) X(plan_dft_r2c_1d)(int n,R *in,C *out,unsigned flags); \ +X(plan) X(plan_dft_r2c_2d)(int nx, int ny, \ + R *in, C *out, unsigned flags); \ +X(plan) X(plan_dft_r2c_3d)(int nx, int ny, \ + int nz, \ + R *in, C *out, unsigned flags); \ + \ + \ +X(plan) X(plan_many_dft_c2r)(int rank, const int *n, \ + int howmany, \ + C *in, const int *inembed, \ + int istride, int idist, \ + R *out, const int *onembed, \ + int ostride, int odist, \ + unsigned flags); \ + \ +X(plan) X(plan_dft_c2r)(int rank, const int *n, \ + C *in, R *out, unsigned flags); \ + \ +X(plan) X(plan_dft_c2r_1d)(int n,C *in,R *out,unsigned flags); \ +X(plan) X(plan_dft_c2r_2d)(int nx, int ny, \ + C *in, R *out, unsigned flags); \ +X(plan) X(plan_dft_c2r_3d)(int nx, int ny, \ + int nz, \ + C *in, R *out, unsigned flags); \ + \ +X(plan) X(plan_guru_dft_r2c)(int rank, const X(iodim) *dims, \ + int howmany_rank, \ + const X(iodim) *howmany_dims, \ + R *in, C *out, \ + unsigned flags); \ +X(plan) X(plan_guru_dft_c2r)(int rank, const X(iodim) *dims, \ + int howmany_rank, \ + const X(iodim) *howmany_dims, \ + C *in, R *out, \ + unsigned flags); \ + \ +X(plan) X(plan_guru_split_dft_r2c)(int rank, const X(iodim) *dims, \ + int howmany_rank, \ + const X(iodim) *howmany_dims, \ + R *in, R *ro, R *io, \ + unsigned flags); \ +X(plan) X(plan_guru_split_dft_c2r)(int rank, const X(iodim) *dims, \ + int howmany_rank, \ + const X(iodim) *howmany_dims, \ + R *ri, R *ii, R *out, \ + unsigned flags); \ + \ +void X(execute_dft_r2c)(const X(plan) p, R *in, C *out); \ +void X(execute_dft_c2r)(const X(plan) p, C *in, R *out); \ + \ +void X(execute_split_dft_r2c)(const X(plan) p, R *in, R *ro, R *io); \ +void X(execute_split_dft_c2r)(const X(plan) p, R *ri, R *ii, R *out); \ + \ +X(plan) X(plan_many_r2r)(int rank, const int *n, \ + int howmany, \ + R *in, const int *inembed, \ + int istride, int idist, \ + R *out, const int *onembed, \ + int ostride, int odist, \ + const X(r2r_kind) *kind, unsigned flags); \ + \ +X(plan) X(plan_r2r)(int rank, const int *n, R *in, R *out, \ + const X(r2r_kind) *kind, unsigned flags); \ + \ +X(plan) X(plan_r2r_1d)(int n, R *in, R *out, \ + X(r2r_kind) kind, unsigned flags); \ +X(plan) X(plan_r2r_2d)(int nx, int ny, R *in, R *out, \ + X(r2r_kind) kindx, X(r2r_kind) kindy, \ + unsigned flags); \ +X(plan) X(plan_r2r_3d)(int nx, int ny, int nz, \ + R *in, R *out, X(r2r_kind) kindx, \ + X(r2r_kind) kindy, X(r2r_kind) kindz, \ + unsigned flags); \ + \ +X(plan) X(plan_guru_r2r)(int rank, const X(iodim) *dims, \ + int howmany_rank, \ + const X(iodim) *howmany_dims, \ + R *in, R *out, \ + const X(r2r_kind) *kind, unsigned flags); \ +void X(execute_r2r)(const X(plan) p, R *in, R *out); \ + \ +void X(destroy_plan)(X(plan) p); \ +void X(forget_wisdom)(void); \ +void X(cleanup)(void); \ + \ +void X(plan_with_nthreads)(int nthreads); \ +int X(init_threads)(void); \ +void X(cleanup_threads)(void); \ + \ +void X(export_wisdom_to_file)(FILE *output_file); \ +char *X(export_wisdom_to_string)(void); \ +void X(export_wisdom)(void (*write_char)(char c, void *), void *data); \ +int X(import_system_wisdom)(void); \ +int X(import_wisdom_from_file)(FILE *input_file); \ +int X(import_wisdom_from_string)(const char *input_string); \ +int X(import_wisdom)(int (*read_char)(void *), void *data); \ + \ +void X(fprint_plan)(const X(plan) p, FILE *output_file); \ +void X(print_plan)(const X(plan) p); \ + \ +void *X(malloc)(size_t n); \ +void X(free)(void *p); \ + \ +void X(flops)(const X(plan) p, double *add, double *mul, double *fma); \ + \ +extern const char X(version)[]; \ +extern const char X(cc)[]; \ +extern const char X(codelet_optim)[]; + + +/* end of FFTW_DEFINE_API macro */ + +FFTW_DEFINE_API(FFTW_MANGLE_DOUBLE, double, fftw_complex) +FFTW_DEFINE_API(FFTW_MANGLE_FLOAT, float, fftwf_complex) +FFTW_DEFINE_API(FFTW_MANGLE_LONG_DOUBLE, long double, fftwl_complex) + +#define FFTW_FORWARD (-1) +#define FFTW_BACKWARD (+1) + +/* documented flags */ +#define FFTW_MEASURE (0U) +#define FFTW_DESTROY_INPUT (1U << 0) +#define FFTW_UNALIGNED (1U << 1) +#define FFTW_CONSERVE_MEMORY (1U << 2) +#define FFTW_EXHAUSTIVE (1U << 3) /* NO_EXHAUSTIVE is default */ +#define FFTW_PRESERVE_INPUT (1U << 4) /* cancels FFTW_DESTROY_INPUT */ +#define FFTW_PATIENT (1U << 5) /* IMPATIENT is default */ +#define FFTW_ESTIMATE (1U << 6) + +/* undocumented beyond-guru flags */ +#define FFTW_ESTIMATE_PATIENT (1U << 7) +#define FFTW_BELIEVE_PCOST (1U << 8) +#define FFTW_DFT_R2HC_ICKY (1U << 9) +#define FFTW_NONTHREADED_ICKY (1U << 10) +#define FFTW_NO_BUFFERING (1U << 11) +#define FFTW_NO_INDIRECT_OP (1U << 12) +#define FFTW_ALLOW_LARGE_GENERIC (1U << 13) /* NO_LARGE_GENERIC is default */ +#define FFTW_NO_RANK_SPLITS (1U << 14) +#define FFTW_NO_VRANK_SPLITS (1U << 15) +#define FFTW_NO_VRECURSE (1U << 16) + +#define FFTW_NO_SIMD (1U << 17) + +#ifdef __cplusplus +} /* extern "C" */ +#endif /* __cplusplus */ + +#endif /* FFTW3_H */ diff --git a/Externals/MusicMod/Player/Src/vis_plainbar/vis_plainbar.cpp b/Externals/MusicMod/Player/Src/vis_plainbar/vis_plainbar.cpp new file mode 100644 index 0000000000..276065e516 --- /dev/null +++ b/Externals/MusicMod/Player/Src/vis_plainbar/vis_plainbar.cpp @@ -0,0 +1,259 @@ +//////////////////////////////////////////////////////////////////////////////// +// Plainamp Toolbar Vis Plugin +// +// Copyright İ 2006 Sebastian Pipping +// +// --> http://www.hartwork.org +// +// This source code is released under the GNU General Public License (GPL). +// See GPL.txt for details. Any non-GPL usage is strictly forbidden. +//////////////////////////////////////////////////////////////////////////////// + + +#define WIN32_LEAN_AND_MEAN +#include + + +#include "../Winamp/vis.h" +#include "../Winamp/wa_ipc.h" + + + +#define PLUGIN_NAME "Plainamp Toolbar Vis Plugin" +#define PLUGIN_VERSION "v0.5" + +#define PLUGIN_DESC PLUGIN_NAME " " PLUGIN_VERSION + + + +static char * szClassName = "PlainbarClass"; + + + +winampVisModule * getModule( int which ); + + +void config( struct winampVisModule * this_mod ); +int init( struct winampVisModule * this_mod ); +int render_spec( struct winampVisModule * this_mod ); +void quit( struct winampVisModule * this_mod ); + + +// Double buffering data +HDC memDC = NULL; // Memory device context +HBITMAP memBM = NULL; // Memory bitmap (for memDC) +HBITMAP oldBM = NULL; // Old bitmap (from memDC) + + +HWND hRender = NULL; +int width = 0; +int height = 0; +bool bRunning = false; +HPEN pen = NULL; + + +WNDPROC WndprocTargetBackup = NULL; +LRESULT CALLBACK WndprocTarget( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ); + + + +winampVisHeader hdr = { + VIS_HDRVER, + PLUGIN_DESC, + getModule +}; + + + +winampVisModule mod_spec = +{ + "Default", + NULL, // hwndParent + NULL, // hDllInstance + 0, // sRate + 0, // nCh + 25, // latencyMS + 25, // delayMS + 2, // spectrumNch + 0, // waveformNch + { 0, }, // spectrumData + { 0, }, // waveformData + config, + init, + render_spec, + quit +}; + + + +#ifdef __cplusplus +extern "C" { +#endif +__declspec( dllexport ) winampVisHeader * winampVisGetHeader() +{ + return &hdr; +} +#ifdef __cplusplus +} +#endif + + + +winampVisModule * getModule( int which ) +{ + return which ? NULL : &mod_spec; +} + + + +void config( struct winampVisModule * this_mod ) +{ + MessageBox( + this_mod->hwndParent, + PLUGIN_DESC "\n" + "\n" + "Copyright İ 2006 Sebastian Pipping \n" + "\n" + "\n" + "--> http://www.hartwork.org", + "About", + MB_ICONINFORMATION + ); +} + + + +LRESULT CALLBACK WndprocTarget( HWND hwnd, UINT message, WPARAM wp, LPARAM lp ) +{ + switch( message ) + { + case WM_SIZE: + width = LOWORD( lp ); + height = HIWORD( lp ); + break; + + case WM_DESTROY: + bRunning = false; + PostQuitMessage( 0 ); + return 0; + + } + + return DefWindowProc( hwnd, message, wp, lp ); +} + + + +int init( struct winampVisModule * this_mod ) +{ + if( !this_mod ) return 1; + + // Register message + const int IPC_GETPLAINBARTARGET = ( int )SendMessage( this_mod->hwndParent, WM_WA_IPC, ( WPARAM )"IPC_GETPLAINBARTARGET", IPC_REGISTER_WINAMP_IPCMESSAGE ); + if( IPC_GETPLAINBARTARGET <= 0 ) return 1; + + // Get render parent + HWND hRenderParent = ( HWND )SendMessage( this_mod->hwndParent, WM_WA_IPC, 0, IPC_GETPLAINBARTARGET ); + if( !IsWindow( hRenderParent ) ) return 1; + + // Plug our child in + WNDCLASS wc; + ZeroMemory( &wc, sizeof( WNDCLASS ) ); + wc.lpfnWndProc = WndprocTarget; + wc.hInstance = this_mod->hDllInstance; + wc.hCursor = LoadCursor( NULL, IDC_ARROW ); + wc.lpszClassName = szClassName; + + if( !RegisterClass( &wc ) ) return 1; + + RECT r; + GetClientRect( hRenderParent, &r ); + width = r.right - r.left; + height = r.bottom - r.top; + + hRender = CreateWindowEx( + 0, + szClassName, + "", + WS_CHILD | WS_VISIBLE, + 0, + 0, + width, + height, + hRenderParent, + NULL, + this_mod->hDllInstance, + 0 + ); + + if( !hRender ) + { + UnregisterClass( szClassName, this_mod->hDllInstance ); + return 1; + } + + // Create doublebuffer + const HDC hdc = GetDC( hRender ); + memDC = CreateCompatibleDC( hdc ); + memBM = CreateCompatibleBitmap( hdc, 576, 256 ); + oldBM = ( HBITMAP )SelectObject( memDC, memBM ); + ReleaseDC( hRender, hdc ); + + pen = CreatePen( PS_SOLID, 0, GetSysColor( COLOR_APPWORKSPACE ) ); + + bRunning = true; + return 0; +} + + + +int render_spec( struct winampVisModule * this_mod ) +{ + // Clear background + RECT rect = { 0, 0, 576, 256 }; + FillRect(memDC, &rect, ( HBRUSH )( COLOR_3DFACE + 1 ) ); + + // Draw analyser + SelectObject( memDC, pen ); + for( int x = 0; x < 576; x++ ) + { + int val = 0; + + for( int y = 0; y < this_mod->nCh; y++ ) + { + if( this_mod->spectrumData[ y ][ x ] > val ) + { + val = this_mod->spectrumData[ y ][ x ]; + } + } + + MoveToEx( memDC, x, 256, NULL ); + LineTo( memDC, x, 256 - val ); + } + + // Copy doublebuffer to window + HDC hdc = GetDC( hRender ); + StretchBlt( hdc, 0, 0, width, height, memDC, 0, 0, 576, 256, SRCCOPY ); + ReleaseDC( hRender, hdc ); + + return bRunning ? 0 : 1; +} + + + +void quit( struct winampVisModule * this_mod ) +{ + if( bRunning ) + { + DestroyWindow( hRender ); + } + + UnregisterClass( szClassName, this_mod->hDllInstance ); + + // Delete doublebuffer + SelectObject( memDC, oldBM ); + DeleteObject( memDC ); + DeleteObject( memBM ); + + DeleteObject( pen ); +} diff --git a/Externals/MusicMod/Player/Src/zlib/zconf.h b/Externals/MusicMod/Player/Src/zlib/zconf.h new file mode 100644 index 0000000000..db9ff5e7e3 --- /dev/null +++ b/Externals/MusicMod/Player/Src/zlib/zconf.h @@ -0,0 +1,332 @@ +/* zconf.h -- configuration of the zlib compression library + * Copyright (C) 1995-2005 Jean-loup Gailly. + * For conditions of distribution and use, see copyright notice in zlib.h + */ + +/* @(#) $Id: zconf.h,v 1.1 2005/10/10 21:08:30 hartwork Exp $ */ + +#ifndef ZCONF_H +#define ZCONF_H + +/* + * If you *really* need a unique prefix for all types and library functions, + * compile with -DZ_PREFIX. The "standard" zlib should be compiled without it. + */ +#ifdef Z_PREFIX +# define deflateInit_ z_deflateInit_ +# define deflate z_deflate +# define deflateEnd z_deflateEnd +# define inflateInit_ z_inflateInit_ +# define inflate z_inflate +# define inflateEnd z_inflateEnd +# define deflateInit2_ z_deflateInit2_ +# define deflateSetDictionary z_deflateSetDictionary +# define deflateCopy z_deflateCopy +# define deflateReset z_deflateReset +# define deflateParams z_deflateParams +# define deflateBound z_deflateBound +# define deflatePrime z_deflatePrime +# define inflateInit2_ z_inflateInit2_ +# define inflateSetDictionary z_inflateSetDictionary +# define inflateSync z_inflateSync +# define inflateSyncPoint z_inflateSyncPoint +# define inflateCopy z_inflateCopy +# define inflateReset z_inflateReset +# define inflateBack z_inflateBack +# define inflateBackEnd z_inflateBackEnd +# define compress z_compress +# define compress2 z_compress2 +# define compressBound z_compressBound +# define uncompress z_uncompress +# define adler32 z_adler32 +# define crc32 z_crc32 +# define get_crc_table z_get_crc_table +# define zError z_zError + +# define alloc_func z_alloc_func +# define free_func z_free_func +# define in_func z_in_func +# define out_func z_out_func +# define Byte z_Byte +# define uInt z_uInt +# define uLong z_uLong +# define Bytef z_Bytef +# define charf z_charf +# define intf z_intf +# define uIntf z_uIntf +# define uLongf z_uLongf +# define voidpf z_voidpf +# define voidp z_voidp +#endif + +#if defined(__MSDOS__) && !defined(MSDOS) +# define MSDOS +#endif +#if (defined(OS_2) || defined(__OS2__)) && !defined(OS2) +# define OS2 +#endif +#if defined(_WINDOWS) && !defined(WINDOWS) +# define WINDOWS +#endif +#if defined(_WIN32) || defined(_WIN32_WCE) || defined(__WIN32__) +# ifndef WIN32 +# define WIN32 +# endif +#endif +#if (defined(MSDOS) || defined(OS2) || defined(WINDOWS)) && !defined(WIN32) +# if !defined(__GNUC__) && !defined(__FLAT__) && !defined(__386__) +# ifndef SYS16BIT +# define SYS16BIT +# endif +# endif +#endif + +/* + * Compile with -DMAXSEG_64K if the alloc function cannot allocate more + * than 64k bytes at a time (needed on systems with 16-bit int). + */ +#ifdef SYS16BIT +# define MAXSEG_64K +#endif +#ifdef MSDOS +# define UNALIGNED_OK +#endif + +#ifdef __STDC_VERSION__ +# ifndef STDC +# define STDC +# endif +# if __STDC_VERSION__ >= 199901L +# ifndef STDC99 +# define STDC99 +# endif +# endif +#endif +#if !defined(STDC) && (defined(__STDC__) || defined(__cplusplus)) +# define STDC +#endif +#if !defined(STDC) && (defined(__GNUC__) || defined(__BORLANDC__)) +# define STDC +#endif +#if !defined(STDC) && (defined(MSDOS) || defined(WINDOWS) || defined(WIN32)) +# define STDC +#endif +#if !defined(STDC) && (defined(OS2) || defined(__HOS_AIX__)) +# define STDC +#endif + +#if defined(__OS400__) && !defined(STDC) /* iSeries (formerly AS/400). */ +# define STDC +#endif + +#ifndef STDC +# ifndef const /* cannot use !defined(STDC) && !defined(const) on Mac */ +# define const /* note: need a more gentle solution here */ +# endif +#endif + +/* Some Mac compilers merge all .h files incorrectly: */ +#if defined(__MWERKS__)||defined(applec)||defined(THINK_C)||defined(__SC__) +# define NO_DUMMY_DECL +#endif + +/* Maximum value for memLevel in deflateInit2 */ +#ifndef MAX_MEM_LEVEL +# ifdef MAXSEG_64K +# define MAX_MEM_LEVEL 8 +# else +# define MAX_MEM_LEVEL 9 +# endif +#endif + +/* Maximum value for windowBits in deflateInit2 and inflateInit2. + * WARNING: reducing MAX_WBITS makes minigzip unable to extract .gz files + * created by gzip. (Files created by minigzip can still be extracted by + * gzip.) + */ +#ifndef MAX_WBITS +# define MAX_WBITS 15 /* 32K LZ77 window */ +#endif + +/* The memory requirements for deflate are (in bytes): + (1 << (windowBits+2)) + (1 << (memLevel+9)) + that is: 128K for windowBits=15 + 128K for memLevel = 8 (default values) + plus a few kilobytes for small objects. For example, if you want to reduce + the default memory requirements from 256K to 128K, compile with + make CFLAGS="-O -DMAX_WBITS=14 -DMAX_MEM_LEVEL=7" + Of course this will generally degrade compression (there's no free lunch). + + The memory requirements for inflate are (in bytes) 1 << windowBits + that is, 32K for windowBits=15 (default value) plus a few kilobytes + for small objects. +*/ + + /* Type declarations */ + +#ifndef OF /* function prototypes */ +# ifdef STDC +# define OF(args) args +# else +# define OF(args) () +# endif +#endif + +/* The following definitions for FAR are needed only for MSDOS mixed + * model programming (small or medium model with some far allocations). + * This was tested only with MSC; for other MSDOS compilers you may have + * to define NO_MEMCPY in zutil.h. If you don't need the mixed model, + * just define FAR to be empty. + */ +#ifdef SYS16BIT +# if defined(M_I86SM) || defined(M_I86MM) + /* MSC small or medium model */ +# define SMALL_MEDIUM +# ifdef _MSC_VER +# define FAR _far +# else +# define FAR far +# endif +# endif +# if (defined(__SMALL__) || defined(__MEDIUM__)) + /* Turbo C small or medium model */ +# define SMALL_MEDIUM +# ifdef __BORLANDC__ +# define FAR _far +# else +# define FAR far +# endif +# endif +#endif + +#if defined(WINDOWS) || defined(WIN32) + /* If building or using zlib as a DLL, define ZLIB_DLL. + * This is not mandatory, but it offers a little performance increase. + */ +# ifdef ZLIB_DLL +# if defined(WIN32) && (!defined(__BORLANDC__) || (__BORLANDC__ >= 0x500)) +# ifdef ZLIB_INTERNAL +# define ZEXTERN extern __declspec(dllexport) +# else +# define ZEXTERN extern __declspec(dllimport) +# endif +# endif +# endif /* ZLIB_DLL */ + /* If building or using zlib with the WINAPI/WINAPIV calling convention, + * define ZLIB_WINAPI. + * Caution: the standard ZLIB1.DLL is NOT compiled using ZLIB_WINAPI. + */ +# ifdef ZLIB_WINAPI +# ifdef FAR +# undef FAR +# endif +# include + /* No need for _export, use ZLIB.DEF instead. */ + /* For complete Windows compatibility, use WINAPI, not __stdcall. */ +# define ZEXPORT WINAPI +# ifdef WIN32 +# define ZEXPORTVA WINAPIV +# else +# define ZEXPORTVA FAR CDECL +# endif +# endif +#endif + +#if defined (__BEOS__) +# ifdef ZLIB_DLL +# ifdef ZLIB_INTERNAL +# define ZEXPORT __declspec(dllexport) +# define ZEXPORTVA __declspec(dllexport) +# else +# define ZEXPORT __declspec(dllimport) +# define ZEXPORTVA __declspec(dllimport) +# endif +# endif +#endif + +#ifndef ZEXTERN +# define ZEXTERN extern +#endif +#ifndef ZEXPORT +# define ZEXPORT +#endif +#ifndef ZEXPORTVA +# define ZEXPORTVA +#endif + +#ifndef FAR +# define FAR +#endif + +#if !defined(__MACTYPES__) +typedef unsigned char Byte; /* 8 bits */ +#endif +typedef unsigned int uInt; /* 16 bits or more */ +typedef unsigned long uLong; /* 32 bits or more */ + +#ifdef SMALL_MEDIUM + /* Borland C/C++ and some old MSC versions ignore FAR inside typedef */ +# define Bytef Byte FAR +#else + typedef Byte FAR Bytef; +#endif +typedef char FAR charf; +typedef int FAR intf; +typedef uInt FAR uIntf; +typedef uLong FAR uLongf; + +#ifdef STDC + typedef void const *voidpc; + typedef void FAR *voidpf; + typedef void *voidp; +#else + typedef Byte const *voidpc; + typedef Byte FAR *voidpf; + typedef Byte *voidp; +#endif + +#if 0 /* HAVE_UNISTD_H -- this line is updated by ./configure */ +# include /* for off_t */ +# include /* for SEEK_* and off_t */ +# ifdef VMS +# include /* for off_t */ +# endif +# define z_off_t off_t +#endif +#ifndef SEEK_SET +# define SEEK_SET 0 /* Seek from beginning of file. */ +# define SEEK_CUR 1 /* Seek from current position. */ +# define SEEK_END 2 /* Set file pointer to EOF plus "offset" */ +#endif +#ifndef z_off_t +# define z_off_t long +#endif + +#if defined(__OS400__) +# define NO_vsnprintf +#endif + +#if defined(__MVS__) +# define NO_vsnprintf +# ifdef FAR +# undef FAR +# endif +#endif + +/* MVS linker does not support external names larger than 8 bytes */ +#if defined(__MVS__) +# pragma map(deflateInit_,"DEIN") +# pragma map(deflateInit2_,"DEIN2") +# pragma map(deflateEnd,"DEEND") +# pragma map(deflateBound,"DEBND") +# pragma map(inflateInit_,"ININ") +# pragma map(inflateInit2_,"ININ2") +# pragma map(inflateEnd,"INEND") +# pragma map(inflateSync,"INSY") +# pragma map(inflateSetDictionary,"INSEDI") +# pragma map(compressBound,"CMBND") +# pragma map(inflate_table,"INTABL") +# pragma map(inflate_fast,"INFA") +# pragma map(inflate_copyright,"INCOPY") +#endif + +#endif /* ZCONF_H */ diff --git a/Externals/MusicMod/Player/Src/zlib/zlib.h b/Externals/MusicMod/Player/Src/zlib/zlib.h new file mode 100644 index 0000000000..62d0e4675b --- /dev/null +++ b/Externals/MusicMod/Player/Src/zlib/zlib.h @@ -0,0 +1,1357 @@ +/* zlib.h -- interface of the 'zlib' general purpose compression library + version 1.2.3, July 18th, 2005 + + Copyright (C) 1995-2005 Jean-loup Gailly and Mark Adler + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. + + Jean-loup Gailly Mark Adler + jloup@gzip.org madler@alumni.caltech.edu + + + The data format used by the zlib library is described by RFCs (Request for + Comments) 1950 to 1952 in the files http://www.ietf.org/rfc/rfc1950.txt + (zlib format), rfc1951.txt (deflate format) and rfc1952.txt (gzip format). +*/ + +#ifndef ZLIB_H +#define ZLIB_H + +#include "zconf.h" + +#ifdef __cplusplus +extern "C" { +#endif + +#define ZLIB_VERSION "1.2.3" +#define ZLIB_VERNUM 0x1230 + +/* + The 'zlib' compression library provides in-memory compression and + decompression functions, including integrity checks of the uncompressed + data. This version of the library supports only one compression method + (deflation) but other algorithms will be added later and will have the same + stream interface. + + Compression can be done in a single step if the buffers are large + enough (for example if an input file is mmap'ed), or can be done by + repeated calls of the compression function. In the latter case, the + application must provide more input and/or consume the output + (providing more output space) before each call. + + The compressed data format used by default by the in-memory functions is + the zlib format, which is a zlib wrapper documented in RFC 1950, wrapped + around a deflate stream, which is itself documented in RFC 1951. + + The library also supports reading and writing files in gzip (.gz) format + with an interface similar to that of stdio using the functions that start + with "gz". The gzip format is different from the zlib format. gzip is a + gzip wrapper, documented in RFC 1952, wrapped around a deflate stream. + + This library can optionally read and write gzip streams in memory as well. + + The zlib format was designed to be compact and fast for use in memory + and on communications channels. The gzip format was designed for single- + file compression on file systems, has a larger header than zlib to maintain + directory information, and uses a different, slower check method than zlib. + + The library does not install any signal handler. The decoder checks + the consistency of the compressed data, so the library should never + crash even in case of corrupted input. +*/ + +typedef voidpf (*alloc_func) OF((voidpf opaque, uInt items, uInt size)); +typedef void (*free_func) OF((voidpf opaque, voidpf address)); + +struct internal_state; + +typedef struct z_stream_s { + Bytef *next_in; /* next input byte */ + uInt avail_in; /* number of bytes available at next_in */ + uLong total_in; /* total nb of input bytes read so far */ + + Bytef *next_out; /* next output byte should be put there */ + uInt avail_out; /* remaining free space at next_out */ + uLong total_out; /* total nb of bytes output so far */ + + char *msg; /* last error message, NULL if no error */ + struct internal_state FAR *state; /* not visible by applications */ + + alloc_func zalloc; /* used to allocate the internal state */ + free_func zfree; /* used to free the internal state */ + voidpf opaque; /* private data object passed to zalloc and zfree */ + + int data_type; /* best guess about the data type: binary or text */ + uLong adler; /* adler32 value of the uncompressed data */ + uLong reserved; /* reserved for future use */ +} z_stream; + +typedef z_stream FAR *z_streamp; + +/* + gzip header information passed to and from zlib routines. See RFC 1952 + for more details on the meanings of these fields. +*/ +typedef struct gz_header_s { + int text; /* true if compressed data believed to be text */ + uLong time; /* modification time */ + int xflags; /* extra flags (not used when writing a gzip file) */ + int os; /* operating system */ + Bytef *extra; /* pointer to extra field or Z_NULL if none */ + uInt extra_len; /* extra field length (valid if extra != Z_NULL) */ + uInt extra_max; /* space at extra (only when reading header) */ + Bytef *name; /* pointer to zero-terminated file name or Z_NULL */ + uInt name_max; /* space at name (only when reading header) */ + Bytef *comment; /* pointer to zero-terminated comment or Z_NULL */ + uInt comm_max; /* space at comment (only when reading header) */ + int hcrc; /* true if there was or will be a header crc */ + int done; /* true when done reading gzip header (not used + when writing a gzip file) */ +} gz_header; + +typedef gz_header FAR *gz_headerp; + +/* + The application must update next_in and avail_in when avail_in has + dropped to zero. It must update next_out and avail_out when avail_out + has dropped to zero. The application must initialize zalloc, zfree and + opaque before calling the init function. All other fields are set by the + compression library and must not be updated by the application. + + The opaque value provided by the application will be passed as the first + parameter for calls of zalloc and zfree. This can be useful for custom + memory management. The compression library attaches no meaning to the + opaque value. + + zalloc must return Z_NULL if there is not enough memory for the object. + If zlib is used in a multi-threaded application, zalloc and zfree must be + thread safe. + + On 16-bit systems, the functions zalloc and zfree must be able to allocate + exactly 65536 bytes, but will not be required to allocate more than this + if the symbol MAXSEG_64K is defined (see zconf.h). WARNING: On MSDOS, + pointers returned by zalloc for objects of exactly 65536 bytes *must* + have their offset normalized to zero. The default allocation function + provided by this library ensures this (see zutil.c). To reduce memory + requirements and avoid any allocation of 64K objects, at the expense of + compression ratio, compile the library with -DMAX_WBITS=14 (see zconf.h). + + The fields total_in and total_out can be used for statistics or + progress reports. After compression, total_in holds the total size of + the uncompressed data and may be saved for use in the decompressor + (particularly if the decompressor wants to decompress everything in + a single step). +*/ + + /* constants */ + +#define Z_NO_FLUSH 0 +#define Z_PARTIAL_FLUSH 1 /* will be removed, use Z_SYNC_FLUSH instead */ +#define Z_SYNC_FLUSH 2 +#define Z_FULL_FLUSH 3 +#define Z_FINISH 4 +#define Z_BLOCK 5 +/* Allowed flush values; see deflate() and inflate() below for details */ + +#define Z_OK 0 +#define Z_STREAM_END 1 +#define Z_NEED_DICT 2 +#define Z_ERRNO (-1) +#define Z_STREAM_ERROR (-2) +#define Z_DATA_ERROR (-3) +#define Z_MEM_ERROR (-4) +#define Z_BUF_ERROR (-5) +#define Z_VERSION_ERROR (-6) +/* Return codes for the compression/decompression functions. Negative + * values are errors, positive values are used for special but normal events. + */ + +#define Z_NO_COMPRESSION 0 +#define Z_BEST_SPEED 1 +#define Z_BEST_COMPRESSION 9 +#define Z_DEFAULT_COMPRESSION (-1) +/* compression levels */ + +#define Z_FILTERED 1 +#define Z_HUFFMAN_ONLY 2 +#define Z_RLE 3 +#define Z_FIXED 4 +#define Z_DEFAULT_STRATEGY 0 +/* compression strategy; see deflateInit2() below for details */ + +#define Z_BINARY 0 +#define Z_TEXT 1 +#define Z_ASCII Z_TEXT /* for compatibility with 1.2.2 and earlier */ +#define Z_UNKNOWN 2 +/* Possible values of the data_type field (though see inflate()) */ + +#define Z_DEFLATED 8 +/* The deflate compression method (the only one supported in this version) */ + +#define Z_NULL 0 /* for initializing zalloc, zfree, opaque */ + +#define zlib_version zlibVersion() +/* for compatibility with versions < 1.0.2 */ + + /* basic functions */ + +ZEXTERN const char * ZEXPORT zlibVersion OF((void)); +/* The application can compare zlibVersion and ZLIB_VERSION for consistency. + If the first character differs, the library code actually used is + not compatible with the zlib.h header file used by the application. + This check is automatically made by deflateInit and inflateInit. + */ + +/* +ZEXTERN int ZEXPORT deflateInit OF((z_streamp strm, int level)); + + Initializes the internal stream state for compression. The fields + zalloc, zfree and opaque must be initialized before by the caller. + If zalloc and zfree are set to Z_NULL, deflateInit updates them to + use default allocation functions. + + The compression level must be Z_DEFAULT_COMPRESSION, or between 0 and 9: + 1 gives best speed, 9 gives best compression, 0 gives no compression at + all (the input data is simply copied a block at a time). + Z_DEFAULT_COMPRESSION requests a default compromise between speed and + compression (currently equivalent to level 6). + + deflateInit returns Z_OK if success, Z_MEM_ERROR if there was not + enough memory, Z_STREAM_ERROR if level is not a valid compression level, + Z_VERSION_ERROR if the zlib library version (zlib_version) is incompatible + with the version assumed by the caller (ZLIB_VERSION). + msg is set to null if there is no error message. deflateInit does not + perform any compression: this will be done by deflate(). +*/ + + +ZEXTERN int ZEXPORT deflate OF((z_streamp strm, int flush)); +/* + deflate compresses as much data as possible, and stops when the input + buffer becomes empty or the output buffer becomes full. It may introduce some + output latency (reading input without producing any output) except when + forced to flush. + + The detailed semantics are as follows. deflate performs one or both of the + following actions: + + - Compress more input starting at next_in and update next_in and avail_in + accordingly. If not all input can be processed (because there is not + enough room in the output buffer), next_in and avail_in are updated and + processing will resume at this point for the next call of deflate(). + + - Provide more output starting at next_out and update next_out and avail_out + accordingly. This action is forced if the parameter flush is non zero. + Forcing flush frequently degrades the compression ratio, so this parameter + should be set only when necessary (in interactive applications). + Some output may be provided even if flush is not set. + + Before the call of deflate(), the application should ensure that at least + one of the actions is possible, by providing more input and/or consuming + more output, and updating avail_in or avail_out accordingly; avail_out + should never be zero before the call. The application can consume the + compressed output when it wants, for example when the output buffer is full + (avail_out == 0), or after each call of deflate(). If deflate returns Z_OK + and with zero avail_out, it must be called again after making room in the + output buffer because there might be more output pending. + + Normally the parameter flush is set to Z_NO_FLUSH, which allows deflate to + decide how much data to accumualte before producing output, in order to + maximize compression. + + If the parameter flush is set to Z_SYNC_FLUSH, all pending output is + flushed to the output buffer and the output is aligned on a byte boundary, so + that the decompressor can get all input data available so far. (In particular + avail_in is zero after the call if enough output space has been provided + before the call.) Flushing may degrade compression for some compression + algorithms and so it should be used only when necessary. + + If flush is set to Z_FULL_FLUSH, all output is flushed as with + Z_SYNC_FLUSH, and the compression state is reset so that decompression can + restart from this point if previous compressed data has been damaged or if + random access is desired. Using Z_FULL_FLUSH too often can seriously degrade + compression. + + If deflate returns with avail_out == 0, this function must be called again + with the same value of the flush parameter and more output space (updated + avail_out), until the flush is complete (deflate returns with non-zero + avail_out). In the case of a Z_FULL_FLUSH or Z_SYNC_FLUSH, make sure that + avail_out is greater than six to avoid repeated flush markers due to + avail_out == 0 on return. + + If the parameter flush is set to Z_FINISH, pending input is processed, + pending output is flushed and deflate returns with Z_STREAM_END if there + was enough output space; if deflate returns with Z_OK, this function must be + called again with Z_FINISH and more output space (updated avail_out) but no + more input data, until it returns with Z_STREAM_END or an error. After + deflate has returned Z_STREAM_END, the only possible operations on the + stream are deflateReset or deflateEnd. + + Z_FINISH can be used immediately after deflateInit if all the compression + is to be done in a single step. In this case, avail_out must be at least + the value returned by deflateBound (see below). If deflate does not return + Z_STREAM_END, then it must be called again as described above. + + deflate() sets strm->adler to the adler32 checksum of all input read + so far (that is, total_in bytes). + + deflate() may update strm->data_type if it can make a good guess about + the input data type (Z_BINARY or Z_TEXT). In doubt, the data is considered + binary. This field is only for information purposes and does not affect + the compression algorithm in any manner. + + deflate() returns Z_OK if some progress has been made (more input + processed or more output produced), Z_STREAM_END if all input has been + consumed and all output has been produced (only when flush is set to + Z_FINISH), Z_STREAM_ERROR if the stream state was inconsistent (for example + if next_in or next_out was NULL), Z_BUF_ERROR if no progress is possible + (for example avail_in or avail_out was zero). Note that Z_BUF_ERROR is not + fatal, and deflate() can be called again with more input and more output + space to continue compressing. +*/ + + +ZEXTERN int ZEXPORT deflateEnd OF((z_streamp strm)); +/* + All dynamically allocated data structures for this stream are freed. + This function discards any unprocessed input and does not flush any + pending output. + + deflateEnd returns Z_OK if success, Z_STREAM_ERROR if the + stream state was inconsistent, Z_DATA_ERROR if the stream was freed + prematurely (some input or output was discarded). In the error case, + msg may be set but then points to a static string (which must not be + deallocated). +*/ + + +/* +ZEXTERN int ZEXPORT inflateInit OF((z_streamp strm)); + + Initializes the internal stream state for decompression. The fields + next_in, avail_in, zalloc, zfree and opaque must be initialized before by + the caller. If next_in is not Z_NULL and avail_in is large enough (the exact + value depends on the compression method), inflateInit determines the + compression method from the zlib header and allocates all data structures + accordingly; otherwise the allocation will be deferred to the first call of + inflate. If zalloc and zfree are set to Z_NULL, inflateInit updates them to + use default allocation functions. + + inflateInit returns Z_OK if success, Z_MEM_ERROR if there was not enough + memory, Z_VERSION_ERROR if the zlib library version is incompatible with the + version assumed by the caller. msg is set to null if there is no error + message. inflateInit does not perform any decompression apart from reading + the zlib header if present: this will be done by inflate(). (So next_in and + avail_in may be modified, but next_out and avail_out are unchanged.) +*/ + + +ZEXTERN int ZEXPORT inflate OF((z_streamp strm, int flush)); +/* + inflate decompresses as much data as possible, and stops when the input + buffer becomes empty or the output buffer becomes full. It may introduce + some output latency (reading input without producing any output) except when + forced to flush. + + The detailed semantics are as follows. inflate performs one or both of the + following actions: + + - Decompress more input starting at next_in and update next_in and avail_in + accordingly. If not all input can be processed (because there is not + enough room in the output buffer), next_in is updated and processing + will resume at this point for the next call of inflate(). + + - Provide more output starting at next_out and update next_out and avail_out + accordingly. inflate() provides as much output as possible, until there + is no more input data or no more space in the output buffer (see below + about the flush parameter). + + Before the call of inflate(), the application should ensure that at least + one of the actions is possible, by providing more input and/or consuming + more output, and updating the next_* and avail_* values accordingly. + The application can consume the uncompressed output when it wants, for + example when the output buffer is full (avail_out == 0), or after each + call of inflate(). If inflate returns Z_OK and with zero avail_out, it + must be called again after making room in the output buffer because there + might be more output pending. + + The flush parameter of inflate() can be Z_NO_FLUSH, Z_SYNC_FLUSH, + Z_FINISH, or Z_BLOCK. Z_SYNC_FLUSH requests that inflate() flush as much + output as possible to the output buffer. Z_BLOCK requests that inflate() stop + if and when it gets to the next deflate block boundary. When decoding the + zlib or gzip format, this will cause inflate() to return immediately after + the header and before the first block. When doing a raw inflate, inflate() + will go ahead and process the first block, and will return when it gets to + the end of that block, or when it runs out of data. + + The Z_BLOCK option assists in appending to or combining deflate streams. + Also to assist in this, on return inflate() will set strm->data_type to the + number of unused bits in the last byte taken from strm->next_in, plus 64 + if inflate() is currently decoding the last block in the deflate stream, + plus 128 if inflate() returned immediately after decoding an end-of-block + code or decoding the complete header up to just before the first byte of the + deflate stream. The end-of-block will not be indicated until all of the + uncompressed data from that block has been written to strm->next_out. The + number of unused bits may in general be greater than seven, except when + bit 7 of data_type is set, in which case the number of unused bits will be + less than eight. + + inflate() should normally be called until it returns Z_STREAM_END or an + error. However if all decompression is to be performed in a single step + (a single call of inflate), the parameter flush should be set to + Z_FINISH. In this case all pending input is processed and all pending + output is flushed; avail_out must be large enough to hold all the + uncompressed data. (The size of the uncompressed data may have been saved + by the compressor for this purpose.) The next operation on this stream must + be inflateEnd to deallocate the decompression state. The use of Z_FINISH + is never required, but can be used to inform inflate that a faster approach + may be used for the single inflate() call. + + In this implementation, inflate() always flushes as much output as + possible to the output buffer, and always uses the faster approach on the + first call. So the only effect of the flush parameter in this implementation + is on the return value of inflate(), as noted below, or when it returns early + because Z_BLOCK is used. + + If a preset dictionary is needed after this call (see inflateSetDictionary + below), inflate sets strm->adler to the adler32 checksum of the dictionary + chosen by the compressor and returns Z_NEED_DICT; otherwise it sets + strm->adler to the adler32 checksum of all output produced so far (that is, + total_out bytes) and returns Z_OK, Z_STREAM_END or an error code as described + below. At the end of the stream, inflate() checks that its computed adler32 + checksum is equal to that saved by the compressor and returns Z_STREAM_END + only if the checksum is correct. + + inflate() will decompress and check either zlib-wrapped or gzip-wrapped + deflate data. The header type is detected automatically. Any information + contained in the gzip header is not retained, so applications that need that + information should instead use raw inflate, see inflateInit2() below, or + inflateBack() and perform their own processing of the gzip header and + trailer. + + inflate() returns Z_OK if some progress has been made (more input processed + or more output produced), Z_STREAM_END if the end of the compressed data has + been reached and all uncompressed output has been produced, Z_NEED_DICT if a + preset dictionary is needed at this point, Z_DATA_ERROR if the input data was + corrupted (input stream not conforming to the zlib format or incorrect check + value), Z_STREAM_ERROR if the stream structure was inconsistent (for example + if next_in or next_out was NULL), Z_MEM_ERROR if there was not enough memory, + Z_BUF_ERROR if no progress is possible or if there was not enough room in the + output buffer when Z_FINISH is used. Note that Z_BUF_ERROR is not fatal, and + inflate() can be called again with more input and more output space to + continue decompressing. If Z_DATA_ERROR is returned, the application may then + call inflateSync() to look for a good compression block if a partial recovery + of the data is desired. +*/ + + +ZEXTERN int ZEXPORT inflateEnd OF((z_streamp strm)); +/* + All dynamically allocated data structures for this stream are freed. + This function discards any unprocessed input and does not flush any + pending output. + + inflateEnd returns Z_OK if success, Z_STREAM_ERROR if the stream state + was inconsistent. In the error case, msg may be set but then points to a + static string (which must not be deallocated). +*/ + + /* Advanced functions */ + +/* + The following functions are needed only in some special applications. +*/ + +/* +ZEXTERN int ZEXPORT deflateInit2 OF((z_streamp strm, + int level, + int method, + int windowBits, + int memLevel, + int strategy)); + + This is another version of deflateInit with more compression options. The + fields next_in, zalloc, zfree and opaque must be initialized before by + the caller. + + The method parameter is the compression method. It must be Z_DEFLATED in + this version of the library. + + The windowBits parameter is the base two logarithm of the window size + (the size of the history buffer). It should be in the range 8..15 for this + version of the library. Larger values of this parameter result in better + compression at the expense of memory usage. The default value is 15 if + deflateInit is used instead. + + windowBits can also be -8..-15 for raw deflate. In this case, -windowBits + determines the window size. deflate() will then generate raw deflate data + with no zlib header or trailer, and will not compute an adler32 check value. + + windowBits can also be greater than 15 for optional gzip encoding. Add + 16 to windowBits to write a simple gzip header and trailer around the + compressed data instead of a zlib wrapper. The gzip header will have no + file name, no extra data, no comment, no modification time (set to zero), + no header crc, and the operating system will be set to 255 (unknown). If a + gzip stream is being written, strm->adler is a crc32 instead of an adler32. + + The memLevel parameter specifies how much memory should be allocated + for the internal compression state. memLevel=1 uses minimum memory but + is slow and reduces compression ratio; memLevel=9 uses maximum memory + for optimal speed. The default value is 8. See zconf.h for total memory + usage as a function of windowBits and memLevel. + + The strategy parameter is used to tune the compression algorithm. Use the + value Z_DEFAULT_STRATEGY for normal data, Z_FILTERED for data produced by a + filter (or predictor), Z_HUFFMAN_ONLY to force Huffman encoding only (no + string match), or Z_RLE to limit match distances to one (run-length + encoding). Filtered data consists mostly of small values with a somewhat + random distribution. In this case, the compression algorithm is tuned to + compress them better. The effect of Z_FILTERED is to force more Huffman + coding and less string matching; it is somewhat intermediate between + Z_DEFAULT and Z_HUFFMAN_ONLY. Z_RLE is designed to be almost as fast as + Z_HUFFMAN_ONLY, but give better compression for PNG image data. The strategy + parameter only affects the compression ratio but not the correctness of the + compressed output even if it is not set appropriately. Z_FIXED prevents the + use of dynamic Huffman codes, allowing for a simpler decoder for special + applications. + + deflateInit2 returns Z_OK if success, Z_MEM_ERROR if there was not enough + memory, Z_STREAM_ERROR if a parameter is invalid (such as an invalid + method). msg is set to null if there is no error message. deflateInit2 does + not perform any compression: this will be done by deflate(). +*/ + +ZEXTERN int ZEXPORT deflateSetDictionary OF((z_streamp strm, + const Bytef *dictionary, + uInt dictLength)); +/* + Initializes the compression dictionary from the given byte sequence + without producing any compressed output. This function must be called + immediately after deflateInit, deflateInit2 or deflateReset, before any + call of deflate. The compressor and decompressor must use exactly the same + dictionary (see inflateSetDictionary). + + The dictionary should consist of strings (byte sequences) that are likely + to be encountered later in the data to be compressed, with the most commonly + used strings preferably put towards the end of the dictionary. Using a + dictionary is most useful when the data to be compressed is short and can be + predicted with good accuracy; the data can then be compressed better than + with the default empty dictionary. + + Depending on the size of the compression data structures selected by + deflateInit or deflateInit2, a part of the dictionary may in effect be + discarded, for example if the dictionary is larger than the window size in + deflate or deflate2. Thus the strings most likely to be useful should be + put at the end of the dictionary, not at the front. In addition, the + current implementation of deflate will use at most the window size minus + 262 bytes of the provided dictionary. + + Upon return of this function, strm->adler is set to the adler32 value + of the dictionary; the decompressor may later use this value to determine + which dictionary has been used by the compressor. (The adler32 value + applies to the whole dictionary even if only a subset of the dictionary is + actually used by the compressor.) If a raw deflate was requested, then the + adler32 value is not computed and strm->adler is not set. + + deflateSetDictionary returns Z_OK if success, or Z_STREAM_ERROR if a + parameter is invalid (such as NULL dictionary) or the stream state is + inconsistent (for example if deflate has already been called for this stream + or if the compression method is bsort). deflateSetDictionary does not + perform any compression: this will be done by deflate(). +*/ + +ZEXTERN int ZEXPORT deflateCopy OF((z_streamp dest, + z_streamp source)); +/* + Sets the destination stream as a complete copy of the source stream. + + This function can be useful when several compression strategies will be + tried, for example when there are several ways of pre-processing the input + data with a filter. The streams that will be discarded should then be freed + by calling deflateEnd. Note that deflateCopy duplicates the internal + compression state which can be quite large, so this strategy is slow and + can consume lots of memory. + + deflateCopy returns Z_OK if success, Z_MEM_ERROR if there was not + enough memory, Z_STREAM_ERROR if the source stream state was inconsistent + (such as zalloc being NULL). msg is left unchanged in both source and + destination. +*/ + +ZEXTERN int ZEXPORT deflateReset OF((z_streamp strm)); +/* + This function is equivalent to deflateEnd followed by deflateInit, + but does not free and reallocate all the internal compression state. + The stream will keep the same compression level and any other attributes + that may have been set by deflateInit2. + + deflateReset returns Z_OK if success, or Z_STREAM_ERROR if the source + stream state was inconsistent (such as zalloc or state being NULL). +*/ + +ZEXTERN int ZEXPORT deflateParams OF((z_streamp strm, + int level, + int strategy)); +/* + Dynamically update the compression level and compression strategy. The + interpretation of level and strategy is as in deflateInit2. This can be + used to switch between compression and straight copy of the input data, or + to switch to a different kind of input data requiring a different + strategy. If the compression level is changed, the input available so far + is compressed with the old level (and may be flushed); the new level will + take effect only at the next call of deflate(). + + Before the call of deflateParams, the stream state must be set as for + a call of deflate(), since the currently available input may have to + be compressed and flushed. In particular, strm->avail_out must be non-zero. + + deflateParams returns Z_OK if success, Z_STREAM_ERROR if the source + stream state was inconsistent or if a parameter was invalid, Z_BUF_ERROR + if strm->avail_out was zero. +*/ + +ZEXTERN int ZEXPORT deflateTune OF((z_streamp strm, + int good_length, + int max_lazy, + int nice_length, + int max_chain)); +/* + Fine tune deflate's internal compression parameters. This should only be + used by someone who understands the algorithm used by zlib's deflate for + searching for the best matching string, and even then only by the most + fanatic optimizer trying to squeeze out the last compressed bit for their + specific input data. Read the deflate.c source code for the meaning of the + max_lazy, good_length, nice_length, and max_chain parameters. + + deflateTune() can be called after deflateInit() or deflateInit2(), and + returns Z_OK on success, or Z_STREAM_ERROR for an invalid deflate stream. + */ + +ZEXTERN uLong ZEXPORT deflateBound OF((z_streamp strm, + uLong sourceLen)); +/* + deflateBound() returns an upper bound on the compressed size after + deflation of sourceLen bytes. It must be called after deflateInit() + or deflateInit2(). This would be used to allocate an output buffer + for deflation in a single pass, and so would be called before deflate(). +*/ + +ZEXTERN int ZEXPORT deflatePrime OF((z_streamp strm, + int bits, + int value)); +/* + deflatePrime() inserts bits in the deflate output stream. The intent + is that this function is used to start off the deflate output with the + bits leftover from a previous deflate stream when appending to it. As such, + this function can only be used for raw deflate, and must be used before the + first deflate() call after a deflateInit2() or deflateReset(). bits must be + less than or equal to 16, and that many of the least significant bits of + value will be inserted in the output. + + deflatePrime returns Z_OK if success, or Z_STREAM_ERROR if the source + stream state was inconsistent. +*/ + +ZEXTERN int ZEXPORT deflateSetHeader OF((z_streamp strm, + gz_headerp head)); +/* + deflateSetHeader() provides gzip header information for when a gzip + stream is requested by deflateInit2(). deflateSetHeader() may be called + after deflateInit2() or deflateReset() and before the first call of + deflate(). The text, time, os, extra field, name, and comment information + in the provided gz_header structure are written to the gzip header (xflag is + ignored -- the extra flags are set according to the compression level). The + caller must assure that, if not Z_NULL, name and comment are terminated with + a zero byte, and that if extra is not Z_NULL, that extra_len bytes are + available there. If hcrc is true, a gzip header crc is included. Note that + the current versions of the command-line version of gzip (up through version + 1.3.x) do not support header crc's, and will report that it is a "multi-part + gzip file" and give up. + + If deflateSetHeader is not used, the default gzip header has text false, + the time set to zero, and os set to 255, with no extra, name, or comment + fields. The gzip header is returned to the default state by deflateReset(). + + deflateSetHeader returns Z_OK if success, or Z_STREAM_ERROR if the source + stream state was inconsistent. +*/ + +/* +ZEXTERN int ZEXPORT inflateInit2 OF((z_streamp strm, + int windowBits)); + + This is another version of inflateInit with an extra parameter. The + fields next_in, avail_in, zalloc, zfree and opaque must be initialized + before by the caller. + + The windowBits parameter is the base two logarithm of the maximum window + size (the size of the history buffer). It should be in the range 8..15 for + this version of the library. The default value is 15 if inflateInit is used + instead. windowBits must be greater than or equal to the windowBits value + provided to deflateInit2() while compressing, or it must be equal to 15 if + deflateInit2() was not used. If a compressed stream with a larger window + size is given as input, inflate() will return with the error code + Z_DATA_ERROR instead of trying to allocate a larger window. + + windowBits can also be -8..-15 for raw inflate. In this case, -windowBits + determines the window size. inflate() will then process raw deflate data, + not looking for a zlib or gzip header, not generating a check value, and not + looking for any check values for comparison at the end of the stream. This + is for use with other formats that use the deflate compressed data format + such as zip. Those formats provide their own check values. If a custom + format is developed using the raw deflate format for compressed data, it is + recommended that a check value such as an adler32 or a crc32 be applied to + the uncompressed data as is done in the zlib, gzip, and zip formats. For + most applications, the zlib format should be used as is. Note that comments + above on the use in deflateInit2() applies to the magnitude of windowBits. + + windowBits can also be greater than 15 for optional gzip decoding. Add + 32 to windowBits to enable zlib and gzip decoding with automatic header + detection, or add 16 to decode only the gzip format (the zlib format will + return a Z_DATA_ERROR). If a gzip stream is being decoded, strm->adler is + a crc32 instead of an adler32. + + inflateInit2 returns Z_OK if success, Z_MEM_ERROR if there was not enough + memory, Z_STREAM_ERROR if a parameter is invalid (such as a null strm). msg + is set to null if there is no error message. inflateInit2 does not perform + any decompression apart from reading the zlib header if present: this will + be done by inflate(). (So next_in and avail_in may be modified, but next_out + and avail_out are unchanged.) +*/ + +ZEXTERN int ZEXPORT inflateSetDictionary OF((z_streamp strm, + const Bytef *dictionary, + uInt dictLength)); +/* + Initializes the decompression dictionary from the given uncompressed byte + sequence. This function must be called immediately after a call of inflate, + if that call returned Z_NEED_DICT. The dictionary chosen by the compressor + can be determined from the adler32 value returned by that call of inflate. + The compressor and decompressor must use exactly the same dictionary (see + deflateSetDictionary). For raw inflate, this function can be called + immediately after inflateInit2() or inflateReset() and before any call of + inflate() to set the dictionary. The application must insure that the + dictionary that was used for compression is provided. + + inflateSetDictionary returns Z_OK if success, Z_STREAM_ERROR if a + parameter is invalid (such as NULL dictionary) or the stream state is + inconsistent, Z_DATA_ERROR if the given dictionary doesn't match the + expected one (incorrect adler32 value). inflateSetDictionary does not + perform any decompression: this will be done by subsequent calls of + inflate(). +*/ + +ZEXTERN int ZEXPORT inflateSync OF((z_streamp strm)); +/* + Skips invalid compressed data until a full flush point (see above the + description of deflate with Z_FULL_FLUSH) can be found, or until all + available input is skipped. No output is provided. + + inflateSync returns Z_OK if a full flush point has been found, Z_BUF_ERROR + if no more input was provided, Z_DATA_ERROR if no flush point has been found, + or Z_STREAM_ERROR if the stream structure was inconsistent. In the success + case, the application may save the current current value of total_in which + indicates where valid compressed data was found. In the error case, the + application may repeatedly call inflateSync, providing more input each time, + until success or end of the input data. +*/ + +ZEXTERN int ZEXPORT inflateCopy OF((z_streamp dest, + z_streamp source)); +/* + Sets the destination stream as a complete copy of the source stream. + + This function can be useful when randomly accessing a large stream. The + first pass through the stream can periodically record the inflate state, + allowing restarting inflate at those points when randomly accessing the + stream. + + inflateCopy returns Z_OK if success, Z_MEM_ERROR if there was not + enough memory, Z_STREAM_ERROR if the source stream state was inconsistent + (such as zalloc being NULL). msg is left unchanged in both source and + destination. +*/ + +ZEXTERN int ZEXPORT inflateReset OF((z_streamp strm)); +/* + This function is equivalent to inflateEnd followed by inflateInit, + but does not free and reallocate all the internal decompression state. + The stream will keep attributes that may have been set by inflateInit2. + + inflateReset returns Z_OK if success, or Z_STREAM_ERROR if the source + stream state was inconsistent (such as zalloc or state being NULL). +*/ + +ZEXTERN int ZEXPORT inflatePrime OF((z_streamp strm, + int bits, + int value)); +/* + This function inserts bits in the inflate input stream. The intent is + that this function is used to start inflating at a bit position in the + middle of a byte. The provided bits will be used before any bytes are used + from next_in. This function should only be used with raw inflate, and + should be used before the first inflate() call after inflateInit2() or + inflateReset(). bits must be less than or equal to 16, and that many of the + least significant bits of value will be inserted in the input. + + inflatePrime returns Z_OK if success, or Z_STREAM_ERROR if the source + stream state was inconsistent. +*/ + +ZEXTERN int ZEXPORT inflateGetHeader OF((z_streamp strm, + gz_headerp head)); +/* + inflateGetHeader() requests that gzip header information be stored in the + provided gz_header structure. inflateGetHeader() may be called after + inflateInit2() or inflateReset(), and before the first call of inflate(). + As inflate() processes the gzip stream, head->done is zero until the header + is completed, at which time head->done is set to one. If a zlib stream is + being decoded, then head->done is set to -1 to indicate that there will be + no gzip header information forthcoming. Note that Z_BLOCK can be used to + force inflate() to return immediately after header processing is complete + and before any actual data is decompressed. + + The text, time, xflags, and os fields are filled in with the gzip header + contents. hcrc is set to true if there is a header CRC. (The header CRC + was valid if done is set to one.) If extra is not Z_NULL, then extra_max + contains the maximum number of bytes to write to extra. Once done is true, + extra_len contains the actual extra field length, and extra contains the + extra field, or that field truncated if extra_max is less than extra_len. + If name is not Z_NULL, then up to name_max characters are written there, + terminated with a zero unless the length is greater than name_max. If + comment is not Z_NULL, then up to comm_max characters are written there, + terminated with a zero unless the length is greater than comm_max. When + any of extra, name, or comment are not Z_NULL and the respective field is + not present in the header, then that field is set to Z_NULL to signal its + absence. This allows the use of deflateSetHeader() with the returned + structure to duplicate the header. However if those fields are set to + allocated memory, then the application will need to save those pointers + elsewhere so that they can be eventually freed. + + If inflateGetHeader is not used, then the header information is simply + discarded. The header is always checked for validity, including the header + CRC if present. inflateReset() will reset the process to discard the header + information. The application would need to call inflateGetHeader() again to + retrieve the header from the next gzip stream. + + inflateGetHeader returns Z_OK if success, or Z_STREAM_ERROR if the source + stream state was inconsistent. +*/ + +/* +ZEXTERN int ZEXPORT inflateBackInit OF((z_streamp strm, int windowBits, + unsigned char FAR *window)); + + Initialize the internal stream state for decompression using inflateBack() + calls. The fields zalloc, zfree and opaque in strm must be initialized + before the call. If zalloc and zfree are Z_NULL, then the default library- + derived memory allocation routines are used. windowBits is the base two + logarithm of the window size, in the range 8..15. window is a caller + supplied buffer of that size. Except for special applications where it is + assured that deflate was used with small window sizes, windowBits must be 15 + and a 32K byte window must be supplied to be able to decompress general + deflate streams. + + See inflateBack() for the usage of these routines. + + inflateBackInit will return Z_OK on success, Z_STREAM_ERROR if any of + the paramaters are invalid, Z_MEM_ERROR if the internal state could not + be allocated, or Z_VERSION_ERROR if the version of the library does not + match the version of the header file. +*/ + +typedef unsigned (*in_func) OF((void FAR *, unsigned char FAR * FAR *)); +typedef int (*out_func) OF((void FAR *, unsigned char FAR *, unsigned)); + +ZEXTERN int ZEXPORT inflateBack OF((z_streamp strm, + in_func in, void FAR *in_desc, + out_func out, void FAR *out_desc)); +/* + inflateBack() does a raw inflate with a single call using a call-back + interface for input and output. This is more efficient than inflate() for + file i/o applications in that it avoids copying between the output and the + sliding window by simply making the window itself the output buffer. This + function trusts the application to not change the output buffer passed by + the output function, at least until inflateBack() returns. + + inflateBackInit() must be called first to allocate the internal state + and to initialize the state with the user-provided window buffer. + inflateBack() may then be used multiple times to inflate a complete, raw + deflate stream with each call. inflateBackEnd() is then called to free + the allocated state. + + A raw deflate stream is one with no zlib or gzip header or trailer. + This routine would normally be used in a utility that reads zip or gzip + files and writes out uncompressed files. The utility would decode the + header and process the trailer on its own, hence this routine expects + only the raw deflate stream to decompress. This is different from the + normal behavior of inflate(), which expects either a zlib or gzip header and + trailer around the deflate stream. + + inflateBack() uses two subroutines supplied by the caller that are then + called by inflateBack() for input and output. inflateBack() calls those + routines until it reads a complete deflate stream and writes out all of the + uncompressed data, or until it encounters an error. The function's + parameters and return types are defined above in the in_func and out_func + typedefs. inflateBack() will call in(in_desc, &buf) which should return the + number of bytes of provided input, and a pointer to that input in buf. If + there is no input available, in() must return zero--buf is ignored in that + case--and inflateBack() will return a buffer error. inflateBack() will call + out(out_desc, buf, len) to write the uncompressed data buf[0..len-1]. out() + should return zero on success, or non-zero on failure. If out() returns + non-zero, inflateBack() will return with an error. Neither in() nor out() + are permitted to change the contents of the window provided to + inflateBackInit(), which is also the buffer that out() uses to write from. + The length written by out() will be at most the window size. Any non-zero + amount of input may be provided by in(). + + For convenience, inflateBack() can be provided input on the first call by + setting strm->next_in and strm->avail_in. If that input is exhausted, then + in() will be called. Therefore strm->next_in must be initialized before + calling inflateBack(). If strm->next_in is Z_NULL, then in() will be called + immediately for input. If strm->next_in is not Z_NULL, then strm->avail_in + must also be initialized, and then if strm->avail_in is not zero, input will + initially be taken from strm->next_in[0 .. strm->avail_in - 1]. + + The in_desc and out_desc parameters of inflateBack() is passed as the + first parameter of in() and out() respectively when they are called. These + descriptors can be optionally used to pass any information that the caller- + supplied in() and out() functions need to do their job. + + On return, inflateBack() will set strm->next_in and strm->avail_in to + pass back any unused input that was provided by the last in() call. The + return values of inflateBack() can be Z_STREAM_END on success, Z_BUF_ERROR + if in() or out() returned an error, Z_DATA_ERROR if there was a format + error in the deflate stream (in which case strm->msg is set to indicate the + nature of the error), or Z_STREAM_ERROR if the stream was not properly + initialized. In the case of Z_BUF_ERROR, an input or output error can be + distinguished using strm->next_in which will be Z_NULL only if in() returned + an error. If strm->next is not Z_NULL, then the Z_BUF_ERROR was due to + out() returning non-zero. (in() will always be called before out(), so + strm->next_in is assured to be defined if out() returns non-zero.) Note + that inflateBack() cannot return Z_OK. +*/ + +ZEXTERN int ZEXPORT inflateBackEnd OF((z_streamp strm)); +/* + All memory allocated by inflateBackInit() is freed. + + inflateBackEnd() returns Z_OK on success, or Z_STREAM_ERROR if the stream + state was inconsistent. +*/ + +ZEXTERN uLong ZEXPORT zlibCompileFlags OF((void)); +/* Return flags indicating compile-time options. + + Type sizes, two bits each, 00 = 16 bits, 01 = 32, 10 = 64, 11 = other: + 1.0: size of uInt + 3.2: size of uLong + 5.4: size of voidpf (pointer) + 7.6: size of z_off_t + + Compiler, assembler, and debug options: + 8: DEBUG + 9: ASMV or ASMINF -- use ASM code + 10: ZLIB_WINAPI -- exported functions use the WINAPI calling convention + 11: 0 (reserved) + + One-time table building (smaller code, but not thread-safe if true): + 12: BUILDFIXED -- build static block decoding tables when needed + 13: DYNAMIC_CRC_TABLE -- build CRC calculation tables when needed + 14,15: 0 (reserved) + + Library content (indicates missing functionality): + 16: NO_GZCOMPRESS -- gz* functions cannot compress (to avoid linking + deflate code when not needed) + 17: NO_GZIP -- deflate can't write gzip streams, and inflate can't detect + and decode gzip streams (to avoid linking crc code) + 18-19: 0 (reserved) + + Operation variations (changes in library functionality): + 20: PKZIP_BUG_WORKAROUND -- slightly more permissive inflate + 21: FASTEST -- deflate algorithm with only one, lowest compression level + 22,23: 0 (reserved) + + The sprintf variant used by gzprintf (zero is best): + 24: 0 = vs*, 1 = s* -- 1 means limited to 20 arguments after the format + 25: 0 = *nprintf, 1 = *printf -- 1 means gzprintf() not secure! + 26: 0 = returns value, 1 = void -- 1 means inferred string length returned + + Remainder: + 27-31: 0 (reserved) + */ + + + /* utility functions */ + +/* + The following utility functions are implemented on top of the + basic stream-oriented functions. To simplify the interface, some + default options are assumed (compression level and memory usage, + standard memory allocation functions). The source code of these + utility functions can easily be modified if you need special options. +*/ + +ZEXTERN int ZEXPORT compress OF((Bytef *dest, uLongf *destLen, + const Bytef *source, uLong sourceLen)); +/* + Compresses the source buffer into the destination buffer. sourceLen is + the byte length of the source buffer. Upon entry, destLen is the total + size of the destination buffer, which must be at least the value returned + by compressBound(sourceLen). Upon exit, destLen is the actual size of the + compressed buffer. + This function can be used to compress a whole file at once if the + input file is mmap'ed. + compress returns Z_OK if success, Z_MEM_ERROR if there was not + enough memory, Z_BUF_ERROR if there was not enough room in the output + buffer. +*/ + +ZEXTERN int ZEXPORT compress2 OF((Bytef *dest, uLongf *destLen, + const Bytef *source, uLong sourceLen, + int level)); +/* + Compresses the source buffer into the destination buffer. The level + parameter has the same meaning as in deflateInit. sourceLen is the byte + length of the source buffer. Upon entry, destLen is the total size of the + destination buffer, which must be at least the value returned by + compressBound(sourceLen). Upon exit, destLen is the actual size of the + compressed buffer. + + compress2 returns Z_OK if success, Z_MEM_ERROR if there was not enough + memory, Z_BUF_ERROR if there was not enough room in the output buffer, + Z_STREAM_ERROR if the level parameter is invalid. +*/ + +ZEXTERN uLong ZEXPORT compressBound OF((uLong sourceLen)); +/* + compressBound() returns an upper bound on the compressed size after + compress() or compress2() on sourceLen bytes. It would be used before + a compress() or compress2() call to allocate the destination buffer. +*/ + +ZEXTERN int ZEXPORT uncompress OF((Bytef *dest, uLongf *destLen, + const Bytef *source, uLong sourceLen)); +/* + Decompresses the source buffer into the destination buffer. sourceLen is + the byte length of the source buffer. Upon entry, destLen is the total + size of the destination buffer, which must be large enough to hold the + entire uncompressed data. (The size of the uncompressed data must have + been saved previously by the compressor and transmitted to the decompressor + by some mechanism outside the scope of this compression library.) + Upon exit, destLen is the actual size of the compressed buffer. + This function can be used to decompress a whole file at once if the + input file is mmap'ed. + + uncompress returns Z_OK if success, Z_MEM_ERROR if there was not + enough memory, Z_BUF_ERROR if there was not enough room in the output + buffer, or Z_DATA_ERROR if the input data was corrupted or incomplete. +*/ + + +typedef voidp gzFile; + +ZEXTERN gzFile ZEXPORT gzopen OF((const char *path, const char *mode)); +/* + Opens a gzip (.gz) file for reading or writing. The mode parameter + is as in fopen ("rb" or "wb") but can also include a compression level + ("wb9") or a strategy: 'f' for filtered data as in "wb6f", 'h' for + Huffman only compression as in "wb1h", or 'R' for run-length encoding + as in "wb1R". (See the description of deflateInit2 for more information + about the strategy parameter.) + + gzopen can be used to read a file which is not in gzip format; in this + case gzread will directly read from the file without decompression. + + gzopen returns NULL if the file could not be opened or if there was + insufficient memory to allocate the (de)compression state; errno + can be checked to distinguish the two cases (if errno is zero, the + zlib error is Z_MEM_ERROR). */ + +ZEXTERN gzFile ZEXPORT gzdopen OF((int fd, const char *mode)); +/* + gzdopen() associates a gzFile with the file descriptor fd. File + descriptors are obtained from calls like open, dup, creat, pipe or + fileno (in the file has been previously opened with fopen). + The mode parameter is as in gzopen. + The next call of gzclose on the returned gzFile will also close the + file descriptor fd, just like fclose(fdopen(fd), mode) closes the file + descriptor fd. If you want to keep fd open, use gzdopen(dup(fd), mode). + gzdopen returns NULL if there was insufficient memory to allocate + the (de)compression state. +*/ + +ZEXTERN int ZEXPORT gzsetparams OF((gzFile file, int level, int strategy)); +/* + Dynamically update the compression level or strategy. See the description + of deflateInit2 for the meaning of these parameters. + gzsetparams returns Z_OK if success, or Z_STREAM_ERROR if the file was not + opened for writing. +*/ + +ZEXTERN int ZEXPORT gzread OF((gzFile file, voidp buf, unsigned len)); +/* + Reads the given number of uncompressed bytes from the compressed file. + If the input file was not in gzip format, gzread copies the given number + of bytes into the buffer. + gzread returns the number of uncompressed bytes actually read (0 for + end of file, -1 for error). */ + +ZEXTERN int ZEXPORT gzwrite OF((gzFile file, + voidpc buf, unsigned len)); +/* + Writes the given number of uncompressed bytes into the compressed file. + gzwrite returns the number of uncompressed bytes actually written + (0 in case of error). +*/ + +ZEXTERN int ZEXPORTVA gzprintf OF((gzFile file, const char *format, ...)); +/* + Converts, formats, and writes the args to the compressed file under + control of the format string, as in fprintf. gzprintf returns the number of + uncompressed bytes actually written (0 in case of error). The number of + uncompressed bytes written is limited to 4095. The caller should assure that + this limit is not exceeded. If it is exceeded, then gzprintf() will return + return an error (0) with nothing written. In this case, there may also be a + buffer overflow with unpredictable consequences, which is possible only if + zlib was compiled with the insecure functions sprintf() or vsprintf() + because the secure snprintf() or vsnprintf() functions were not available. +*/ + +ZEXTERN int ZEXPORT gzputs OF((gzFile file, const char *s)); +/* + Writes the given null-terminated string to the compressed file, excluding + the terminating null character. + gzputs returns the number of characters written, or -1 in case of error. +*/ + +ZEXTERN char * ZEXPORT gzgets OF((gzFile file, char *buf, int len)); +/* + Reads bytes from the compressed file until len-1 characters are read, or + a newline character is read and transferred to buf, or an end-of-file + condition is encountered. The string is then terminated with a null + character. + gzgets returns buf, or Z_NULL in case of error. +*/ + +ZEXTERN int ZEXPORT gzputc OF((gzFile file, int c)); +/* + Writes c, converted to an unsigned char, into the compressed file. + gzputc returns the value that was written, or -1 in case of error. +*/ + +ZEXTERN int ZEXPORT gzgetc OF((gzFile file)); +/* + Reads one byte from the compressed file. gzgetc returns this byte + or -1 in case of end of file or error. +*/ + +ZEXTERN int ZEXPORT gzungetc OF((int c, gzFile file)); +/* + Push one character back onto the stream to be read again later. + Only one character of push-back is allowed. gzungetc() returns the + character pushed, or -1 on failure. gzungetc() will fail if a + character has been pushed but not read yet, or if c is -1. The pushed + character will be discarded if the stream is repositioned with gzseek() + or gzrewind(). +*/ + +ZEXTERN int ZEXPORT gzflush OF((gzFile file, int flush)); +/* + Flushes all pending output into the compressed file. The parameter + flush is as in the deflate() function. The return value is the zlib + error number (see function gzerror below). gzflush returns Z_OK if + the flush parameter is Z_FINISH and all output could be flushed. + gzflush should be called only when strictly necessary because it can + degrade compression. +*/ + +ZEXTERN z_off_t ZEXPORT gzseek OF((gzFile file, + z_off_t offset, int whence)); +/* + Sets the starting position for the next gzread or gzwrite on the + given compressed file. The offset represents a number of bytes in the + uncompressed data stream. The whence parameter is defined as in lseek(2); + the value SEEK_END is not supported. + If the file is opened for reading, this function is emulated but can be + extremely slow. If the file is opened for writing, only forward seeks are + supported; gzseek then compresses a sequence of zeroes up to the new + starting position. + + gzseek returns the resulting offset location as measured in bytes from + the beginning of the uncompressed stream, or -1 in case of error, in + particular if the file is opened for writing and the new starting position + would be before the current position. +*/ + +ZEXTERN int ZEXPORT gzrewind OF((gzFile file)); +/* + Rewinds the given file. This function is supported only for reading. + + gzrewind(file) is equivalent to (int)gzseek(file, 0L, SEEK_SET) +*/ + +ZEXTERN z_off_t ZEXPORT gztell OF((gzFile file)); +/* + Returns the starting position for the next gzread or gzwrite on the + given compressed file. This position represents a number of bytes in the + uncompressed data stream. + + gztell(file) is equivalent to gzseek(file, 0L, SEEK_CUR) +*/ + +ZEXTERN int ZEXPORT gzeof OF((gzFile file)); +/* + Returns 1 when EOF has previously been detected reading the given + input stream, otherwise zero. +*/ + +ZEXTERN int ZEXPORT gzdirect OF((gzFile file)); +/* + Returns 1 if file is being read directly without decompression, otherwise + zero. +*/ + +ZEXTERN int ZEXPORT gzclose OF((gzFile file)); +/* + Flushes all pending output if necessary, closes the compressed file + and deallocates all the (de)compression state. The return value is the zlib + error number (see function gzerror below). +*/ + +ZEXTERN const char * ZEXPORT gzerror OF((gzFile file, int *errnum)); +/* + Returns the error message for the last error which occurred on the + given compressed file. errnum is set to zlib error number. If an + error occurred in the file system and not in the compression library, + errnum is set to Z_ERRNO and the application may consult errno + to get the exact error code. +*/ + +ZEXTERN void ZEXPORT gzclearerr OF((gzFile file)); +/* + Clears the error and end-of-file flags for file. This is analogous to the + clearerr() function in stdio. This is useful for continuing to read a gzip + file that is being written concurrently. +*/ + + /* checksum functions */ + +/* + These functions are not related to compression but are exported + anyway because they might be useful in applications using the + compression library. +*/ + +ZEXTERN uLong ZEXPORT adler32 OF((uLong adler, const Bytef *buf, uInt len)); +/* + Update a running Adler-32 checksum with the bytes buf[0..len-1] and + return the updated checksum. If buf is NULL, this function returns + the required initial value for the checksum. + An Adler-32 checksum is almost as reliable as a CRC32 but can be computed + much faster. Usage example: + + uLong adler = adler32(0L, Z_NULL, 0); + + while (read_buffer(buffer, length) != EOF) { + adler = adler32(adler, buffer, length); + } + if (adler != original_adler) error(); +*/ + +ZEXTERN uLong ZEXPORT adler32_combine OF((uLong adler1, uLong adler2, + z_off_t len2)); +/* + Combine two Adler-32 checksums into one. For two sequences of bytes, seq1 + and seq2 with lengths len1 and len2, Adler-32 checksums were calculated for + each, adler1 and adler2. adler32_combine() returns the Adler-32 checksum of + seq1 and seq2 concatenated, requiring only adler1, adler2, and len2. +*/ + +ZEXTERN uLong ZEXPORT crc32 OF((uLong crc, const Bytef *buf, uInt len)); +/* + Update a running CRC-32 with the bytes buf[0..len-1] and return the + updated CRC-32. If buf is NULL, this function returns the required initial + value for the for the crc. Pre- and post-conditioning (one's complement) is + performed within this function so it shouldn't be done by the application. + Usage example: + + uLong crc = crc32(0L, Z_NULL, 0); + + while (read_buffer(buffer, length) != EOF) { + crc = crc32(crc, buffer, length); + } + if (crc != original_crc) error(); +*/ + +ZEXTERN uLong ZEXPORT crc32_combine OF((uLong crc1, uLong crc2, z_off_t len2)); + +/* + Combine two CRC-32 check values into one. For two sequences of bytes, + seq1 and seq2 with lengths len1 and len2, CRC-32 check values were + calculated for each, crc1 and crc2. crc32_combine() returns the CRC-32 + check value of seq1 and seq2 concatenated, requiring only crc1, crc2, and + len2. +*/ + + + /* various hacks, don't look :) */ + +/* deflateInit and inflateInit are macros to allow checking the zlib version + * and the compiler's view of z_stream: + */ +ZEXTERN int ZEXPORT deflateInit_ OF((z_streamp strm, int level, + const char *version, int stream_size)); +ZEXTERN int ZEXPORT inflateInit_ OF((z_streamp strm, + const char *version, int stream_size)); +ZEXTERN int ZEXPORT deflateInit2_ OF((z_streamp strm, int level, int method, + int windowBits, int memLevel, + int strategy, const char *version, + int stream_size)); +ZEXTERN int ZEXPORT inflateInit2_ OF((z_streamp strm, int windowBits, + const char *version, int stream_size)); +ZEXTERN int ZEXPORT inflateBackInit_ OF((z_streamp strm, int windowBits, + unsigned char FAR *window, + const char *version, + int stream_size)); +#define deflateInit(strm, level) \ + deflateInit_((strm), (level), ZLIB_VERSION, sizeof(z_stream)) +#define inflateInit(strm) \ + inflateInit_((strm), ZLIB_VERSION, sizeof(z_stream)) +#define deflateInit2(strm, level, method, windowBits, memLevel, strategy) \ + deflateInit2_((strm),(level),(method),(windowBits),(memLevel),\ + (strategy), ZLIB_VERSION, sizeof(z_stream)) +#define inflateInit2(strm, windowBits) \ + inflateInit2_((strm), (windowBits), ZLIB_VERSION, sizeof(z_stream)) +#define inflateBackInit(strm, windowBits, window) \ + inflateBackInit_((strm), (windowBits), (window), \ + ZLIB_VERSION, sizeof(z_stream)) + + +#if !defined(ZUTIL_H) && !defined(NO_DUMMY_DECL) + struct internal_state {int dummy;}; /* hack for buggy compilers */ +#endif + +ZEXTERN const char * ZEXPORT zError OF((int)); +ZEXTERN int ZEXPORT inflateSyncPoint OF((z_streamp z)); +ZEXTERN const uLongf * ZEXPORT get_crc_table OF((void)); + +#ifdef __cplusplus +} +#endif + +#endif /* ZLIB_H */ diff --git a/Externals/MusicMod/TestPlayer/Src/Player.cpp b/Externals/MusicMod/TestPlayer/Src/Player.cpp new file mode 100644 index 0000000000..d22de77d8a --- /dev/null +++ b/Externals/MusicMod/TestPlayer/Src/Player.cpp @@ -0,0 +1,37 @@ +// Copyright (C) 2003-2008 Dolphin Project. + +// 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, version 2.0. + +// 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 2.0 for more details. + +// A copy of the GPL 2.0 should have been included with the program. +// If not, see http://www.gnu.org/licenses/ + +// Official SVN repository and contact information can be found at +// http://code.google.com/p/dolphin-emu/ + + + +////////////////////////////////////////////////////////////////////////////////////////// +// File description +/* ŻŻŻŻŻŻŻŻŻŻ +// This project loads the dll */ +/////////////////////////////////// + + +#include // System + +#include "../../Player/Src/PlayerExport.h" // Local player + + +void main() +{ + Player_Main(true); + + std::cin.get(); // Keep the window open +} \ No newline at end of file diff --git a/Externals/MusicMod/TestPlayer/TestPlayer.vcproj b/Externals/MusicMod/TestPlayer/TestPlayer.vcproj new file mode 100644 index 0000000000..d19f9f460b --- /dev/null +++ b/Externals/MusicMod/TestPlayer/TestPlayer.vcproj @@ -0,0 +1,403 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Source/MusicMod.sln b/Source/MusicMod.sln new file mode 100644 index 0000000000..b14f3cf139 --- /dev/null +++ b/Source/MusicMod.sln @@ -0,0 +1,498 @@ +Microsoft Visual Studio Solution File, Format Version 10.00 +# Visual C++ Express 2008 +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Core", "Core\Core\Core.vcproj", "{F0B874CB-4476-4199-9315-8343D05AE684}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} = {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF} = {95CCAABC-7062-47C4-B8C1-A064DD5F16FF} + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510} = {29C2ABC1-ADA5-42CD-A5FC-96022D52A510} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C} = {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_VideoDX9", "Plugins\Plugin_VideoDX9\Plugin_VideoDX9.vcproj", "{636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}" + ProjectSection(ProjectDependencies) = postProject + {3E03C179-8251-46E4-81F4-466F114BAC63} = {3E03C179-8251-46E4-81F4-466F114BAC63} + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA} = {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_PadSimple", "Plugins\Plugin_PadSimple\Plugin_PadSimple.vcproj", "{9A183B48-ECC2-4121-876A-9B3793686073}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} = {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Bochs_disasm", "..\Externals\Bochs_disasm\Bochs_disasm.vcproj", "{29C2ABC1-ADA5-42CD-A5FC-96022D52A510}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "zlib", "..\Externals\zlib\zlib.vcproj", "{3E03C179-8251-46E4-81F4-466F114BAC63}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_DSP_LLE", "Plugins\Plugin_DSP_LLE\Plugin_DSP_LLE.vcproj", "{3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}" + ProjectSection(ProjectDependencies) = postProject + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "DiscIO", "Core\DiscIO\DiscIO.vcproj", "{B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}" + ProjectSection(ProjectDependencies) = postProject + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF} = {95CCAABC-7062-47C4-B8C1-A064DD5F16FF} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_VideoOGL", "Plugins\Plugin_VideoOGL\Plugin_VideoOGL.vcproj", "{CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA} = {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Common", "Core\Common\Common.vcproj", "{C573CAF7-EE6A-458E-8049-16C0BF34C2E9}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Dolphin", "Core\DolphinWX\DolphinWX.vcproj", "{A72606EF-C5C1-4954-90AD-F0F93A8D97D9}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160} = {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {8D612734-FAA5-4B8A-804F-4DEA2367D495} = {8D612734-FAA5-4B8A-804F-4DEA2367D495} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {33546D62-7F34-4EA6-A88E-D538B36E16BF} = {33546D62-7F34-4EA6-A88E-D538B36E16BF} + {3E03C179-8251-46E4-81F4-466F114BAC63} = {3E03C179-8251-46E4-81F4-466F114BAC63} + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF} = {95CCAABC-7062-47C4-B8C1-A064DD5F16FF} + {521498BE-6089-4780-8223-E67C22F4E068} = {521498BE-6089-4780-8223-E67C22F4E068} + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510} = {29C2ABC1-ADA5-42CD-A5FC-96022D52A510} + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77} = {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77} + {F0B874CB-4476-4199-9315-8343D05AE684} = {F0B874CB-4476-4199-9315-8343D05AE684} + {0B72B5D6-5D72-4391-84A7-9CCA5392668A} = {0B72B5D6-5D72-4391-84A7-9CCA5392668A} + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C} = {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "wxBase28", "..\Externals\wxWidgets\build\msw\wx_base.vcproj", "{1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "wxCore28", "..\Externals\wxWidgets\build\msw\wx_core.vcproj", "{11F55366-12EC-4C44-A8CB-1D4E315D61ED}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Debugger", "Core\DebuggerWX\DebuggerWX.vcproj", "{4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510} = {29C2ABC1-ADA5-42CD-A5FC-96022D52A510} + {F0B874CB-4476-4199-9315-8343D05AE684} = {F0B874CB-4476-4199-9315-8343D05AE684} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "VideoCommon", "Core\VideoCommon\VideoCommon.vcproj", "{E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}" + ProjectSection(ProjectDependencies) = postProject + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_DSP_HLE", "Plugins\Plugin_DSP_HLE\Plugin_DSP_HLE.vcproj", "{D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "LZO", "..\Externals\LZO\LZO.vcproj", "{33546D62-7F34-4EA6-A88E-D538B36E16BF}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_Wiimote", "Plugins\Plugin_Wiimote\Plugin_Wiimote.vcproj", "{8D612734-FAA5-4B8A-804F-4DEA2367D495}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} = {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_nJoy_SDL_Test", "Plugins\Plugin_nJoy_Testing\Plugin_nJoy_SDL_Test.vcproj", "{ADF64291-57ED-4B7A-AB76-37B4A991504B}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "wxAdv28", "..\Externals\wxWidgets\build\msw\wx_adv.vcproj", "{0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_nJoy_SDL", "Plugins\Plugin_nJoy_SDL\Plugin_nJoy_SDL.vcproj", "{521498BE-6089-4780-8223-E67C22F4E068}" + ProjectSection(ProjectDependencies) = postProject + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} = {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} + {11F55366-12EC-4C44-A8CB-1D4E315D61ED} = {11F55366-12EC-4C44-A8CB-1D4E315D61ED} + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} = {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "InputCommon", "Core\InputCommon\InputCommon.vcproj", "{C7E5D50A-2916-464B-86A7-E10B3CC88ADA}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Main", "..\Externals\MusicMod\Main\Main.vcproj", "{95CCAABC-7062-47C4-B8C1-A064DD5F16FF}" + ProjectSection(ProjectDependencies) = postProject + {0B72B5D6-5D72-4391-84A7-9CCA5392668A} = {0B72B5D6-5D72-4391-84A7-9CCA5392668A} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Player", "..\Externals\MusicMod\Player\Player.vcproj", "{0B72B5D6-5D72-4391-84A7-9CCA5392668A}" + ProjectSection(ProjectDependencies) = postProject + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "TestPlayer", "..\Externals\MusicMod\TestPlayer\TestPlayer.vcproj", "{0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}" + ProjectSection(ProjectDependencies) = postProject + {0B72B5D6-5D72-4391-84A7-9CCA5392668A} = {0B72B5D6-5D72-4391-84A7-9CCA5392668A} + EndProjectSection +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Win32 = Debug|Win32 + Debug|x64 = Debug|x64 + DebugFast|Win32 = DebugFast|Win32 + DebugFast|x64 = DebugFast|x64 + Release JITIL|Win32 = Release JITIL|Win32 + Release JITIL|x64 = Release JITIL|x64 + Release|Win32 = Release|Win32 + Release|x64 = Release|x64 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {F0B874CB-4476-4199-9315-8343D05AE684}.Debug|Win32.ActiveCfg = Debug|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.Debug|Win32.Build.0 = Debug|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.Debug|x64.ActiveCfg = Debug|x64 + {F0B874CB-4476-4199-9315-8343D05AE684}.Debug|x64.Build.0 = Debug|x64 + {F0B874CB-4476-4199-9315-8343D05AE684}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {F0B874CB-4476-4199-9315-8343D05AE684}.DebugFast|x64.Build.0 = DebugFast|x64 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release JITIL|Win32.ActiveCfg = Release_JITIL|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release JITIL|Win32.Build.0 = Release_JITIL|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release JITIL|x64.ActiveCfg = Release_JITIL|x64 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release JITIL|x64.Build.0 = Release_JITIL|x64 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release|Win32.ActiveCfg = Release|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release|Win32.Build.0 = Release|Win32 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release|x64.ActiveCfg = Release|x64 + {F0B874CB-4476-4199-9315-8343D05AE684}.Release|x64.Build.0 = Release|x64 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Debug|Win32.ActiveCfg = Debug|Win32 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Debug|x64.ActiveCfg = Debug|x64 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Release JITIL|x64.ActiveCfg = Release|x64 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Release JITIL|x64.Build.0 = Release|x64 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Release|Win32.ActiveCfg = Release|Win32 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Release|x64.ActiveCfg = Release|x64 + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18}.Release|x64.Build.0 = Release|x64 + {9A183B48-ECC2-4121-876A-9B3793686073}.Debug|Win32.ActiveCfg = Debug|Win32 + {9A183B48-ECC2-4121-876A-9B3793686073}.Debug|x64.ActiveCfg = Debug|x64 + {9A183B48-ECC2-4121-876A-9B3793686073}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {9A183B48-ECC2-4121-876A-9B3793686073}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {9A183B48-ECC2-4121-876A-9B3793686073}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {9A183B48-ECC2-4121-876A-9B3793686073}.Release JITIL|x64.ActiveCfg = Release|x64 + {9A183B48-ECC2-4121-876A-9B3793686073}.Release JITIL|x64.Build.0 = Release|x64 + {9A183B48-ECC2-4121-876A-9B3793686073}.Release|Win32.ActiveCfg = Release|Win32 + {9A183B48-ECC2-4121-876A-9B3793686073}.Release|x64.ActiveCfg = Release|x64 + {9A183B48-ECC2-4121-876A-9B3793686073}.Release|x64.Build.0 = Release|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Debug|Win32.ActiveCfg = Debug|Win32 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Debug|Win32.Build.0 = Debug|Win32 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Debug|x64.ActiveCfg = Debug|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Debug|x64.Build.0 = Debug|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.DebugFast|x64.Build.0 = DebugFast|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Release JITIL|x64.ActiveCfg = Release|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Release JITIL|x64.Build.0 = Release|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Release|Win32.ActiveCfg = Release|Win32 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Release|x64.ActiveCfg = Release|x64 + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510}.Release|x64.Build.0 = Release|x64 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Debug|Win32.ActiveCfg = Debug|Win32 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Debug|Win32.Build.0 = Debug|Win32 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Debug|x64.ActiveCfg = Debug|x64 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Debug|x64.Build.0 = Debug|x64 + {3E03C179-8251-46E4-81F4-466F114BAC63}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {3E03C179-8251-46E4-81F4-466F114BAC63}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Release JITIL|x64.ActiveCfg = Release|x64 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Release JITIL|x64.Build.0 = Release|x64 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Release|Win32.ActiveCfg = Release|Win32 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Release|x64.ActiveCfg = Release|x64 + {3E03C179-8251-46E4-81F4-466F114BAC63}.Release|x64.Build.0 = Release|x64 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.Debug|Win32.ActiveCfg = Debug|Win32 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.Debug|x64.ActiveCfg = Debug|x64 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.DebugFast|x64.Build.0 = DebugFast|x64 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.Release JITIL|Win32.ActiveCfg = Release|x64 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.Release JITIL|x64.ActiveCfg = Release|x64 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.Release|Win32.ActiveCfg = Release|Win32 + {3D8156A9-64D1-4C8E-ADBE-1B319030E4A4}.Release|x64.ActiveCfg = Release|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Debug|Win32.ActiveCfg = Debug|Win32 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Debug|Win32.Build.0 = Debug|Win32 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Debug|x64.ActiveCfg = Debug|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Debug|x64.Build.0 = Debug|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.DebugFast|x64.Build.0 = DebugFast|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Release JITIL|x64.ActiveCfg = Release|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Release JITIL|x64.Build.0 = Release|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Release|Win32.ActiveCfg = Release|Win32 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Release|Win32.Build.0 = Release|Win32 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Release|x64.ActiveCfg = Release|x64 + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C}.Release|x64.Build.0 = Release|x64 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Debug|Win32.ActiveCfg = Debug|Win32 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Debug|x64.ActiveCfg = Debug|x64 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Release JITIL|x64.ActiveCfg = Release|x64 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Release JITIL|x64.Build.0 = Release|x64 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Release|Win32.ActiveCfg = Release|Win32 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Release|Win32.Build.0 = Release|Win32 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Release|x64.ActiveCfg = Release|x64 + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160}.Release|x64.Build.0 = Release|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Debug|Win32.ActiveCfg = Debug|Win32 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Debug|Win32.Build.0 = Debug|Win32 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Debug|x64.ActiveCfg = Debug|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Debug|x64.Build.0 = Debug|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.DebugFast|x64.Build.0 = DebugFast|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Release JITIL|x64.ActiveCfg = Release|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Release JITIL|x64.Build.0 = Release|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Release|Win32.ActiveCfg = Release|Win32 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Release|Win32.Build.0 = Release|Win32 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Release|x64.ActiveCfg = Release|x64 + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9}.Release|x64.Build.0 = Release|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Debug|Win32.ActiveCfg = Debug|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Debug|Win32.Build.0 = Debug|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Debug|x64.ActiveCfg = Debug|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Debug|x64.Build.0 = Debug|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.DebugFast|x64.Build.0 = DebugFast|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release JITIL|Win32.ActiveCfg = Release_JITIL|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release JITIL|Win32.Build.0 = Release_JITIL|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release JITIL|x64.ActiveCfg = Release_JITIL|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release JITIL|x64.Build.0 = Release_JITIL|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release|Win32.ActiveCfg = Release|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release|Win32.Build.0 = Release|Win32 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release|x64.ActiveCfg = Release|x64 + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9}.Release|x64.Build.0 = Release|x64 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Debug|Win32.ActiveCfg = Debug|Win32 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Debug|Win32.Build.0 = Debug|Win32 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Debug|x64.ActiveCfg = Debug|x64 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Debug|x64.Build.0 = Debug|x64 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Release JITIL|x64.ActiveCfg = Release|x64 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Release JITIL|x64.Build.0 = Release|x64 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Release|Win32.ActiveCfg = Release|Win32 + {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}.Release|x64.ActiveCfg = Release|x64 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Debug|Win32.ActiveCfg = Debug|Win32 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Debug|Win32.Build.0 = Debug|Win32 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Debug|x64.ActiveCfg = Debug|x64 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Debug|x64.Build.0 = Debug|x64 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release JITIL|x64.ActiveCfg = Release|x64 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release JITIL|x64.Build.0 = Release|x64 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release|Win32.ActiveCfg = Release|Win32 + {11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release|x64.ActiveCfg = Release|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Debug|Win32.ActiveCfg = Debug|Win32 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Debug|Win32.Build.0 = Debug|Win32 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Debug|x64.ActiveCfg = Debug|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Debug|x64.Build.0 = Debug|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.DebugFast|x64.Build.0 = DebugFast|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Release JITIL|x64.ActiveCfg = Release|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Release JITIL|x64.Build.0 = Release|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Release|Win32.ActiveCfg = Release|Win32 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Release|Win32.Build.0 = Release|Win32 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Release|x64.ActiveCfg = Release|x64 + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77}.Release|x64.Build.0 = Release|x64 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Debug|Win32.ActiveCfg = Debug|Win32 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Debug|x64.ActiveCfg = Debug|x64 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Release JITIL|x64.ActiveCfg = Release|x64 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Release JITIL|x64.Build.0 = Release|x64 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Release|Win32.ActiveCfg = Release|Win32 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Release|Win32.Build.0 = Release|Win32 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Release|x64.ActiveCfg = Release|x64 + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA}.Release|x64.Build.0 = Release|x64 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.Debug|Win32.ActiveCfg = Debug|Win32 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.Debug|x64.ActiveCfg = Debug|x64 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.DebugFast|x64.Build.0 = DebugFast|x64 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.Release JITIL|x64.ActiveCfg = Release|x64 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.Release JITIL|x64.Build.0 = Release|x64 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.Release|Win32.ActiveCfg = Release|Win32 + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8}.Release|x64.ActiveCfg = Release|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Debug|Win32.ActiveCfg = Debug|Win32 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Debug|x64.ActiveCfg = Debug|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Debug|x64.Build.0 = Debug|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.DebugFast|x64.Build.0 = DebugFast|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Release JITIL|x64.ActiveCfg = Release|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Release JITIL|x64.Build.0 = Release|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Release|Win32.ActiveCfg = Release|Win32 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Release|x64.ActiveCfg = Release|x64 + {33546D62-7F34-4EA6-A88E-D538B36E16BF}.Release|x64.Build.0 = Release|x64 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Debug|Win32.ActiveCfg = Debug|Win32 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Debug|x64.ActiveCfg = Debug|x64 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Release JITIL|x64.ActiveCfg = Release|x64 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Release JITIL|x64.Build.0 = Release|x64 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Release|Win32.ActiveCfg = Release|Win32 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Release|Win32.Build.0 = Release|Win32 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Release|x64.ActiveCfg = Release|x64 + {8D612734-FAA5-4B8A-804F-4DEA2367D495}.Release|x64.Build.0 = Release|x64 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.Debug|Win32.ActiveCfg = Debug|Win32 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.Debug|x64.ActiveCfg = Debug|x64 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.Release JITIL|Win32.ActiveCfg = Release|x64 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.Release JITIL|x64.ActiveCfg = Release|x64 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.Release|Win32.ActiveCfg = Release|Win32 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.Release|x64.ActiveCfg = Release|x64 + {ADF64291-57ED-4B7A-AB76-37B4A991504B}.Release|x64.Build.0 = Release|x64 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Debug|Win32.ActiveCfg = Debug|Win32 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Debug|Win32.Build.0 = Debug|Win32 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Debug|x64.ActiveCfg = Debug|x64 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Debug|x64.Build.0 = Debug|x64 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Release JITIL|x64.ActiveCfg = Release|x64 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Release JITIL|x64.Build.0 = Release|x64 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Release|Win32.ActiveCfg = Release|Win32 + {0E231FB1-F3C9-4724-ACCB-DE8BCB3C089E}.Release|x64.ActiveCfg = Release|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.Debug|Win32.ActiveCfg = Debug|Win32 + {521498BE-6089-4780-8223-E67C22F4E068}.Debug|Win32.Build.0 = Debug|Win32 + {521498BE-6089-4780-8223-E67C22F4E068}.Debug|x64.ActiveCfg = Debug|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.Debug|x64.Build.0 = Debug|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {521498BE-6089-4780-8223-E67C22F4E068}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.DebugFast|x64.Build.0 = DebugFast|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {521498BE-6089-4780-8223-E67C22F4E068}.Release JITIL|x64.ActiveCfg = Release|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.Release JITIL|x64.Build.0 = Release|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.Release|Win32.ActiveCfg = Release|Win32 + {521498BE-6089-4780-8223-E67C22F4E068}.Release|Win32.Build.0 = Release|Win32 + {521498BE-6089-4780-8223-E67C22F4E068}.Release|x64.ActiveCfg = Release|x64 + {521498BE-6089-4780-8223-E67C22F4E068}.Release|x64.Build.0 = Release|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Debug|Win32.ActiveCfg = Debug|Win32 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Debug|Win32.Build.0 = Debug|Win32 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Debug|x64.ActiveCfg = Debug|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Debug|x64.Build.0 = Debug|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.DebugFast|Win32.Build.0 = DebugFast|Win32 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.DebugFast|x64.Build.0 = DebugFast|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Release JITIL|x64.ActiveCfg = Release|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Release JITIL|x64.Build.0 = Release|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Release|Win32.ActiveCfg = Release|Win32 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Release|Win32.Build.0 = Release|Win32 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Release|x64.ActiveCfg = Release|x64 + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA}.Release|x64.Build.0 = Release|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Debug|Win32.ActiveCfg = Debug|Win32 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Debug|Win32.Build.0 = Debug|Win32 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Debug|x64.ActiveCfg = Debug|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Debug|x64.Build.0 = Debug|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.DebugFast|x64.Build.0 = DebugFast|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Release JITIL|x64.ActiveCfg = Release|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Release JITIL|x64.Build.0 = Release|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Release|Win32.ActiveCfg = Release|Win32 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Release|x64.ActiveCfg = Release|x64 + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF}.Release|x64.Build.0 = Release|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Debug|Win32.ActiveCfg = Debug|Win32 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Debug|Win32.Build.0 = Debug|Win32 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Debug|x64.ActiveCfg = Debug|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Debug|x64.Build.0 = Debug|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.DebugFast|x64.Build.0 = DebugFast|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Release JITIL|x64.ActiveCfg = Release|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Release JITIL|x64.Build.0 = Release|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Release|Win32.ActiveCfg = Release|Win32 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Release|x64.ActiveCfg = Release|x64 + {0B72B5D6-5D72-4391-84A7-9CCA5392668A}.Release|x64.Build.0 = Release|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Debug|Win32.ActiveCfg = Debug|Win32 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Debug|Win32.Build.0 = Debug|Win32 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Debug|x64.ActiveCfg = Debug|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Debug|x64.Build.0 = Debug|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.DebugFast|Win32.ActiveCfg = DebugFast|Win32 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.DebugFast|x64.ActiveCfg = DebugFast|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.DebugFast|x64.Build.0 = DebugFast|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Release JITIL|Win32.ActiveCfg = Release|Win32 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Release JITIL|x64.ActiveCfg = Release|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Release JITIL|x64.Build.0 = Release|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Release|Win32.ActiveCfg = Release|Win32 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Release|x64.ActiveCfg = Release|x64 + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718}.Release|x64.Build.0 = Release|x64 + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + AMDCaProjectFile = D:\Dev\Dolphin\trunk\Source\CodeAnalyst\Dolphin.caw + EndGlobalSection + GlobalSection(NestedProjects) = preSolution + {B7F1A9FB-BEA8-416E-9460-AE35A6A5165C} = {61C7F431-0623-4A8D-9C4B-EDE35696554A} + {C573CAF7-EE6A-458E-8049-16C0BF34C2E9} = {61C7F431-0623-4A8D-9C4B-EDE35696554A} + {A72606EF-C5C1-4954-90AD-F0F93A8D97D9} = {61C7F431-0623-4A8D-9C4B-EDE35696554A} + {4D3CD4C5-412B-4B49-9B1B-A68A2A129C77} = {61C7F431-0623-4A8D-9C4B-EDE35696554A} + {F0B874CB-4476-4199-9315-8343D05AE684} = {61C7F431-0623-4A8D-9C4B-EDE35696554A} + {C7E5D50A-2916-464B-86A7-E10B3CC88ADA} = {61C7F431-0623-4A8D-9C4B-EDE35696554A} + {9A183B48-ECC2-4121-876A-9B3793686073} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {C60D0E7A-ED05-4C67-9EE7-3A6C0D7801C8} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {CFDCEE0E-FA45-4F72-9FCC-0B88F5A75160} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {E5D1F0C0-AA07-4841-A4EB-4CF4DAA6B0FA} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {D6E56527-BBB9-4EAD-A6EC-49D4BF6AFCD8} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {8D612734-FAA5-4B8A-804F-4DEA2367D495} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {ADF64291-57ED-4B7A-AB76-37B4A991504B} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {521498BE-6089-4780-8223-E67C22F4E068} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {636FAD5F-02D1-4E9A-BE67-FB8EA99B9A18} = {5C17B1EB-6C76-438A-A503-8F3C7831023B} + {3E03C179-8251-46E4-81F4-466F114BAC63} = {4F427D1B-8C90-4D9C-B23D-A51493A1C471} + {48AD7E0A-25B1-4974-A1E3-03F8C438D34F} = {4F427D1B-8C90-4D9C-B23D-A51493A1C471} + {0318BA30-EF48-441A-9E10-DC85EFAE39F0} = {4F427D1B-8C90-4D9C-B23D-A51493A1C471} + {33546D62-7F34-4EA6-A88E-D538B36E16BF} = {4F427D1B-8C90-4D9C-B23D-A51493A1C471} + {71B16F46-0B00-4EDA-B253-D6D9D03A215C} = {4F427D1B-8C90-4D9C-B23D-A51493A1C471} + {29C2ABC1-ADA5-42CD-A5FC-96022D52A510} = {4F427D1B-8C90-4D9C-B23D-A51493A1C471} + {DE7C596C-CBC4-4278-8909-146D63990803} = {77CF6E34-3038-4B23-A2E7-90AD17801609} + {95CCAABC-7062-47C4-B8C1-A064DD5F16FF} = {77CF6E34-3038-4B23-A2E7-90AD17801609} + {0B72B5D6-5D72-4391-84A7-9CCA5392668A} = {77CF6E34-3038-4B23-A2E7-90AD17801609} + {0D14F1E9-490B-4A2D-A4EF-0535E8B3C718} = {77CF6E34-3038-4B23-A2E7-90AD17801609} + EndGlobalSection + GlobalSection(DPCodeReviewSolutionGUID) = preSolution + DPCodeReviewSolutionGUID = {00000000-0000-0000-0000-000000000000} + EndGlobalSection +EndGlobal