mirror of
https://github.com/tildearrow/furnace.git
synced 2024-11-26 14:33:01 +00:00
Merge branch 'master' of https://github.com/tildearrow/furnace into es5506_alt
This commit is contained in:
commit
dd79ae401b
52 changed files with 578 additions and 329 deletions
|
@ -614,6 +614,7 @@ src/gui/scaling.cpp
|
||||||
src/gui/settings.cpp
|
src/gui/settings.cpp
|
||||||
src/gui/songInfo.cpp
|
src/gui/songInfo.cpp
|
||||||
src/gui/songNotes.cpp
|
src/gui/songNotes.cpp
|
||||||
|
src/gui/speed.cpp
|
||||||
src/gui/spoiler.cpp
|
src/gui/spoiler.cpp
|
||||||
src/gui/stats.cpp
|
src/gui/stats.cpp
|
||||||
src/gui/subSongs.cpp
|
src/gui/subSongs.cpp
|
||||||
|
|
BIN
demos/arcade/Salamander_Starfield.fur
Normal file
BIN
demos/arcade/Salamander_Starfield.fur
Normal file
Binary file not shown.
Binary file not shown.
BIN
demos/arcade/iji_tor_segaxboard.fur
Normal file
BIN
demos/arcade/iji_tor_segaxboard.fur
Normal file
Binary file not shown.
BIN
demos/genesis/Plok_Beach.fur
Normal file
BIN
demos/genesis/Plok_Beach.fur
Normal file
Binary file not shown.
BIN
demos/misc/Wicked_Express_X68000.fur
Normal file
BIN
demos/misc/Wicked_Express_X68000.fur
Normal file
Binary file not shown.
BIN
demos/msx/Gyruss_Stage_2.fur
Normal file
BIN
demos/msx/Gyruss_Stage_2.fur
Normal file
Binary file not shown.
BIN
demos/opl/e3m2_opl3.fur
Normal file
BIN
demos/opl/e3m2_opl3.fur
Normal file
Binary file not shown.
BIN
demos/specs2/Tim_Follin.fur
Normal file
BIN
demos/specs2/Tim_Follin.fur
Normal file
Binary file not shown.
|
@ -8,11 +8,11 @@
|
||||||
|
|
||||||
#include "k005289.hpp"
|
#include "k005289.hpp"
|
||||||
|
|
||||||
void k005289_core::tick()
|
void k005289_core::tick(const unsigned int cycles)
|
||||||
{
|
{
|
||||||
for (timer_t &elem : m_timer)
|
for (timer_t &elem : m_timer)
|
||||||
{
|
{
|
||||||
elem.tick();
|
elem.tick(cycles);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,12 +24,12 @@ void k005289_core::reset()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void k005289_core::timer_t::tick()
|
void k005289_core::timer_t::tick(const unsigned int cycles) {
|
||||||
{
|
m_counter-=cycles;
|
||||||
if (bitfield(++m_counter, 0, 12) == 0)
|
while (m_counter < 0)
|
||||||
{
|
{
|
||||||
m_addr = bitfield(m_addr + 1, 0, 5);
|
m_addr = bitfield(m_addr + 1, 0, 5);
|
||||||
m_counter = m_freq;
|
m_counter += 0x1000-(m_freq&0xfff);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,7 @@ class k005289_core : public vgsound_emu_core
|
||||||
|
|
||||||
// internal state
|
// internal state
|
||||||
void reset();
|
void reset();
|
||||||
void tick();
|
void tick(const unsigned int cycles=1);
|
||||||
|
|
||||||
// accessors
|
// accessors
|
||||||
// Replace current frequency to lastest loaded pitch
|
// Replace current frequency to lastest loaded pitch
|
||||||
|
@ -63,7 +63,7 @@ class k005289_core : public vgsound_emu_core
|
||||||
|
|
||||||
// internal state
|
// internal state
|
||||||
void reset();
|
void reset();
|
||||||
void tick();
|
void tick(const unsigned int cycles);
|
||||||
|
|
||||||
// accessors
|
// accessors
|
||||||
// TG1/2 pin
|
// TG1/2 pin
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Konami SCC
|
# Konami SCC
|
||||||
|
|
||||||
|
## modified
|
||||||
|
|
||||||
|
the emulation core has been modified for optimization.
|
||||||
|
|
||||||
## Summary
|
## Summary
|
||||||
|
|
||||||
- 5 voice wavetable
|
- 5 voice wavetable
|
||||||
|
|
|
@ -10,17 +10,17 @@
|
||||||
#include "scc.hpp"
|
#include "scc.hpp"
|
||||||
|
|
||||||
// shared SCC features
|
// shared SCC features
|
||||||
void scc_core::tick()
|
void scc_core::tick(const int cycles)
|
||||||
{
|
{
|
||||||
m_out = 0;
|
m_out = 0;
|
||||||
for (auto &elem : m_voice)
|
for (auto &elem : m_voice)
|
||||||
{
|
{
|
||||||
elem.tick();
|
elem.tick(cycles);
|
||||||
m_out += elem.out();
|
m_out += elem.out();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void scc_core::voice_t::tick()
|
void scc_core::voice_t::tick(const int cycles)
|
||||||
{
|
{
|
||||||
if (m_pitch >= 9) // or voice is halted
|
if (m_pitch >= 9) // or voice is halted
|
||||||
{
|
{
|
||||||
|
@ -28,23 +28,27 @@ void scc_core::voice_t::tick()
|
||||||
const u16 temp = m_counter;
|
const u16 temp = m_counter;
|
||||||
if (m_host.m_test.freq_4bit()) // 4 bit frequency mode
|
if (m_host.m_test.freq_4bit()) // 4 bit frequency mode
|
||||||
{
|
{
|
||||||
m_counter = (m_counter & ~0x0ff) | (bitfield(bitfield(m_counter, 0, 8) - 1, 0, 8) << 0);
|
m_counter = (m_counter & ~0x0ff) | (bitfield(bitfield(m_counter, 0, 8) - cycles, 0, 8) << 0);
|
||||||
m_counter = (m_counter & ~0xf00) | (bitfield(bitfield(m_counter, 8, 4) - 1, 0, 4) << 8);
|
m_counter = (m_counter & ~0xf00) | (bitfield(bitfield(m_counter, 8, 4) - cycles, 0, 4) << 8);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_counter = bitfield(m_counter - 1, 0, 12);
|
m_counter = bitfield(m_counter - cycles, 0, 12);
|
||||||
}
|
}
|
||||||
|
|
||||||
// handle counter carry
|
// handle counter carry
|
||||||
const bool carry = m_host.m_test.freq_8bit()
|
const bool carry = (temp<cycles) || (m_host.m_test.freq_8bit()
|
||||||
? (bitfield(temp, 0, 8) == 0)
|
? (bitfield(temp, 0, 8) == 0)
|
||||||
: (m_host.m_test.freq_4bit() ? (bitfield(temp, 8, 4) == 0)
|
: (m_host.m_test.freq_4bit() ? (bitfield(temp, 8, 4) == 0)
|
||||||
: (bitfield(temp, 0, 12) == 0));
|
: (bitfield(temp, 0, 12) == 0)));
|
||||||
if (carry)
|
if (carry)
|
||||||
{
|
{
|
||||||
m_addr = bitfield(m_addr + 1, 0, 5);
|
m_addr = bitfield(m_addr + 1, 0, 5);
|
||||||
m_counter = m_pitch;
|
m_counter = m_pitch - ((temp<cycles)?(cycles-temp-1):0);
|
||||||
|
while (m_counter>m_pitch) {
|
||||||
|
m_addr = bitfield(m_addr + 1, 0, 5);
|
||||||
|
m_counter+=m_pitch-1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// get output
|
// get output
|
||||||
|
|
|
@ -41,7 +41,7 @@ class scc_core : public vgsound_emu_core
|
||||||
|
|
||||||
// internal state
|
// internal state
|
||||||
void reset();
|
void reset();
|
||||||
void tick();
|
void tick(const int cycles=1);
|
||||||
|
|
||||||
// accessors
|
// accessors
|
||||||
inline void reset_addr() { m_addr = 0; }
|
inline void reset_addr() { m_addr = 0; }
|
||||||
|
@ -151,7 +151,7 @@ class scc_core : public vgsound_emu_core
|
||||||
|
|
||||||
// internal state
|
// internal state
|
||||||
virtual void reset();
|
virtual void reset();
|
||||||
void tick();
|
void tick(const int cycles=1);
|
||||||
|
|
||||||
// getters
|
// getters
|
||||||
inline s32 out() { return m_out; } // output to DA0...DA10 pin
|
inline s32 out() { return m_out; } // output to DA0...DA10 pin
|
||||||
|
|
|
@ -45,6 +45,16 @@ no plans have been made for TX81Z MIDI passthrough, because:
|
||||||
- `1Bxx`: set attack of operator 2.
|
- `1Bxx`: set attack of operator 2.
|
||||||
- `1Cxx`: set attack of operator 3.
|
- `1Cxx`: set attack of operator 3.
|
||||||
- `1Dxx`: set attack of operator 4.
|
- `1Dxx`: set attack of operator 4.
|
||||||
|
- `1Exx`: set LFO AM depth.
|
||||||
|
- `1Fxx`: set LFO PM depth.
|
||||||
|
- `24xx`: set LFO 2 speed.
|
||||||
|
- `25xx`: set LFO 2 waveform. `xx` may be one of the following:
|
||||||
|
- `00`: saw
|
||||||
|
- `01`: square
|
||||||
|
- `02`: triangle
|
||||||
|
- `03`: noise
|
||||||
|
- `26xx`: set LFO 2 AM depth.
|
||||||
|
- `27xx`: set LFO 2 PM depth.
|
||||||
- `28xy`: set reverb of operator.
|
- `28xy`: set reverb of operator.
|
||||||
- `x` is the operator (1-4). a value of 0 means "all operators".
|
- `x` is the operator (1-4). a value of 0 means "all operators".
|
||||||
- `y` is the value.
|
- `y` is the value.
|
||||||
|
|
|
@ -103,9 +103,8 @@ enum DivDispatchCmds {
|
||||||
DIV_CMD_FM_AM_DEPTH, // (depth)
|
DIV_CMD_FM_AM_DEPTH, // (depth)
|
||||||
DIV_CMD_FM_PM_DEPTH, // (depth)
|
DIV_CMD_FM_PM_DEPTH, // (depth)
|
||||||
|
|
||||||
DIV_CMD_GENESIS_LFO, // unused?
|
DIV_CMD_FM_LFO2, // (speed)
|
||||||
|
DIV_CMD_FM_LFO2_WAVE, // (waveform)
|
||||||
DIV_CMD_ARCADE_LFO, // unused?
|
|
||||||
|
|
||||||
DIV_CMD_STD_NOISE_FREQ, // (freq)
|
DIV_CMD_STD_NOISE_FREQ, // (freq)
|
||||||
DIV_CMD_STD_NOISE_MODE, // (mode)
|
DIV_CMD_STD_NOISE_MODE, // (mode)
|
||||||
|
@ -227,6 +226,9 @@ enum DivDispatchCmds {
|
||||||
|
|
||||||
DIV_CMD_SURROUND_PANNING, // (out, val)
|
DIV_CMD_SURROUND_PANNING, // (out, val)
|
||||||
|
|
||||||
|
DIV_CMD_FM_AM2_DEPTH, // (depth)
|
||||||
|
DIV_CMD_FM_PM2_DEPTH, // (depth)
|
||||||
|
|
||||||
DIV_ALWAYS_SET_VOLUME, // () -> alwaysSetVol
|
DIV_ALWAYS_SET_VOLUME, // () -> alwaysSetVol
|
||||||
|
|
||||||
DIV_CMD_MAX
|
DIV_CMD_MAX
|
||||||
|
|
|
@ -4472,6 +4472,9 @@ bool DivEngine::init() {
|
||||||
for (int i=0; i<64; i++) {
|
for (int i=0; i<64; i++) {
|
||||||
vibTable[i]=127*sin(((double)i/64.0)*(2*M_PI));
|
vibTable[i]=127*sin(((double)i/64.0)*(2*M_PI));
|
||||||
}
|
}
|
||||||
|
for (int i=0; i<128; i++) {
|
||||||
|
tremTable[i]=255*0.5*(1.0-cos(((double)i/128.0)*(2*M_PI)));
|
||||||
|
}
|
||||||
for (int i=0; i<4096; i++) {
|
for (int i=0; i<4096; i++) {
|
||||||
reversePitchTable[i]=round(1024.0*pow(2.0,(2048.0-(double)i)/(12.0*128.0)));
|
reversePitchTable[i]=round(1024.0*pow(2.0,(2048.0-(double)i)/(12.0*128.0)));
|
||||||
pitchTable[i]=round(1024.0*pow(2.0,((double)i-2048.0)/(12.0*128.0)));
|
pitchTable[i]=round(1024.0*pow(2.0,((double)i-2048.0)/(12.0*128.0)));
|
||||||
|
|
|
@ -413,6 +413,7 @@ class DivEngine {
|
||||||
} sPreview;
|
} sPreview;
|
||||||
|
|
||||||
short vibTable[64];
|
short vibTable[64];
|
||||||
|
short tremTable[128];
|
||||||
int reversePitchTable[4096];
|
int reversePitchTable[4096];
|
||||||
int pitchTable[4096];
|
int pitchTable[4096];
|
||||||
char c163NameCS[1024];
|
char c163NameCS[1024];
|
||||||
|
@ -1158,6 +1159,7 @@ class DivEngine {
|
||||||
memset(dispatchOfChan,0,DIV_MAX_CHANS*sizeof(int));
|
memset(dispatchOfChan,0,DIV_MAX_CHANS*sizeof(int));
|
||||||
memset(sysOfChan,0,DIV_MAX_CHANS*sizeof(int));
|
memset(sysOfChan,0,DIV_MAX_CHANS*sizeof(int));
|
||||||
memset(vibTable,0,64*sizeof(short));
|
memset(vibTable,0,64*sizeof(short));
|
||||||
|
memset(tremTable,0,128*sizeof(short));
|
||||||
memset(reversePitchTable,0,4096*sizeof(int));
|
memset(reversePitchTable,0,4096*sizeof(int));
|
||||||
memset(pitchTable,0,4096*sizeof(int));
|
memset(pitchTable,0,4096*sizeof(int));
|
||||||
memset(sysDefs,0,256*sizeof(void*));
|
memset(sysDefs,0,256*sizeof(void*));
|
||||||
|
|
|
@ -56,7 +56,7 @@ void DivMacroStruct::doMacro(DivInstrumentMacro& source, bool released, bool tic
|
||||||
}
|
}
|
||||||
if (delay>0) {
|
if (delay>0) {
|
||||||
delay--;
|
delay--;
|
||||||
had=false;
|
if (!linger) had=false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (began && source.delay>0) {
|
if (began && source.delay>0) {
|
||||||
|
@ -523,4 +523,4 @@ DivMacroStruct* DivMacroInt::structByName(const String& name) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
#undef CONSIDER
|
#undef CONSIDER
|
||||||
|
|
|
@ -44,7 +44,7 @@ void DivPlatformBubSysWSG::acquire(short** buf, size_t len) {
|
||||||
for (size_t h=0; h<len; h++) {
|
for (size_t h=0; h<len; h++) {
|
||||||
signed int out=0;
|
signed int out=0;
|
||||||
// K005289 part
|
// K005289 part
|
||||||
k005289.tick();
|
k005289.tick(8);
|
||||||
|
|
||||||
// Wavetable part
|
// Wavetable part
|
||||||
for (int i=0; i<2; i++) {
|
for (int i=0; i<2; i++) {
|
||||||
|
@ -60,7 +60,7 @@ void DivPlatformBubSysWSG::acquire(short** buf, size_t len) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (++writeOscBuf>=64) writeOscBuf=0;
|
if (++writeOscBuf>=8) writeOscBuf=0;
|
||||||
|
|
||||||
out<<=6; // scale output to 16 bit
|
out<<=6; // scale output to 16 bit
|
||||||
|
|
||||||
|
@ -332,9 +332,9 @@ void DivPlatformBubSysWSG::notifyInsDeletion(void* ins) {
|
||||||
void DivPlatformBubSysWSG::setFlags(const DivConfig& flags) {
|
void DivPlatformBubSysWSG::setFlags(const DivConfig& flags) {
|
||||||
chipClock=COLOR_NTSC;
|
chipClock=COLOR_NTSC;
|
||||||
CHECK_CUSTOM_CLOCK;
|
CHECK_CUSTOM_CLOCK;
|
||||||
rate=chipClock;
|
rate=chipClock/8;
|
||||||
for (int i=0; i<2; i++) {
|
for (int i=0; i<2; i++) {
|
||||||
oscBuf[i]->rate=rate/64;
|
oscBuf[i]->rate=rate/8;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -41,8 +41,14 @@ class DivPlatformOPM: public DivPlatformFMBase {
|
||||||
0x00, 0x08, 0x10, 0x18
|
0x00, 0x08, 0x10, 0x18
|
||||||
};
|
};
|
||||||
|
|
||||||
|
unsigned char lfoValue, lfoValue2, lfoShape, lfoShape2;
|
||||||
|
|
||||||
DivPlatformOPM():
|
DivPlatformOPM():
|
||||||
DivPlatformFMBase() {}
|
DivPlatformFMBase(),
|
||||||
|
lfoValue(0),
|
||||||
|
lfoValue2(0),
|
||||||
|
lfoShape(0),
|
||||||
|
lfoShape2(0) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -225,6 +225,9 @@ void DivPlatformGenesis::acquire_ymfm(short** buf, size_t len) {
|
||||||
//OPN2_Write(&fm,0,0);
|
//OPN2_Write(&fm,0,0);
|
||||||
|
|
||||||
for (int i=0; i<6; i++) {
|
for (int i=0; i<6; i++) {
|
||||||
|
int chOut=(fme->debug_channel(i)->debug_output(0)+fme->debug_channel(i)->debug_output(1))<<6;
|
||||||
|
if (chOut<-32768) chOut=-32768;
|
||||||
|
if (chOut>32767) chOut=32767;
|
||||||
if (i==5) {
|
if (i==5) {
|
||||||
if (fm_ymfm->debug_dac_enable()) {
|
if (fm_ymfm->debug_dac_enable()) {
|
||||||
if (softPCM) {
|
if (softPCM) {
|
||||||
|
@ -234,10 +237,10 @@ void DivPlatformGenesis::acquire_ymfm(short** buf, size_t len) {
|
||||||
oscBuf[i]->data[oscBuf[i]->needle++]=fm_ymfm->debug_dac_data()<<7;
|
oscBuf[i]->data[oscBuf[i]->needle++]=fm_ymfm->debug_dac_data()<<7;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
oscBuf[i]->data[oscBuf[i]->needle++]=(fme->debug_channel(i)->debug_output(0)+fme->debug_channel(i)->debug_output(1))<<6;
|
oscBuf[i]->data[oscBuf[i]->needle++]=chOut;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
oscBuf[i]->data[oscBuf[i]->needle++]=(fme->debug_channel(i)->debug_output(0)+fme->debug_channel(i)->debug_output(1))<<6;
|
oscBuf[i]->data[oscBuf[i]->needle++]=chOut;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -651,10 +651,10 @@ void DivPlatformGenesisExt::forceIns() {
|
||||||
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
||||||
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
||||||
if (i==2 && extMode) { // extended channel
|
if (i==2 && extMode) { // extended channel
|
||||||
if (isOpMuted[j]) {
|
if (isOpMuted[orderedOps[j]]) {
|
||||||
rWrite(baseAddr+0x40,127);
|
rWrite(baseAddr+0x40,127);
|
||||||
} else if (KVS(i,j)) {
|
} else if (KVS(i,j)) {
|
||||||
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[j].outVol&0x7f,127));
|
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[orderedOps[j]].outVol&0x7f,127));
|
||||||
} else {
|
} else {
|
||||||
rWrite(baseAddr+0x40,op.tl);
|
rWrite(baseAddr+0x40,op.tl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1555,7 +1555,11 @@ DivMacroInt* DivPlatformOPL::getChanMacroInt(int ch) {
|
||||||
}
|
}
|
||||||
|
|
||||||
DivDispatchOscBuffer* DivPlatformOPL::getOscBuffer(int ch) {
|
DivDispatchOscBuffer* DivPlatformOPL::getOscBuffer(int ch) {
|
||||||
if (ch>=totalChans) return NULL;
|
if (oplType==759) {
|
||||||
|
if (ch>=totalChans+1) return NULL;
|
||||||
|
} else {
|
||||||
|
if (ch>=totalChans) return NULL;
|
||||||
|
}
|
||||||
if (oplType==3 && ch<12) {
|
if (oplType==3 && ch<12) {
|
||||||
if (chan[ch&(~1)].fourOp) {
|
if (chan[ch&(~1)].fourOp) {
|
||||||
if (ch&1) {
|
if (ch&1) {
|
||||||
|
|
|
@ -82,9 +82,7 @@ const char** DivPlatformSCC::getRegisterSheet() {
|
||||||
|
|
||||||
void DivPlatformSCC::acquire(short** buf, size_t len) {
|
void DivPlatformSCC::acquire(short** buf, size_t len) {
|
||||||
for (size_t h=0; h<len; h++) {
|
for (size_t h=0; h<len; h++) {
|
||||||
for (int i=0; i<16; i++) {
|
scc->tick(16);
|
||||||
scc->tick();
|
|
||||||
}
|
|
||||||
short out=(short)scc->out()<<5;
|
short out=(short)scc->out()<<5;
|
||||||
buf[0][h]=out;
|
buf[0][h]=out;
|
||||||
|
|
||||||
|
|
|
@ -22,7 +22,7 @@
|
||||||
#include "../../ta-log.h"
|
#include "../../ta-log.h"
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
#define rWrite(a,v) {if (!skipRegisterWrites) {writes.emplace(a,v); if (dumpWrites) {addWrite(0x200+a,v);}}}
|
#define rWrite(a,v) {if (!skipRegisterWrites) {writes.emplace(a,v); if (dumpWrites) {addWrite(a,v);}}}
|
||||||
|
|
||||||
const char* regCheatSheetSN[]={
|
const char* regCheatSheetSN[]={
|
||||||
"DATA", "0",
|
"DATA", "0",
|
||||||
|
|
|
@ -811,7 +811,7 @@ const void* DivPlatformSNES::getSampleMem(int index) {
|
||||||
|
|
||||||
size_t DivPlatformSNES::getSampleMemCapacity(int index) {
|
size_t DivPlatformSNES::getSampleMemCapacity(int index) {
|
||||||
// TODO change it based on current echo buffer size
|
// TODO change it based on current echo buffer size
|
||||||
return index == 0 ? 65536 : 0;
|
return index == 0 ? (65536-echoDelay*2048) : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t DivPlatformSNES::getSampleMemUsage(int index) {
|
size_t DivPlatformSNES::getSampleMemUsage(int index) {
|
||||||
|
@ -825,7 +825,7 @@ bool DivPlatformSNES::isSampleLoaded(int index, int sample) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void DivPlatformSNES::renderSamples(int sysID) {
|
void DivPlatformSNES::renderSamples(int sysID) {
|
||||||
memset(copyOfSampleMem,0,getSampleMemCapacity());
|
memset(copyOfSampleMem,0,65536);
|
||||||
memset(sampleOff,0,256*sizeof(unsigned int));
|
memset(sampleOff,0,256*sizeof(unsigned int));
|
||||||
memset(sampleLoaded,0,256*sizeof(bool));
|
memset(sampleLoaded,0,256*sizeof(bool));
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,25 @@
|
||||||
|
/* su.cpp/su.h - Sound Unit emulator
|
||||||
|
* Copyright (C) 2015-2023 tildearrow
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
#define _USE_MATH_DEFINES
|
#define _USE_MATH_DEFINES
|
||||||
#include "su.h"
|
#include "su.h"
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
@ -391,7 +413,7 @@ void SoundUnit::Reset() {
|
||||||
|
|
||||||
#ifdef TA_BIG_ENDIAN
|
#ifdef TA_BIG_ENDIAN
|
||||||
const unsigned char suBERemap[32]={
|
const unsigned char suBERemap[32]={
|
||||||
0x01, 0x00, 0x02, 0x03, 0x05, 0x04, 0x07, 0x06, 0x08, 0x09, 0x0b, 0x0a, 0x0d, 0x0c, 0x0f, 0x0e,
|
0x01, 0x00, 0x02, 0x03, 0x04, 0x05, 0x07, 0x06, 0x08, 0x09, 0x0b, 0x0a, 0x0d, 0x0c, 0x0f, 0x0e,
|
||||||
0x11, 0x10, 0x12, 0x13, 0x15, 0x14, 0x16, 0x17, 0x19, 0x18, 0x1a, 0x1b, 0x1c, 0x1d, 0x1f, 0x1e
|
0x11, 0x10, 0x12, 0x13, 0x15, 0x14, 0x16, 0x17, 0x19, 0x18, 0x1a, 0x1b, 0x1c, 0x1d, 0x1f, 0x1e
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,7 +1,27 @@
|
||||||
|
/* su.cpp/su.h - Sound Unit emulator
|
||||||
|
* Copyright (C) 2015-2023 tildearrow
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
#include <assert.h>
|
|
||||||
|
|
||||||
class SoundUnit {
|
class SoundUnit {
|
||||||
signed char SCsine[256];
|
signed char SCsine[256];
|
||||||
|
|
|
@ -497,6 +497,9 @@ void DivPlatformSoundUnit::reset() {
|
||||||
rWrite(0x9d,ilCtrl);
|
rWrite(0x9d,ilCtrl);
|
||||||
rWrite(0xbc,ilSize);
|
rWrite(0xbc,ilSize);
|
||||||
rWrite(0xbd,fil1);
|
rWrite(0xbd,fil1);
|
||||||
|
|
||||||
|
// copy sample memory
|
||||||
|
memcpy(su->pcm,sampleMem,sampleMemSize?65536:8192);
|
||||||
}
|
}
|
||||||
|
|
||||||
int DivPlatformSoundUnit::getOutputCount() {
|
int DivPlatformSoundUnit::getOutputCount() {
|
||||||
|
@ -545,7 +548,7 @@ void DivPlatformSoundUnit::poke(std::vector<DivRegWrite>& wlist) {
|
||||||
}
|
}
|
||||||
|
|
||||||
const void* DivPlatformSoundUnit::getSampleMem(int index) {
|
const void* DivPlatformSoundUnit::getSampleMem(int index) {
|
||||||
return (index==0)?su->pcm:NULL;
|
return (index==0)?sampleMem:NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t DivPlatformSoundUnit::getSampleMemCapacity(int index) {
|
size_t DivPlatformSoundUnit::getSampleMemCapacity(int index) {
|
||||||
|
@ -563,7 +566,7 @@ bool DivPlatformSoundUnit::isSampleLoaded(int index, int sample) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void DivPlatformSoundUnit::renderSamples(int sysID) {
|
void DivPlatformSoundUnit::renderSamples(int sysID) {
|
||||||
memset(su->pcm,0,getSampleMemCapacity(0));
|
memset(sampleMem,0,sampleMemSize?65536:8192);
|
||||||
memset(sampleOffSU,0,256*sizeof(unsigned int));
|
memset(sampleOffSU,0,256*sizeof(unsigned int));
|
||||||
memset(sampleLoaded,0,256*sizeof(bool));
|
memset(sampleLoaded,0,256*sizeof(bool));
|
||||||
|
|
||||||
|
@ -582,10 +585,10 @@ void DivPlatformSoundUnit::renderSamples(int sysID) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (memPos+paddedLen>=getSampleMemCapacity(0)) {
|
if (memPos+paddedLen>=getSampleMemCapacity(0)) {
|
||||||
memcpy(su->pcm+memPos,s->data8,getSampleMemCapacity(0)-memPos);
|
memcpy(sampleMem+memPos,s->data8,getSampleMemCapacity(0)-memPos);
|
||||||
logW("out of PCM memory for sample %d!",i);
|
logW("out of PCM memory for sample %d!",i);
|
||||||
} else {
|
} else {
|
||||||
memcpy(su->pcm+memPos,s->data8,paddedLen);
|
memcpy(sampleMem+memPos,s->data8,paddedLen);
|
||||||
sampleLoaded[i]=true;
|
sampleLoaded[i]=true;
|
||||||
}
|
}
|
||||||
sampleOffSU[i]=memPos;
|
sampleOffSU[i]=memPos;
|
||||||
|
@ -593,6 +596,8 @@ void DivPlatformSoundUnit::renderSamples(int sysID) {
|
||||||
}
|
}
|
||||||
sampleMemLen=memPos;
|
sampleMemLen=memPos;
|
||||||
sysIDCache=sysID;
|
sysIDCache=sysID;
|
||||||
|
|
||||||
|
memcpy(su->pcm,sampleMem,sampleMemSize?65536:8192);
|
||||||
}
|
}
|
||||||
|
|
||||||
int DivPlatformSoundUnit::init(DivEngine* p, int channels, int sugRate, const DivConfig& flags) {
|
int DivPlatformSoundUnit::init(DivEngine* p, int channels, int sugRate, const DivConfig& flags) {
|
||||||
|
@ -604,6 +609,8 @@ int DivPlatformSoundUnit::init(DivEngine* p, int channels, int sugRate, const Di
|
||||||
oscBuf[i]=new DivDispatchOscBuffer;
|
oscBuf[i]=new DivDispatchOscBuffer;
|
||||||
}
|
}
|
||||||
su=new SoundUnit();
|
su=new SoundUnit();
|
||||||
|
sampleMem=new unsigned char[65536];
|
||||||
|
memset(sampleMem,0,65536);
|
||||||
sysIDCache=0;
|
sysIDCache=0;
|
||||||
setFlags(flags);
|
setFlags(flags);
|
||||||
reset();
|
reset();
|
||||||
|
@ -615,6 +622,7 @@ void DivPlatformSoundUnit::quit() {
|
||||||
delete oscBuf[i];
|
delete oscBuf[i];
|
||||||
}
|
}
|
||||||
delete su;
|
delete su;
|
||||||
|
delete sampleMem;
|
||||||
}
|
}
|
||||||
|
|
||||||
DivPlatformSoundUnit::~DivPlatformSoundUnit() {
|
DivPlatformSoundUnit::~DivPlatformSoundUnit() {
|
||||||
|
|
|
@ -89,6 +89,7 @@ class DivPlatformSoundUnit: public DivDispatch {
|
||||||
short tempR;
|
short tempR;
|
||||||
unsigned char sampleBank, lfoMode, lfoSpeed;
|
unsigned char sampleBank, lfoMode, lfoSpeed;
|
||||||
SoundUnit* su;
|
SoundUnit* su;
|
||||||
|
unsigned char* sampleMem;
|
||||||
size_t sampleMemLen;
|
size_t sampleMemLen;
|
||||||
unsigned char regPool[128];
|
unsigned char regPool[128];
|
||||||
double NOTE_SU(int ch, int note);
|
double NOTE_SU(int ch, int note);
|
||||||
|
|
|
@ -138,14 +138,14 @@ void DivPlatformTIA::tick(bool sysTick) {
|
||||||
bf+=chan[i].arpOff;
|
bf+=chan[i].arpOff;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
chan[i].freq=dealWithFreq(chan[i].shape,bf,chan[i].pitch)+chan[i].pitch2;
|
chan[i].freq=dealWithFreq(chan[i].shape,bf,chan[i].pitch+chan[i].pitch2);
|
||||||
if ((chan[i].shape==4 || chan[i].shape==5) && !(chan[i].baseFreq&0x80000000 && ((chan[i].baseFreq&0x7fffffff)<32))) {
|
if ((chan[i].shape==4 || chan[i].shape==5) && !(chan[i].baseFreq&0x80000000 && ((chan[i].baseFreq&0x7fffffff)<32))) {
|
||||||
if (bf<39*256) {
|
if (bf<39*256) {
|
||||||
rWrite(0x15+i,6);
|
rWrite(0x15+i,6);
|
||||||
chan[i].freq=dealWithFreq(6,bf,chan[i].pitch)+chan[i].pitch2;
|
chan[i].freq=dealWithFreq(6,bf,chan[i].pitch+chan[i].pitch2);
|
||||||
} else if (bf<59*256) {
|
} else if (bf<59*256) {
|
||||||
rWrite(0x15+i,12);
|
rWrite(0x15+i,12);
|
||||||
chan[i].freq=dealWithFreq(12,bf,chan[i].pitch)+chan[i].pitch2;
|
chan[i].freq=dealWithFreq(12,bf,chan[i].pitch+chan[i].pitch2);
|
||||||
} else {
|
} else {
|
||||||
rWrite(0x15+i,chan[i].shape);
|
rWrite(0x15+i,chan[i].shape);
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,8 @@
|
||||||
#define ADDR_WS_FINE 0x100
|
#define ADDR_WS_FINE 0x100
|
||||||
// actually 0xc0 but bit 5 of data selects address
|
// actually 0xc0 but bit 5 of data selects address
|
||||||
#define ADDR_EGS_REV 0x120
|
#define ADDR_EGS_REV 0x120
|
||||||
|
// actually 0x38 but bits 7 and 2 select address
|
||||||
|
#define ADDR_FMS2_AMS2 0x140
|
||||||
|
|
||||||
const char* regCheatSheetOPZ[]={
|
const char* regCheatSheetOPZ[]={
|
||||||
"Test", "00",
|
"Test", "00",
|
||||||
|
@ -139,7 +141,8 @@ void DivPlatformTX81Z::tick(bool sysTick) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (chan[i].std.wave.had) {
|
if (chan[i].std.wave.had) {
|
||||||
rWrite(0x1b,chan[i].std.wave.val&3);
|
lfoShape=chan[i].std.wave.val&3;
|
||||||
|
immWrite(0x1b,lfoShape|(lfoShape2<<2));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (chan[i].std.pitch.had) {
|
if (chan[i].std.pitch.had) {
|
||||||
|
@ -177,7 +180,28 @@ void DivPlatformTX81Z::tick(bool sysTick) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (chan[i].std.ex3.had) {
|
if (chan[i].std.ex3.had) {
|
||||||
immWrite(0x18,chan[i].std.ex3.val);
|
lfoValue=chan[i].std.ex3.val;
|
||||||
|
immWrite(0x18,lfoValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chan[i].std.ex5.had) {
|
||||||
|
amDepth2=chan[i].std.ex5.val;
|
||||||
|
immWrite(0x17,amDepth2);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chan[i].std.ex6.had) {
|
||||||
|
pmDepth2=chan[i].std.ex6.val;
|
||||||
|
immWrite(0x17,0x80|pmDepth2);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chan[i].std.ex7.had) {
|
||||||
|
lfoValue2=chan[i].std.ex7.val;
|
||||||
|
immWrite(0x16,lfoValue2);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chan[i].std.ex8.had) {
|
||||||
|
lfoShape2=chan[i].std.ex8.val&3;
|
||||||
|
immWrite(0x1b,lfoShape|(lfoShape2<<2));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (chan[i].std.alg.had) {
|
if (chan[i].std.alg.had) {
|
||||||
|
@ -286,6 +310,12 @@ void DivPlatformTX81Z::tick(bool sysTick) {
|
||||||
oldWrites[i]=pendingWrites[i];
|
oldWrites[i]=pendingWrites[i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for (int i=320; i<328; i++) {
|
||||||
|
if (pendingWrites[i]!=oldWrites[i]) {
|
||||||
|
immWrite(0x38+(i&7),(0x84|pendingWrites[i]));
|
||||||
|
oldWrites[i]=pendingWrites[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int hardResetElapsed=0;
|
int hardResetElapsed=0;
|
||||||
bool mustHardReset=false;
|
bool mustHardReset=false;
|
||||||
|
@ -405,7 +435,7 @@ void DivPlatformTX81Z::commitState(int ch, DivInstrument* ins) {
|
||||||
rWrite(chanOffs[ch]+ADDR_LR_FB_ALG,(chan[ch].state.alg&7)|(chan[ch].state.fb<<3)|((chan[ch].chVolL&1)<<6)|((chan[ch].chVolR&1)<<7));
|
rWrite(chanOffs[ch]+ADDR_LR_FB_ALG,(chan[ch].state.alg&7)|(chan[ch].state.fb<<3)|((chan[ch].chVolL&1)<<6)|((chan[ch].chVolR&1)<<7));
|
||||||
}*/
|
}*/
|
||||||
rWrite(chanOffs[ch]+ADDR_FMS_AMS,((chan[ch].state.fms&7)<<4)|(chan[ch].state.ams&3));
|
rWrite(chanOffs[ch]+ADDR_FMS_AMS,((chan[ch].state.fms&7)<<4)|(chan[ch].state.ams&3));
|
||||||
//rWrite(chanOffs[ch]+ADDR_FMS_AMS,0x84|((chan[ch].state.fms2&7)<<4)|(chan[ch].state.ams2&3));
|
rWrite(chanOffs[ch]+ADDR_FMS2_AMS2,((chan[ch].state.fms2&7)<<4)|(chan[ch].state.ams2&3));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -528,11 +558,23 @@ int DivPlatformTX81Z::dispatch(DivCommand c) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case DIV_CMD_FM_LFO: {
|
case DIV_CMD_FM_LFO: {
|
||||||
rWrite(0x18,c.value);
|
lfoValue=c.value;
|
||||||
|
immWrite(0x18,lfoValue);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case DIV_CMD_FM_LFO_WAVE: {
|
case DIV_CMD_FM_LFO_WAVE: {
|
||||||
rWrite(0x1b,c.value&3);
|
lfoShape=c.value&3;
|
||||||
|
immWrite(0x1b,lfoShape|(lfoShape2<<2));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DIV_CMD_FM_LFO2: {
|
||||||
|
lfoValue2=c.value;
|
||||||
|
immWrite(0x16,lfoValue2);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DIV_CMD_FM_LFO2_WAVE: {
|
||||||
|
lfoShape2=c.value&3;
|
||||||
|
immWrite(0x1b,lfoShape|(lfoShape2<<2));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case DIV_CMD_FM_FB: {
|
case DIV_CMD_FM_FB: {
|
||||||
|
@ -810,6 +852,16 @@ int DivPlatformTX81Z::dispatch(DivCommand c) {
|
||||||
immWrite(0x19,0x80|pmDepth);
|
immWrite(0x19,0x80|pmDepth);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case DIV_CMD_FM_AM2_DEPTH: {
|
||||||
|
amDepth2=c.value;
|
||||||
|
immWrite(0x17,amDepth);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case DIV_CMD_FM_PM2_DEPTH: {
|
||||||
|
pmDepth2=c.value;
|
||||||
|
immWrite(0x17,0x80|pmDepth);
|
||||||
|
break;
|
||||||
|
}
|
||||||
case DIV_CMD_FM_HARD_RESET:
|
case DIV_CMD_FM_HARD_RESET:
|
||||||
chan[c.chan].hardReset=c.value;
|
chan[c.chan].hardReset=c.value;
|
||||||
break;
|
break;
|
||||||
|
@ -880,7 +932,7 @@ void DivPlatformTX81Z::forceIns() {
|
||||||
rWrite(chanOffs[i]+ADDR_LR_FB_ALG,(chan[i].state.alg&7)|(chan[i].state.fb<<3)|((chan[i].chVolL&1)<<6)|((chan[i].chVolR&1)<<7));
|
rWrite(chanOffs[i]+ADDR_LR_FB_ALG,(chan[i].state.alg&7)|(chan[i].state.fb<<3)|((chan[i].chVolL&1)<<6)|((chan[i].chVolR&1)<<7));
|
||||||
}*/
|
}*/
|
||||||
rWrite(chanOffs[i]+ADDR_FMS_AMS,((chan[i].state.fms&7)<<4)|(chan[i].state.ams&3));
|
rWrite(chanOffs[i]+ADDR_FMS_AMS,((chan[i].state.fms&7)<<4)|(chan[i].state.ams&3));
|
||||||
//rWrite(chanOffs[i]+ADDR_FMS_AMS,0x84|((chan[i].state.fms2&7)<<4)|(chan[i].state.ams2&3));
|
rWrite(chanOffs[i]+ADDR_FMS2_AMS2,((chan[i].state.fms2&7)<<4)|(chan[i].state.ams2&3));
|
||||||
if (chan[i].active) {
|
if (chan[i].active) {
|
||||||
chan[i].keyOn=true;
|
chan[i].keyOn=true;
|
||||||
chan[i].freqChanged=true;
|
chan[i].freqChanged=true;
|
||||||
|
@ -888,6 +940,11 @@ void DivPlatformTX81Z::forceIns() {
|
||||||
}
|
}
|
||||||
immWrite(0x19,amDepth);
|
immWrite(0x19,amDepth);
|
||||||
immWrite(0x19,0x80|pmDepth);
|
immWrite(0x19,0x80|pmDepth);
|
||||||
|
immWrite(0x17,amDepth2);
|
||||||
|
immWrite(0x17,0x80|pmDepth2);
|
||||||
|
immWrite(0x18,lfoValue);
|
||||||
|
immWrite(0x16,lfoValue2);
|
||||||
|
immWrite(0x1b,lfoShape|(lfoShape2<<2));
|
||||||
}
|
}
|
||||||
|
|
||||||
void DivPlatformTX81Z::notifyInsChange(int ins) {
|
void DivPlatformTX81Z::notifyInsChange(int ins) {
|
||||||
|
@ -958,12 +1015,19 @@ void DivPlatformTX81Z::reset() {
|
||||||
delay=0;
|
delay=0;
|
||||||
amDepth=0x7f;
|
amDepth=0x7f;
|
||||||
pmDepth=0x7f;
|
pmDepth=0x7f;
|
||||||
|
amDepth2=0x7f;
|
||||||
|
pmDepth2=0x7f;
|
||||||
|
lfoValue=0;
|
||||||
|
lfoValue2=0;
|
||||||
|
lfoShape=0;
|
||||||
|
lfoShape2=0;
|
||||||
|
|
||||||
//rWrite(0x18,0x10);
|
|
||||||
immWrite(0x18,0x00); // LFO Freq Off
|
immWrite(0x18,0x00); // LFO Freq Off
|
||||||
|
immWrite(0x16,0x00);
|
||||||
immWrite(0x19,amDepth);
|
immWrite(0x19,amDepth);
|
||||||
immWrite(0x19,0x80|pmDepth);
|
immWrite(0x19,0x80|pmDepth);
|
||||||
//rWrite(0x1b,0x00);
|
immWrite(0x17,amDepth2);
|
||||||
|
immWrite(0x17,0x80|pmDepth2);
|
||||||
|
|
||||||
extMode=false;
|
extMode=false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,7 +45,7 @@ class DivPlatformTX81Z: public DivPlatformOPM {
|
||||||
DivDispatchOscBuffer* oscBuf[8];
|
DivDispatchOscBuffer* oscBuf[8];
|
||||||
int baseFreqOff;
|
int baseFreqOff;
|
||||||
int pcmL, pcmR, pcmCycles;
|
int pcmL, pcmR, pcmCycles;
|
||||||
unsigned char amDepth, pmDepth;
|
unsigned char amDepth, pmDepth, amDepth2, pmDepth2;
|
||||||
|
|
||||||
ymfm::ym2414* fm_ymfm;
|
ymfm::ym2414* fm_ymfm;
|
||||||
ymfm::ym2414::output_data out_ymfm;
|
ymfm::ym2414::output_data out_ymfm;
|
||||||
|
|
|
@ -558,10 +558,10 @@ void DivPlatformYM2203Ext::forceIns() {
|
||||||
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
||||||
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
||||||
if (i==2 && extMode) { // extended channel
|
if (i==2 && extMode) { // extended channel
|
||||||
if (isOpMuted[j]) {
|
if (isOpMuted[orderedOps[j]]) {
|
||||||
rWrite(baseAddr+0x40,127);
|
rWrite(baseAddr+0x40,127);
|
||||||
} else if (KVS(i,j)) {
|
} else if (KVS(i,j)) {
|
||||||
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[j].outVol&0x7f,127));
|
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[orderedOps[j]].outVol&0x7f,127));
|
||||||
} else {
|
} else {
|
||||||
rWrite(baseAddr+0x40,op.tl);
|
rWrite(baseAddr+0x40,op.tl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -582,10 +582,10 @@ void DivPlatformYM2608Ext::forceIns() {
|
||||||
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
||||||
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
||||||
if (i==2 && extMode) { // extended channel
|
if (i==2 && extMode) { // extended channel
|
||||||
if (isOpMuted[j]) {
|
if (isOpMuted[orderedOps[j]]) {
|
||||||
rWrite(baseAddr+0x40,127);
|
rWrite(baseAddr+0x40,127);
|
||||||
} else if (KVS(i,j)) {
|
} else if (KVS(i,j)) {
|
||||||
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[j].outVol&0x7f,127));
|
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[orderedOps[j]].outVol&0x7f,127));
|
||||||
} else {
|
} else {
|
||||||
rWrite(baseAddr+0x40,op.tl);
|
rWrite(baseAddr+0x40,op.tl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -578,10 +578,10 @@ void DivPlatformYM2610BExt::forceIns() {
|
||||||
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
||||||
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
||||||
if (i==extChanOffs && extMode) { // extended channel
|
if (i==extChanOffs && extMode) { // extended channel
|
||||||
if (isOpMuted[j]) {
|
if (isOpMuted[orderedOps[j]]) {
|
||||||
rWrite(baseAddr+0x40,127);
|
rWrite(baseAddr+0x40,127);
|
||||||
} else if (KVS(i,j)) {
|
} else if (KVS(i,j)) {
|
||||||
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[j].outVol&0x7f,127));
|
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[orderedOps[j]].outVol&0x7f,127));
|
||||||
} else {
|
} else {
|
||||||
rWrite(baseAddr+0x40,op.tl);
|
rWrite(baseAddr+0x40,op.tl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -578,10 +578,10 @@ void DivPlatformYM2610Ext::forceIns() {
|
||||||
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
unsigned short baseAddr=chanOffs[i]|opOffs[j];
|
||||||
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
DivInstrumentFM::Operator& op=chan[i].state.op[j];
|
||||||
if (i==extChanOffs && extMode) { // extended channel
|
if (i==extChanOffs && extMode) { // extended channel
|
||||||
if (isOpMuted[j]) {
|
if (isOpMuted[orderedOps[j]]) {
|
||||||
rWrite(baseAddr+0x40,127);
|
rWrite(baseAddr+0x40,127);
|
||||||
} else if (KVS(i,j)) {
|
} else if (KVS(i,j)) {
|
||||||
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[j].outVol&0x7f,127));
|
rWrite(baseAddr+0x40,127-VOL_SCALE_LOG_BROKEN(127-op.tl,opChan[orderedOps[j]].outVol&0x7f,127));
|
||||||
} else {
|
} else {
|
||||||
rWrite(baseAddr+0x40,op.tl);
|
rWrite(baseAddr+0x40,op.tl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -103,9 +103,8 @@ const char* cmdName[]={
|
||||||
"FM_AM_DEPTH",
|
"FM_AM_DEPTH",
|
||||||
"FM_PM_DEPTH",
|
"FM_PM_DEPTH",
|
||||||
|
|
||||||
"GENESIS_LFO",
|
"FM_LFO2",
|
||||||
|
"FM_LFO2_WAVE",
|
||||||
"ARCADE_LFO",
|
|
||||||
|
|
||||||
"STD_NOISE_FREQ",
|
"STD_NOISE_FREQ",
|
||||||
"STD_NOISE_MODE",
|
"STD_NOISE_MODE",
|
||||||
|
@ -227,6 +226,9 @@ const char* cmdName[]={
|
||||||
|
|
||||||
"SURROUND_PANNING",
|
"SURROUND_PANNING",
|
||||||
|
|
||||||
|
"FM_AM2_DEPTH",
|
||||||
|
"FM_PM2_DEPTH",
|
||||||
|
|
||||||
"ALWAYS_SET_VOLUME"
|
"ALWAYS_SET_VOLUME"
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -699,6 +701,13 @@ void DivEngine::processRow(int i, bool afterDelay) {
|
||||||
// - then a volume slide down starts to the low boundary, and then when this is reached a volume slide up begins
|
// - then a volume slide down starts to the low boundary, and then when this is reached a volume slide up begins
|
||||||
// - this process repeats until 0700 or 0Axy are found
|
// - this process repeats until 0700 or 0Axy are found
|
||||||
// - note that a volume value does not stop tremolo - instead it glitches this whole thing up
|
// - note that a volume value does not stop tremolo - instead it glitches this whole thing up
|
||||||
|
if (chan[i].tremoloDepth==0) {
|
||||||
|
chan[i].tremoloPos=0;
|
||||||
|
}
|
||||||
|
chan[i].tremoloDepth=effectVal&15;
|
||||||
|
chan[i].tremoloRate=effectVal>>4;
|
||||||
|
// tremolo and vol slides are incompatiblw
|
||||||
|
chan[i].volSpeed=0;
|
||||||
break;
|
break;
|
||||||
case 0x0a: // volume ramp
|
case 0x0a: // volume ramp
|
||||||
// TODO: non-0x-or-x0 value should be treated as 00
|
// TODO: non-0x-or-x0 value should be treated as 00
|
||||||
|
@ -708,6 +717,9 @@ void DivEngine::processRow(int i, bool afterDelay) {
|
||||||
} else {
|
} else {
|
||||||
chan[i].volSpeed=(effectVal>>4)*64;
|
chan[i].volSpeed=(effectVal>>4)*64;
|
||||||
}
|
}
|
||||||
|
// tremolo and vol slides are incompatible
|
||||||
|
chan[i].tremoloDepth=0;
|
||||||
|
chan[i].tremoloRate=0;
|
||||||
} else {
|
} else {
|
||||||
chan[i].volSpeed=0;
|
chan[i].volSpeed=0;
|
||||||
}
|
}
|
||||||
|
@ -848,10 +860,16 @@ void DivEngine::processRow(int i, bool afterDelay) {
|
||||||
if (!song.arpNonPorta) dispatchCmd(DivCommand(DIV_CMD_PRE_PORTA,i,false,0));
|
if (!song.arpNonPorta) dispatchCmd(DivCommand(DIV_CMD_PRE_PORTA,i,false,0));
|
||||||
break;
|
break;
|
||||||
case 0xf3: // fine volume ramp up
|
case 0xf3: // fine volume ramp up
|
||||||
|
// tremolo and vol slides are incompatible
|
||||||
|
chan[i].tremoloDepth=0;
|
||||||
|
chan[i].tremoloRate=0;
|
||||||
chan[i].volSpeed=effectVal;
|
chan[i].volSpeed=effectVal;
|
||||||
dispatchCmd(DivCommand(DIV_CMD_HINT_VOL_SLIDE,i,chan[i].volSpeed));
|
dispatchCmd(DivCommand(DIV_CMD_HINT_VOL_SLIDE,i,chan[i].volSpeed));
|
||||||
break;
|
break;
|
||||||
case 0xf4: // fine volume ramp down
|
case 0xf4: // fine volume ramp down
|
||||||
|
// tremolo and vol slides are incompatible
|
||||||
|
chan[i].tremoloDepth=0;
|
||||||
|
chan[i].tremoloRate=0;
|
||||||
chan[i].volSpeed=-effectVal;
|
chan[i].volSpeed=-effectVal;
|
||||||
dispatchCmd(DivCommand(DIV_CMD_HINT_VOL_SLIDE,i,chan[i].volSpeed));
|
dispatchCmd(DivCommand(DIV_CMD_HINT_VOL_SLIDE,i,chan[i].volSpeed));
|
||||||
break;
|
break;
|
||||||
|
@ -878,6 +896,9 @@ void DivEngine::processRow(int i, bool afterDelay) {
|
||||||
} else {
|
} else {
|
||||||
chan[i].volSpeed=(effectVal>>4)*256;
|
chan[i].volSpeed=(effectVal>>4)*256;
|
||||||
}
|
}
|
||||||
|
// tremolo and vol slides are incompatible
|
||||||
|
chan[i].tremoloDepth=0;
|
||||||
|
chan[i].tremoloRate=0;
|
||||||
} else {
|
} else {
|
||||||
chan[i].volSpeed=0;
|
chan[i].volSpeed=0;
|
||||||
}
|
}
|
||||||
|
@ -1249,6 +1270,10 @@ bool DivEngine::nextTick(bool noAccum, bool inhibitLowLat) {
|
||||||
} else {
|
} else {
|
||||||
dispatchCmd(DivCommand(DIV_CMD_VOLUME,i,chan[i].volume>>8));
|
dispatchCmd(DivCommand(DIV_CMD_VOLUME,i,chan[i].volume>>8));
|
||||||
}
|
}
|
||||||
|
} else if (chan[i].tremoloDepth>0) {
|
||||||
|
chan[i].tremoloPos+=chan[i].tremoloRate;
|
||||||
|
chan[i].tremoloPos&=127;
|
||||||
|
dispatchCmd(DivCommand(DIV_CMD_VOLUME,i,MAX(0,chan[i].volume-(tremTable[chan[i].tremoloPos]*chan[i].tremoloDepth))>>8));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (chan[i].vibratoDepth>0) {
|
if (chan[i].vibratoDepth>0) {
|
||||||
|
|
|
@ -512,6 +512,10 @@ void DivEngine::registerSystems() {
|
||||||
|
|
||||||
EffectHandlerMap fmOPZPostEffectHandlerMap(fmOPMPostEffectHandlerMap);
|
EffectHandlerMap fmOPZPostEffectHandlerMap(fmOPMPostEffectHandlerMap);
|
||||||
fmOPZPostEffectHandlerMap.insert({
|
fmOPZPostEffectHandlerMap.insert({
|
||||||
|
{0x24, {DIV_CMD_FM_LFO2, "24xx: Set LFO 2 speed"}},
|
||||||
|
{0x25, {DIV_CMD_FM_LFO2_WAVE, "25xx: Set LFO 2 waveform (0 saw, 1 square, 2 triangle, 3 noise)"}},
|
||||||
|
{0x26, {DIV_CMD_FM_AM2_DEPTH, "26xx: Set AM 2 depth (0 to 7F)", effectValAnd<127>}},
|
||||||
|
{0x27, {DIV_CMD_FM_PM2_DEPTH, "27xx: Set PM 2 depth (0 to 7F)", effectValAnd<127>}},
|
||||||
{0x28, {DIV_CMD_FM_REV, "28xy: Set reverb (x: operator from 1 to 4 (0 for all ops); y: reverb from 0 to 7)", effectOpVal<4>, effectValAnd<7>}},
|
{0x28, {DIV_CMD_FM_REV, "28xy: Set reverb (x: operator from 1 to 4 (0 for all ops); y: reverb from 0 to 7)", effectOpVal<4>, effectValAnd<7>}},
|
||||||
{0x2a, {DIV_CMD_FM_WS, "2Axy: Set waveform (x: operator from 1 to 4 (0 for all ops); y: waveform from 0 to 7)", effectOpVal<4>, effectValAnd<7>}},
|
{0x2a, {DIV_CMD_FM_WS, "2Axy: Set waveform (x: operator from 1 to 4 (0 for all ops); y: waveform from 0 to 7)", effectOpVal<4>, effectValAnd<7>}},
|
||||||
{0x2b, {DIV_CMD_FM_EG_SHIFT, "2Bxy: Set envelope generator shift (x: operator from 1 to 4 (0 for all ops); y: shift from 0 to 3)", effectOpVal<4>, effectValAnd<3>}},
|
{0x2b, {DIV_CMD_FM_EG_SHIFT, "2Bxy: Set envelope generator shift (x: operator from 1 to 4 (0 for all ops); y: shift from 0 to 3)", effectOpVal<4>, effectValAnd<3>}},
|
||||||
|
|
|
@ -29,6 +29,7 @@ void DivEngine::performVGMWrite(SafeWriter* w, DivSystem sys, DivRegWrite& write
|
||||||
unsigned char baseAddr2=isSecond?0x80:0;
|
unsigned char baseAddr2=isSecond?0x80:0;
|
||||||
unsigned short baseAddr2S=isSecond?0x8000:0;
|
unsigned short baseAddr2S=isSecond?0x8000:0;
|
||||||
unsigned char smsAddr=isSecond?0x30:0x50;
|
unsigned char smsAddr=isSecond?0x30:0x50;
|
||||||
|
unsigned char ggAddr=isSecond?0x3f:0x4f;
|
||||||
unsigned char rf5c68Addr=isSecond?0xb1:0xb0;
|
unsigned char rf5c68Addr=isSecond?0xb1:0xb0;
|
||||||
if (write.addr==0xffffffff) { // Furnace fake reset
|
if (write.addr==0xffffffff) { // Furnace fake reset
|
||||||
switch (sys) {
|
switch (sys) {
|
||||||
|
@ -647,7 +648,11 @@ void DivEngine::performVGMWrite(SafeWriter* w, DivSystem sys, DivRegWrite& write
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case DIV_SYSTEM_SMS:
|
case DIV_SYSTEM_SMS:
|
||||||
w->writeC(smsAddr);
|
if (write.addr==1) {
|
||||||
|
w->writeC(ggAddr);
|
||||||
|
} else {
|
||||||
|
w->writeC(smsAddr);
|
||||||
|
}
|
||||||
w->writeC(write.val);
|
w->writeC(write.val);
|
||||||
break;
|
break;
|
||||||
case DIV_SYSTEM_T6W28:
|
case DIV_SYSTEM_T6W28:
|
||||||
|
|
|
@ -75,6 +75,7 @@ const char* aboutLine[]={
|
||||||
"Dippy",
|
"Dippy",
|
||||||
"djtuBIG-MaliceX",
|
"djtuBIG-MaliceX",
|
||||||
"dumbut",
|
"dumbut",
|
||||||
|
"EpicTyphlosion",
|
||||||
"FΛDE",
|
"FΛDE",
|
||||||
"Forte",
|
"Forte",
|
||||||
"Fragmare",
|
"Fragmare",
|
||||||
|
@ -92,6 +93,7 @@ const char* aboutLine[]={
|
||||||
"LVintageNerd",
|
"LVintageNerd",
|
||||||
"Mahbod Karamoozian",
|
"Mahbod Karamoozian",
|
||||||
"Martin Demsky",
|
"Martin Demsky",
|
||||||
|
"MelonadeM",
|
||||||
"Miker",
|
"Miker",
|
||||||
"nicco1690",
|
"nicco1690",
|
||||||
"<nk>",
|
"<nk>",
|
||||||
|
|
|
@ -50,8 +50,10 @@ void FurnaceGUI::startSelection(int xCoarse, int xFine, int y, bool fullRow) {
|
||||||
selecting=true;
|
selecting=true;
|
||||||
selectingFull=false;
|
selectingFull=false;
|
||||||
dragSourceX=xCoarse;
|
dragSourceX=xCoarse;
|
||||||
|
dragSourceXFine=xFine;
|
||||||
dragSourceY=y;
|
dragSourceY=y;
|
||||||
dragDestinationX=xCoarse;
|
dragDestinationX=xCoarse;
|
||||||
|
dragDestinationXFine=xFine;
|
||||||
dragDestinationY=y;
|
dragDestinationY=y;
|
||||||
dragStart=selStart;
|
dragStart=selStart;
|
||||||
dragEnd=selEnd;
|
dragEnd=selEnd;
|
||||||
|
@ -86,6 +88,7 @@ void FurnaceGUI::updateSelection(int xCoarse, int xFine, int y, bool fullRow) {
|
||||||
if (!selecting) return;
|
if (!selecting) return;
|
||||||
if (dragging) {
|
if (dragging) {
|
||||||
dragDestinationX=xCoarse;
|
dragDestinationX=xCoarse;
|
||||||
|
if (dragStart.xFine>=3 && dragStart.xCoarse==dragEnd.xCoarse) dragDestinationXFine=(dragSourceXFine&1)?((xFine-1)|1):((xFine+1)&(~1));
|
||||||
dragDestinationY=y;
|
dragDestinationY=y;
|
||||||
cursorDrag.xCoarse=xCoarse;
|
cursorDrag.xCoarse=xCoarse;
|
||||||
cursorDrag.xFine=xFine;
|
cursorDrag.xFine=xFine;
|
||||||
|
@ -104,6 +107,15 @@ void FurnaceGUI::updateSelection(int xCoarse, int xFine, int y, bool fullRow) {
|
||||||
dragDestinationX=lastChannel-(dragEnd.xCoarse-dragSourceX)-1;
|
dragDestinationX=lastChannel-(dragEnd.xCoarse-dragSourceX)-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (dragStart.xFine>=3 && dragStart.xCoarse==dragEnd.xCoarse) {
|
||||||
|
if (dragEnd.xFine+(dragDestinationXFine-dragSourceXFine)>(2+e->curPat[dragDestinationX].effectCols*2)) {
|
||||||
|
dragDestinationXFine=(2+e->curPat[dragDestinationX].effectCols*2)-dragEnd.xFine+dragSourceXFine;
|
||||||
|
}
|
||||||
|
if (dragStart.xFine+(dragDestinationXFine-dragSourceXFine)<3) {
|
||||||
|
dragDestinationXFine=3-dragStart.xFine+dragSourceXFine;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (dragStart.y+(dragDestinationY-dragSourceY)<0) {
|
if (dragStart.y+(dragDestinationY-dragSourceY)<0) {
|
||||||
dragDestinationY=dragSourceY-dragStart.y;
|
dragDestinationY=dragSourceY-dragStart.y;
|
||||||
}
|
}
|
||||||
|
@ -113,10 +125,10 @@ void FurnaceGUI::updateSelection(int xCoarse, int xFine, int y, bool fullRow) {
|
||||||
}
|
}
|
||||||
|
|
||||||
selStart.xCoarse=dragStart.xCoarse+(dragDestinationX-dragSourceX);
|
selStart.xCoarse=dragStart.xCoarse+(dragDestinationX-dragSourceX);
|
||||||
selStart.xFine=dragStart.xFine;
|
selStart.xFine=dragStart.xFine+(dragDestinationXFine-dragSourceXFine);
|
||||||
selStart.y=dragStart.y+(dragDestinationY-dragSourceY);
|
selStart.y=dragStart.y+(dragDestinationY-dragSourceY);
|
||||||
selEnd.xCoarse=dragEnd.xCoarse+(dragDestinationX-dragSourceX);
|
selEnd.xCoarse=dragEnd.xCoarse+(dragDestinationX-dragSourceX);
|
||||||
selEnd.xFine=dragEnd.xFine;
|
selEnd.xFine=dragEnd.xFine+(dragDestinationXFine-dragSourceXFine);
|
||||||
selEnd.y=dragEnd.y+(dragDestinationY-dragSourceY);
|
selEnd.y=dragEnd.y+(dragDestinationY-dragSourceY);
|
||||||
} else {
|
} else {
|
||||||
if (selectingFull) {
|
if (selectingFull) {
|
||||||
|
@ -156,7 +168,7 @@ void FurnaceGUI::finishSelection() {
|
||||||
selectingFull=false;
|
selectingFull=false;
|
||||||
|
|
||||||
if (dragging) {
|
if (dragging) {
|
||||||
if (dragSourceX==dragDestinationX && dragSourceY==dragDestinationY) {
|
if (dragSourceX==dragDestinationX && dragSourceY==dragDestinationY && dragSourceXFine==dragDestinationXFine) {
|
||||||
cursor=cursorDrag;
|
cursor=cursorDrag;
|
||||||
selStart=cursorDrag;
|
selStart=cursorDrag;
|
||||||
selEnd=cursorDrag;
|
selEnd=cursorDrag;
|
||||||
|
|
|
@ -193,6 +193,9 @@ void FurnaceGUI::doAction(int what) {
|
||||||
case GUI_ACTION_WINDOW_SONG_INFO:
|
case GUI_ACTION_WINDOW_SONG_INFO:
|
||||||
nextWindow=GUI_WINDOW_SONG_INFO;
|
nextWindow=GUI_WINDOW_SONG_INFO;
|
||||||
break;
|
break;
|
||||||
|
case GUI_ACTION_WINDOW_SPEED:
|
||||||
|
nextWindow=GUI_WINDOW_SPEED;
|
||||||
|
break;
|
||||||
case GUI_ACTION_WINDOW_PATTERN:
|
case GUI_ACTION_WINDOW_PATTERN:
|
||||||
nextWindow=GUI_WINDOW_PATTERN;
|
nextWindow=GUI_WINDOW_PATTERN;
|
||||||
break;
|
break;
|
||||||
|
@ -274,6 +277,9 @@ void FurnaceGUI::doAction(int what) {
|
||||||
case GUI_WINDOW_SONG_INFO:
|
case GUI_WINDOW_SONG_INFO:
|
||||||
songInfoOpen=false;
|
songInfoOpen=false;
|
||||||
break;
|
break;
|
||||||
|
case GUI_WINDOW_SPEED:
|
||||||
|
speedOpen=false;
|
||||||
|
break;
|
||||||
case GUI_WINDOW_ORDERS:
|
case GUI_WINDOW_ORDERS:
|
||||||
ordersOpen=false;
|
ordersOpen=false;
|
||||||
break;
|
break;
|
||||||
|
@ -386,10 +392,10 @@ void FurnaceGUI::doAction(int what) {
|
||||||
doSelectAll();
|
doSelectAll();
|
||||||
break;
|
break;
|
||||||
case GUI_ACTION_PAT_CUT:
|
case GUI_ACTION_PAT_CUT:
|
||||||
doCopy(true);
|
doCopy(true,true,selStart,selEnd);
|
||||||
break;
|
break;
|
||||||
case GUI_ACTION_PAT_COPY:
|
case GUI_ACTION_PAT_COPY:
|
||||||
doCopy(false);
|
doCopy(false,true,selStart,selEnd);
|
||||||
break;
|
break;
|
||||||
case GUI_ACTION_PAT_PASTE:
|
case GUI_ACTION_PAT_PASTE:
|
||||||
doPaste();
|
doPaste();
|
||||||
|
|
|
@ -358,66 +358,75 @@ void FurnaceGUI::doTranspose(int amount, OperationMask& mask) {
|
||||||
makeUndo(GUI_UNDO_PATTERN_DELETE);
|
makeUndo(GUI_UNDO_PATTERN_DELETE);
|
||||||
}
|
}
|
||||||
|
|
||||||
void FurnaceGUI::doCopy(bool cut) {
|
String FurnaceGUI::doCopy(bool cut, bool writeClipboard, const SelectionPoint& sStart, const SelectionPoint& sEnd) {
|
||||||
finishSelection();
|
if (writeClipboard) {
|
||||||
if (cut) {
|
finishSelection();
|
||||||
curNibble=false;
|
if (cut) {
|
||||||
prepareUndo(GUI_UNDO_PATTERN_CUT);
|
curNibble=false;
|
||||||
|
prepareUndo(GUI_UNDO_PATTERN_CUT);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
clipboard=fmt::sprintf("org.tildearrow.furnace - Pattern Data (%d)\n%d",DIV_ENGINE_VERSION,selStart.xFine);
|
String clipb=fmt::sprintf("org.tildearrow.furnace - Pattern Data (%d)\n%d",DIV_ENGINE_VERSION,sStart.xFine);
|
||||||
|
|
||||||
for (int j=selStart.y; j<=selEnd.y; j++) {
|
for (int j=sStart.y; j<=sEnd.y; j++) {
|
||||||
int iCoarse=selStart.xCoarse;
|
int iCoarse=sStart.xCoarse;
|
||||||
int iFine=selStart.xFine;
|
int iFine=sStart.xFine;
|
||||||
if (iFine>3 && !(iFine&1)) {
|
if (iFine>3 && !(iFine&1)) {
|
||||||
iFine--;
|
iFine--;
|
||||||
}
|
}
|
||||||
clipboard+='\n';
|
clipb+='\n';
|
||||||
for (; iCoarse<=selEnd.xCoarse; iCoarse++) {
|
for (; iCoarse<=sEnd.xCoarse; iCoarse++) {
|
||||||
if (!e->curSubSong->chanShow[iCoarse]) continue;
|
if (!e->curSubSong->chanShow[iCoarse]) continue;
|
||||||
DivPattern* pat=e->curPat[iCoarse].getPattern(e->curOrders->ord[iCoarse][curOrder],true);
|
DivPattern* pat=e->curPat[iCoarse].getPattern(e->curOrders->ord[iCoarse][curOrder],true);
|
||||||
for (; iFine<3+e->curPat[iCoarse].effectCols*2 && (iCoarse<selEnd.xCoarse || iFine<=selEnd.xFine); iFine++) {
|
for (; iFine<3+e->curPat[iCoarse].effectCols*2 && (iCoarse<sEnd.xCoarse || iFine<=sEnd.xFine); iFine++) {
|
||||||
if (iFine==0) {
|
if (iFine==0) {
|
||||||
clipboard+=noteNameNormal(pat->data[j][0],pat->data[j][1]);
|
clipb+=noteNameNormal(pat->data[j][0],pat->data[j][1]);
|
||||||
if (cut) {
|
if (cut) {
|
||||||
pat->data[j][0]=0;
|
pat->data[j][0]=0;
|
||||||
pat->data[j][1]=0;
|
pat->data[j][1]=0;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (pat->data[j][iFine+1]==-1) {
|
if (pat->data[j][iFine+1]==-1) {
|
||||||
clipboard+="..";
|
clipb+="..";
|
||||||
} else {
|
} else {
|
||||||
clipboard+=fmt::sprintf("%.2X",pat->data[j][iFine+1]);
|
clipb+=fmt::sprintf("%.2X",pat->data[j][iFine+1]);
|
||||||
}
|
}
|
||||||
if (cut) {
|
if (cut) {
|
||||||
pat->data[j][iFine+1]=-1;
|
pat->data[j][iFine+1]=-1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
clipboard+='|';
|
clipb+='|';
|
||||||
iFine=0;
|
iFine=0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SDL_SetClipboardText(clipboard.c_str());
|
|
||||||
|
|
||||||
if (cut) {
|
if (writeClipboard) {
|
||||||
makeUndo(GUI_UNDO_PATTERN_CUT);
|
SDL_SetClipboardText(clipb.c_str());
|
||||||
|
if (cut) {
|
||||||
|
makeUndo(GUI_UNDO_PATTERN_CUT);
|
||||||
|
}
|
||||||
|
clipboard=clipb;
|
||||||
}
|
}
|
||||||
|
return clipb;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FurnaceGUI::doPaste(PasteMode mode, int arg) {
|
void FurnaceGUI::doPaste(PasteMode mode, int arg, bool readClipboard, String clipb) {
|
||||||
finishSelection();
|
if (readClipboard) {
|
||||||
prepareUndo(GUI_UNDO_PATTERN_PASTE);
|
finishSelection();
|
||||||
char* clipText=SDL_GetClipboardText();
|
prepareUndo(GUI_UNDO_PATTERN_PASTE);
|
||||||
if (clipText!=NULL) {
|
char* clipText=SDL_GetClipboardText();
|
||||||
if (clipText[0]) {
|
if (clipText!=NULL) {
|
||||||
clipboard=clipText;
|
if (clipText[0]) {
|
||||||
|
clipboard=clipText;
|
||||||
|
}
|
||||||
|
SDL_free(clipText);
|
||||||
}
|
}
|
||||||
SDL_free(clipText);
|
clipb=clipboard;
|
||||||
}
|
}
|
||||||
std::vector<String> data;
|
std::vector<String> data;
|
||||||
String tempS;
|
String tempS;
|
||||||
for (char i: clipboard) {
|
for (char i: clipb) {
|
||||||
if (i=='\r') continue;
|
if (i=='\r') continue;
|
||||||
if (i=='\n') {
|
if (i=='\n') {
|
||||||
data.push_back(tempS);
|
data.push_back(tempS);
|
||||||
|
@ -562,15 +571,18 @@ void FurnaceGUI::doPaste(PasteMode mode, int arg) {
|
||||||
i=1;
|
i=1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (settings.cursorPastePos) {
|
|
||||||
cursor.y=j;
|
|
||||||
if (cursor.y>=e->curSubSong->patLen) cursor.y=e->curSubSong->patLen-1;
|
|
||||||
selStart=cursor;
|
|
||||||
selEnd=cursor;
|
|
||||||
updateScroll(cursor.y);
|
|
||||||
}
|
|
||||||
|
|
||||||
makeUndo(GUI_UNDO_PATTERN_PASTE);
|
if (readClipboard) {
|
||||||
|
if (settings.cursorPastePos) {
|
||||||
|
cursor.y=j;
|
||||||
|
if (cursor.y>=e->curSubSong->patLen) cursor.y=e->curSubSong->patLen-1;
|
||||||
|
selStart=cursor;
|
||||||
|
selEnd=cursor;
|
||||||
|
updateScroll(cursor.y);
|
||||||
|
}
|
||||||
|
|
||||||
|
makeUndo(GUI_UNDO_PATTERN_PASTE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FurnaceGUI::doChangeIns(int ins) {
|
void FurnaceGUI::doChangeIns(int ins) {
|
||||||
|
@ -926,72 +938,21 @@ void FurnaceGUI::doExpand(int multiplier) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void FurnaceGUI::doDrag() {
|
void FurnaceGUI::doDrag() {
|
||||||
DivPattern* patBuffer=NULL;
|
|
||||||
int len=dragEnd.xCoarse-dragStart.xCoarse+1;
|
int len=dragEnd.xCoarse-dragStart.xCoarse+1;
|
||||||
|
|
||||||
DETERMINE_FIRST_LAST;
|
|
||||||
|
|
||||||
if (len<1) return;
|
if (len<1) return;
|
||||||
|
|
||||||
patBuffer=new DivPattern[len];
|
|
||||||
prepareUndo(GUI_UNDO_PATTERN_DRAG);
|
prepareUndo(GUI_UNDO_PATTERN_DRAG);
|
||||||
|
|
||||||
// copy and clear
|
// copy and clear
|
||||||
{
|
String c=doCopy(true,false,dragStart,dragEnd);
|
||||||
int iCoarse=dragStart.xCoarse;
|
|
||||||
int iFine=dragStart.xFine;
|
|
||||||
int iCoarseP=0;
|
|
||||||
for (; iCoarse<=dragEnd.xCoarse; iCoarse++) {
|
|
||||||
if (!e->curSubSong->chanShow[iCoarse]) continue;
|
|
||||||
DivPattern* pat=e->curPat[iCoarse].getPattern(e->curOrders->ord[iCoarse][curOrder],true);
|
|
||||||
for (; iFine<3+e->curPat[iCoarse].effectCols*2 && (iCoarse<dragEnd.xCoarse || iFine<=dragEnd.xFine); iFine++) {
|
|
||||||
int row=0;
|
|
||||||
for (int j=dragStart.y; j<=dragEnd.y; j++) {
|
|
||||||
if (iFine==0) {
|
|
||||||
patBuffer[iCoarseP].data[row][iFine]=pat->data[j][iFine];
|
|
||||||
pat->data[j][iFine]=0;
|
|
||||||
if (dragStart.y==dragEnd.y) pat->data[j][2]=-1;
|
|
||||||
}
|
|
||||||
patBuffer[iCoarseP].data[row][iFine+1]=pat->data[j][iFine+1];
|
|
||||||
pat->data[j][iFine+1]=(iFine<1)?0:-1;
|
|
||||||
|
|
||||||
if (dragStart.y==dragEnd.y && iFine>2 && iFine&1 && settings.effectDeletionAltersValue) {
|
logV("copy: %s",c);
|
||||||
pat->data[j][iFine+2]=-1;
|
|
||||||
}
|
|
||||||
row++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
iFine=0;
|
|
||||||
iCoarseP++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// replace
|
// replace
|
||||||
{
|
cursor=selStart;
|
||||||
int iCoarse=selStart.xCoarse;
|
doPaste(GUI_PASTE_MODE_NORMAL,0,false,c);
|
||||||
int iFine=selStart.xFine;
|
|
||||||
int iCoarseP=0;
|
|
||||||
for (; iCoarse<=selEnd.xCoarse && iCoarseP<len; iCoarse++) {
|
|
||||||
if (iCoarse<firstChannel || iCoarse>lastChannel) continue;
|
|
||||||
if (!e->curSubSong->chanShow[iCoarse]) continue;
|
|
||||||
DivPattern* pat=e->curPat[iCoarse].getPattern(e->curOrders->ord[iCoarse][curOrder],true);
|
|
||||||
for (; iFine<3+e->curPat[iCoarse].effectCols*2 && (iCoarse<selEnd.xCoarse || iFine<=selEnd.xFine); iFine++) {
|
|
||||||
int row=-1;
|
|
||||||
for (int j=selStart.y; j<=selEnd.y; j++) {
|
|
||||||
row++;
|
|
||||||
if (j<0 || j>=e->curSubSong->patLen) continue;
|
|
||||||
if (iFine==0) {
|
|
||||||
pat->data[j][iFine]=patBuffer[iCoarseP].data[row][iFine];
|
|
||||||
}
|
|
||||||
pat->data[j][iFine+1]=patBuffer[iCoarseP].data[row][iFine+1];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
iFine=0;
|
|
||||||
iCoarseP++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
delete[] patBuffer;
|
|
||||||
makeUndo(GUI_UNDO_PATTERN_DRAG);
|
makeUndo(GUI_UNDO_PATTERN_DRAG);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1427,7 +1427,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
hasOpened=fileDialog->openLoad(
|
hasOpened=fileDialog->openLoad(
|
||||||
"Open File",
|
"Open File",
|
||||||
{"compatible files", "*.fur *.dmf *.mod *.fc13 *.fc14 *.smod *.fc",
|
{"compatible files", "*.fur *.dmf *.mod *.fc13 *.fc14 *.smod *.fc",
|
||||||
"all files", ".*"},
|
"all files", "*"},
|
||||||
"compatible files{.fur,.dmf,.mod,.fc13,.fc14,.smod,.fc},.*",
|
"compatible files{.fur,.dmf,.mod,.fc13,.fc14,.smod,.fc},.*",
|
||||||
workingDirSong,
|
workingDirSong,
|
||||||
dpiScale
|
dpiScale
|
||||||
|
@ -1493,7 +1493,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
"VOPM preset bank", "*.opm",
|
"VOPM preset bank", "*.opm",
|
||||||
"Wohlstand WOPL bank", "*.wopl",
|
"Wohlstand WOPL bank", "*.wopl",
|
||||||
"Wohlstand WOPN bank", "*.wopn",
|
"Wohlstand WOPN bank", "*.wopn",
|
||||||
"all files", ".*"},
|
"all files", "*"},
|
||||||
"all compatible files{.fui,.dmp,.tfi,.vgi,.s3i,.sbi,.opli,.opni,.y12,.bnk,.ff,.gyb,.opm,.wopl,.wopn},.*",
|
"all compatible files{.fui,.dmp,.tfi,.vgi,.s3i,.sbi,.opli,.opni,.y12,.bnk,.ff,.gyb,.opm,.wopl,.wopn},.*",
|
||||||
workingDirIns,
|
workingDirIns,
|
||||||
dpiScale,
|
dpiScale,
|
||||||
|
@ -1560,7 +1560,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
hasOpened=fileDialog->openLoad(
|
hasOpened=fileDialog->openLoad(
|
||||||
"Load Wavetable",
|
"Load Wavetable",
|
||||||
{"compatible files", "*.fuw *.dmw",
|
{"compatible files", "*.fuw *.dmw",
|
||||||
"all files", ".*"},
|
"all files", "*"},
|
||||||
"compatible files{.fuw,.dmw},.*",
|
"compatible files{.fuw,.dmw},.*",
|
||||||
workingDirWave,
|
workingDirWave,
|
||||||
dpiScale,
|
dpiScale,
|
||||||
|
@ -1604,7 +1604,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
hasOpened=fileDialog->openLoad(
|
hasOpened=fileDialog->openLoad(
|
||||||
"Load Sample",
|
"Load Sample",
|
||||||
{"compatible files", "*.wav *.dmc *.brr",
|
{"compatible files", "*.wav *.dmc *.brr",
|
||||||
"all files", ".*"},
|
"all files", "*"},
|
||||||
"compatible files{.wav,.dmc,.brr},.*",
|
"compatible files{.wav,.dmc,.brr},.*",
|
||||||
workingDirSample,
|
workingDirSample,
|
||||||
dpiScale,
|
dpiScale,
|
||||||
|
@ -1617,7 +1617,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
if (!dirExists(workingDirSample)) workingDirSample=getHomeDir();
|
if (!dirExists(workingDirSample)) workingDirSample=getHomeDir();
|
||||||
hasOpened=fileDialog->openLoad(
|
hasOpened=fileDialog->openLoad(
|
||||||
"Load Raw Sample",
|
"Load Raw Sample",
|
||||||
{"all files", ".*"},
|
{"all files", "*"},
|
||||||
".*",
|
".*",
|
||||||
workingDirSample,
|
workingDirSample,
|
||||||
dpiScale
|
dpiScale
|
||||||
|
@ -1784,7 +1784,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
hasOpened=fileDialog->openLoad(
|
hasOpened=fileDialog->openLoad(
|
||||||
"Load ROM",
|
"Load ROM",
|
||||||
{"compatible files", "*.rom *.bin",
|
{"compatible files", "*.rom *.bin",
|
||||||
"all files", ".*"},
|
"all files", "*"},
|
||||||
"compatible files{.rom,.bin},.*",
|
"compatible files{.rom,.bin},.*",
|
||||||
workingDirROM,
|
workingDirROM,
|
||||||
dpiScale
|
dpiScale
|
||||||
|
@ -1796,7 +1796,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
"Open Test",
|
"Open Test",
|
||||||
{"compatible files", "*.fur *.dmf *.mod",
|
{"compatible files", "*.fur *.dmf *.mod",
|
||||||
"another option", "*.wav *.ttf",
|
"another option", "*.wav *.ttf",
|
||||||
"all files", ".*"},
|
"all files", "*"},
|
||||||
"compatible files{.fur,.dmf,.mod},another option{.wav,.ttf},.*",
|
"compatible files{.fur,.dmf,.mod},another option{.wav,.ttf},.*",
|
||||||
workingDirTest,
|
workingDirTest,
|
||||||
dpiScale,
|
dpiScale,
|
||||||
|
@ -1815,7 +1815,7 @@ void FurnaceGUI::openFileDialog(FurnaceGUIFileDialogs type) {
|
||||||
"Open Test (Multi)",
|
"Open Test (Multi)",
|
||||||
{"compatible files", "*.fur *.dmf *.mod",
|
{"compatible files", "*.fur *.dmf *.mod",
|
||||||
"another option", "*.wav *.ttf",
|
"another option", "*.wav *.ttf",
|
||||||
"all files", ".*"},
|
"all files", "*"},
|
||||||
"compatible files{.fur,.dmf,.mod},another option{.wav,.ttf},.*",
|
"compatible files{.fur,.dmf,.mod},another option{.wav,.ttf},.*",
|
||||||
workingDirTest,
|
workingDirTest,
|
||||||
dpiScale,
|
dpiScale,
|
||||||
|
@ -2303,8 +2303,8 @@ void FurnaceGUI::editOptions(bool topMenu) {
|
||||||
char id[4096];
|
char id[4096];
|
||||||
editOptsVisible=true;
|
editOptsVisible=true;
|
||||||
|
|
||||||
if (ImGui::MenuItem("cut",BIND_FOR(GUI_ACTION_PAT_CUT))) doCopy(true);
|
if (ImGui::MenuItem("cut",BIND_FOR(GUI_ACTION_PAT_CUT))) doCopy(true,true,selStart,selEnd);
|
||||||
if (ImGui::MenuItem("copy",BIND_FOR(GUI_ACTION_PAT_COPY))) doCopy(false);
|
if (ImGui::MenuItem("copy",BIND_FOR(GUI_ACTION_PAT_COPY))) doCopy(false,true,selStart,selEnd);
|
||||||
if (ImGui::MenuItem("paste",BIND_FOR(GUI_ACTION_PAT_PASTE))) doPaste();
|
if (ImGui::MenuItem("paste",BIND_FOR(GUI_ACTION_PAT_PASTE))) doPaste();
|
||||||
if (ImGui::BeginMenu("paste special...")) {
|
if (ImGui::BeginMenu("paste special...")) {
|
||||||
if (ImGui::MenuItem("paste mix",BIND_FOR(GUI_ACTION_PAT_PASTE_MIX))) doPaste(GUI_PASTE_MODE_MIX_FG);
|
if (ImGui::MenuItem("paste mix",BIND_FOR(GUI_ACTION_PAT_PASTE_MIX))) doPaste(GUI_PASTE_MODE_MIX_FG);
|
||||||
|
@ -3729,6 +3729,7 @@ bool FurnaceGUI::loop() {
|
||||||
if (ImGui::BeginMenu("window")) {
|
if (ImGui::BeginMenu("window")) {
|
||||||
if (ImGui::MenuItem("song information",BIND_FOR(GUI_ACTION_WINDOW_SONG_INFO),songInfoOpen)) songInfoOpen=!songInfoOpen;
|
if (ImGui::MenuItem("song information",BIND_FOR(GUI_ACTION_WINDOW_SONG_INFO),songInfoOpen)) songInfoOpen=!songInfoOpen;
|
||||||
if (ImGui::MenuItem("subsongs",BIND_FOR(GUI_ACTION_WINDOW_SUBSONGS),subSongsOpen)) subSongsOpen=!subSongsOpen;
|
if (ImGui::MenuItem("subsongs",BIND_FOR(GUI_ACTION_WINDOW_SUBSONGS),subSongsOpen)) subSongsOpen=!subSongsOpen;
|
||||||
|
if (ImGui::MenuItem("speed",BIND_FOR(GUI_ACTION_WINDOW_SPEED),speedOpen)) speedOpen=!speedOpen;
|
||||||
if (settings.unifiedDataView) {
|
if (settings.unifiedDataView) {
|
||||||
if (ImGui::MenuItem("assets",BIND_FOR(GUI_ACTION_WINDOW_INS_LIST),insListOpen)) insListOpen=!insListOpen;
|
if (ImGui::MenuItem("assets",BIND_FOR(GUI_ACTION_WINDOW_INS_LIST),insListOpen)) insListOpen=!insListOpen;
|
||||||
} else {
|
} else {
|
||||||
|
@ -3915,6 +3916,7 @@ bool FurnaceGUI::loop() {
|
||||||
drawSpoiler();
|
drawSpoiler();
|
||||||
drawPattern();
|
drawPattern();
|
||||||
drawEditControls();
|
drawEditControls();
|
||||||
|
drawSpeed();
|
||||||
drawSongInfo();
|
drawSongInfo();
|
||||||
drawOrders();
|
drawOrders();
|
||||||
drawSampleList();
|
drawSampleList();
|
||||||
|
@ -5245,6 +5247,7 @@ bool FurnaceGUI::init() {
|
||||||
patManagerOpen=e->getConfBool("patManagerOpen",false);
|
patManagerOpen=e->getConfBool("patManagerOpen",false);
|
||||||
sysManagerOpen=e->getConfBool("sysManagerOpen",false);
|
sysManagerOpen=e->getConfBool("sysManagerOpen",false);
|
||||||
clockOpen=e->getConfBool("clockOpen",false);
|
clockOpen=e->getConfBool("clockOpen",false);
|
||||||
|
speedOpen=e->getConfBool("speedOpen",true);
|
||||||
regViewOpen=e->getConfBool("regViewOpen",false);
|
regViewOpen=e->getConfBool("regViewOpen",false);
|
||||||
logOpen=e->getConfBool("logOpen",false);
|
logOpen=e->getConfBool("logOpen",false);
|
||||||
effectListOpen=e->getConfBool("effectListOpen",false);
|
effectListOpen=e->getConfBool("effectListOpen",false);
|
||||||
|
@ -5357,6 +5360,10 @@ bool FurnaceGUI::init() {
|
||||||
logD("auto-detecting UI scale factor.");
|
logD("auto-detecting UI scale factor.");
|
||||||
dpiScale=getScaleFactor(videoBackend);
|
dpiScale=getScaleFactor(videoBackend);
|
||||||
logD("scale factor: %f",dpiScale);
|
logD("scale factor: %f",dpiScale);
|
||||||
|
if (dpiScale<0.1f) {
|
||||||
|
logW("scale what?");
|
||||||
|
dpiScale=1.0f;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if !(defined(__APPLE__) || defined(_WIN32))
|
#if !(defined(__APPLE__) || defined(_WIN32))
|
||||||
|
@ -5615,6 +5622,7 @@ void FurnaceGUI::commitState() {
|
||||||
e->setConf("patManagerOpen",patManagerOpen);
|
e->setConf("patManagerOpen",patManagerOpen);
|
||||||
e->setConf("sysManagerOpen",sysManagerOpen);
|
e->setConf("sysManagerOpen",sysManagerOpen);
|
||||||
e->setConf("clockOpen",clockOpen);
|
e->setConf("clockOpen",clockOpen);
|
||||||
|
e->setConf("speedOpen",speedOpen);
|
||||||
e->setConf("regViewOpen",regViewOpen);
|
e->setConf("regViewOpen",regViewOpen);
|
||||||
e->setConf("logOpen",logOpen);
|
e->setConf("logOpen",logOpen);
|
||||||
e->setConf("effectListOpen",effectListOpen);
|
e->setConf("effectListOpen",effectListOpen);
|
||||||
|
@ -5861,6 +5869,7 @@ FurnaceGUI::FurnaceGUI():
|
||||||
patManagerOpen(false),
|
patManagerOpen(false),
|
||||||
sysManagerOpen(false),
|
sysManagerOpen(false),
|
||||||
clockOpen(false),
|
clockOpen(false),
|
||||||
|
speedOpen(true),
|
||||||
clockShowReal(true),
|
clockShowReal(true),
|
||||||
clockShowRow(true),
|
clockShowRow(true),
|
||||||
clockShowBeat(true),
|
clockShowBeat(true),
|
||||||
|
|
|
@ -292,6 +292,7 @@ enum FurnaceGUIWindows {
|
||||||
GUI_WINDOW_NOTHING=0,
|
GUI_WINDOW_NOTHING=0,
|
||||||
GUI_WINDOW_EDIT_CONTROLS,
|
GUI_WINDOW_EDIT_CONTROLS,
|
||||||
GUI_WINDOW_SONG_INFO,
|
GUI_WINDOW_SONG_INFO,
|
||||||
|
GUI_WINDOW_SPEED,
|
||||||
GUI_WINDOW_ORDERS,
|
GUI_WINDOW_ORDERS,
|
||||||
GUI_WINDOW_INS_LIST,
|
GUI_WINDOW_INS_LIST,
|
||||||
GUI_WINDOW_PATTERN,
|
GUI_WINDOW_PATTERN,
|
||||||
|
@ -439,6 +440,7 @@ enum FurnaceGUIActions {
|
||||||
GUI_ACTION_WINDOW_INS_LIST,
|
GUI_ACTION_WINDOW_INS_LIST,
|
||||||
GUI_ACTION_WINDOW_INS_EDIT,
|
GUI_ACTION_WINDOW_INS_EDIT,
|
||||||
GUI_ACTION_WINDOW_SONG_INFO,
|
GUI_ACTION_WINDOW_SONG_INFO,
|
||||||
|
GUI_ACTION_WINDOW_SPEED,
|
||||||
GUI_ACTION_WINDOW_PATTERN,
|
GUI_ACTION_WINDOW_PATTERN,
|
||||||
GUI_ACTION_WINDOW_WAVE_LIST,
|
GUI_ACTION_WINDOW_WAVE_LIST,
|
||||||
GUI_ACTION_WINDOW_WAVE_EDIT,
|
GUI_ACTION_WINDOW_WAVE_EDIT,
|
||||||
|
@ -1453,7 +1455,7 @@ class FurnaceGUI {
|
||||||
|
|
||||||
int curIns, curWave, curSample, curOctave, curOrder, prevIns, oldRow, oldOrder, oldOrder1, editStep, exportLoops, soloChan,orderEditMode, orderCursor;
|
int curIns, curWave, curSample, curOctave, curOrder, prevIns, oldRow, oldOrder, oldOrder1, editStep, exportLoops, soloChan,orderEditMode, orderCursor;
|
||||||
int loopOrder, loopRow, loopEnd, isClipping, extraChannelButtons, patNameTarget, newSongCategory, latchTarget;
|
int loopOrder, loopRow, loopEnd, isClipping, extraChannelButtons, patNameTarget, newSongCategory, latchTarget;
|
||||||
int wheelX, wheelY, dragSourceX, dragSourceY, dragDestinationX, dragDestinationY, oldBeat, oldBar;
|
int wheelX, wheelY, dragSourceX, dragSourceXFine, dragSourceY, dragDestinationX, dragDestinationXFine, dragDestinationY, oldBeat, oldBar;
|
||||||
float soloTimeout;
|
float soloTimeout;
|
||||||
|
|
||||||
double exportFadeOut;
|
double exportFadeOut;
|
||||||
|
@ -1462,7 +1464,7 @@ class FurnaceGUI {
|
||||||
bool waveListOpen, waveEditOpen, sampleListOpen, sampleEditOpen, aboutOpen, settingsOpen;
|
bool waveListOpen, waveEditOpen, sampleListOpen, sampleEditOpen, aboutOpen, settingsOpen;
|
||||||
bool mixerOpen, debugOpen, inspectorOpen, oscOpen, volMeterOpen, statsOpen, compatFlagsOpen;
|
bool mixerOpen, debugOpen, inspectorOpen, oscOpen, volMeterOpen, statsOpen, compatFlagsOpen;
|
||||||
bool pianoOpen, notesOpen, channelsOpen, regViewOpen, logOpen, effectListOpen, chanOscOpen;
|
bool pianoOpen, notesOpen, channelsOpen, regViewOpen, logOpen, effectListOpen, chanOscOpen;
|
||||||
bool subSongsOpen, findOpen, spoilerOpen, patManagerOpen, sysManagerOpen, clockOpen;
|
bool subSongsOpen, findOpen, spoilerOpen, patManagerOpen, sysManagerOpen, clockOpen, speedOpen;
|
||||||
|
|
||||||
bool clockShowReal, clockShowRow, clockShowBeat, clockShowMetro, clockShowTime;
|
bool clockShowReal, clockShowRow, clockShowBeat, clockShowMetro, clockShowTime;
|
||||||
float clockMetroTick[16];
|
float clockMetroTick[16];
|
||||||
|
@ -1818,6 +1820,7 @@ class FurnaceGUI {
|
||||||
void drawMobileOrderSel();
|
void drawMobileOrderSel();
|
||||||
void drawEditControls();
|
void drawEditControls();
|
||||||
void drawSongInfo(bool asChild=false);
|
void drawSongInfo(bool asChild=false);
|
||||||
|
void drawSpeed(bool asChild=false);
|
||||||
void drawOrders();
|
void drawOrders();
|
||||||
void drawPattern();
|
void drawPattern();
|
||||||
void drawInsList(bool asChild=false);
|
void drawInsList(bool asChild=false);
|
||||||
|
@ -1889,8 +1892,8 @@ class FurnaceGUI {
|
||||||
void doPullDelete();
|
void doPullDelete();
|
||||||
void doInsert();
|
void doInsert();
|
||||||
void doTranspose(int amount, OperationMask& mask);
|
void doTranspose(int amount, OperationMask& mask);
|
||||||
void doCopy(bool cut);
|
String doCopy(bool cut, bool writeClipboard, const SelectionPoint& sStart, const SelectionPoint& sEnd);
|
||||||
void doPaste(PasteMode mode=GUI_PASTE_MODE_NORMAL, int arg=0);
|
void doPaste(PasteMode mode=GUI_PASTE_MODE_NORMAL, int arg=0, bool readClipboard=true, String clipb="");
|
||||||
void doChangeIns(int ins);
|
void doChangeIns(int ins);
|
||||||
void doInterpolate();
|
void doInterpolate();
|
||||||
void doFade(int p0, int p1, bool mode);
|
void doFade(int p0, int p1, bool mode);
|
||||||
|
|
|
@ -494,6 +494,7 @@ const FurnaceGUIActionDef guiActions[GUI_ACTION_MAX]={
|
||||||
D("WINDOW_INS_LIST", "Instrument List", 0),
|
D("WINDOW_INS_LIST", "Instrument List", 0),
|
||||||
D("WINDOW_INS_EDIT", "Instrument Editor", 0),
|
D("WINDOW_INS_EDIT", "Instrument Editor", 0),
|
||||||
D("WINDOW_SONG_INFO", "Song Information", 0),
|
D("WINDOW_SONG_INFO", "Song Information", 0),
|
||||||
|
D("WINDOW_SPEED", "Speed", 0),
|
||||||
D("WINDOW_PATTERN", "Pattern", 0),
|
D("WINDOW_PATTERN", "Pattern", 0),
|
||||||
D("WINDOW_WAVE_LIST", "Wavetable List", 0),
|
D("WINDOW_WAVE_LIST", "Wavetable List", 0),
|
||||||
D("WINDOW_WAVE_EDIT", "Wavetable Editor", 0),
|
D("WINDOW_WAVE_EDIT", "Wavetable Editor", 0),
|
||||||
|
|
|
@ -1782,14 +1782,22 @@ void FurnaceGUI::drawMacros(std::vector<FurnaceGUIMacroDesc>& macros, FurnaceGUI
|
||||||
ImGui::TableNextColumn();
|
ImGui::TableNextColumn();
|
||||||
float availableWidth=ImGui::GetContentRegionAvail().x-reservedSpace;
|
float availableWidth=ImGui::GetContentRegionAvail().x-reservedSpace;
|
||||||
int totalFit=MIN(255,availableWidth/MAX(1,macroPointSize*dpiScale));
|
int totalFit=MIN(255,availableWidth/MAX(1,macroPointSize*dpiScale));
|
||||||
if (macroDragScroll>255-totalFit) {
|
int scrollMax=0;
|
||||||
macroDragScroll=255-totalFit;
|
for (FurnaceGUIMacroDesc& i: macros) {
|
||||||
|
if (i.macro->len>scrollMax) scrollMax=i.macro->len;
|
||||||
}
|
}
|
||||||
|
scrollMax-=totalFit;
|
||||||
|
if (scrollMax<0) scrollMax=0;
|
||||||
|
if (macroDragScroll>scrollMax) {
|
||||||
|
macroDragScroll=scrollMax;
|
||||||
|
}
|
||||||
|
ImGui::BeginDisabled(scrollMax<1);
|
||||||
ImGui::SetNextItemWidth(availableWidth);
|
ImGui::SetNextItemWidth(availableWidth);
|
||||||
if (CWSliderInt("##MacroScroll",¯oDragScroll,0,255-totalFit,"")) {
|
if (CWSliderInt("##MacroScroll",¯oDragScroll,0,scrollMax,"")) {
|
||||||
if (macroDragScroll<0) macroDragScroll=0;
|
if (macroDragScroll<0) macroDragScroll=0;
|
||||||
if (macroDragScroll>255-totalFit) macroDragScroll=255-totalFit;
|
if (macroDragScroll>scrollMax) macroDragScroll=scrollMax;
|
||||||
}
|
}
|
||||||
|
ImGui::EndDisabled();
|
||||||
|
|
||||||
// draw macros
|
// draw macros
|
||||||
for (FurnaceGUIMacroDesc& i: macros) {
|
for (FurnaceGUIMacroDesc& i: macros) {
|
||||||
|
@ -1838,11 +1846,13 @@ void FurnaceGUI::drawMacros(std::vector<FurnaceGUIMacroDesc>& macros, FurnaceGUI
|
||||||
ImGui::TableNextRow();
|
ImGui::TableNextRow();
|
||||||
ImGui::TableNextColumn();
|
ImGui::TableNextColumn();
|
||||||
ImGui::TableNextColumn();
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::BeginDisabled(scrollMax<1);
|
||||||
ImGui::SetNextItemWidth(availableWidth);
|
ImGui::SetNextItemWidth(availableWidth);
|
||||||
if (CWSliderInt("##MacroScroll",¯oDragScroll,0,255-totalFit,"")) {
|
if (CWSliderInt("##MacroScroll",¯oDragScroll,0,scrollMax,"")) {
|
||||||
if (macroDragScroll<0) macroDragScroll=0;
|
if (macroDragScroll<0) macroDragScroll=0;
|
||||||
if (macroDragScroll>255-totalFit) macroDragScroll=255-totalFit;
|
if (macroDragScroll>scrollMax) macroDragScroll=scrollMax;
|
||||||
}
|
}
|
||||||
|
ImGui::EndDisabled();
|
||||||
ImGui::EndTable();
|
ImGui::EndTable();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -3334,6 +3334,8 @@ void FurnaceGUI::applyUISettings(bool updateFonts) {
|
||||||
ImGui::StyleColorsDark(&sty);
|
ImGui::StyleColorsDark(&sty);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (dpiScale<0.1) dpiScale=0.1;
|
||||||
|
|
||||||
setupLabel(settings.noteOffLabel.c_str(),noteOffLabel,3);
|
setupLabel(settings.noteOffLabel.c_str(),noteOffLabel,3);
|
||||||
setupLabel(settings.noteRelLabel.c_str(),noteRelLabel,3);
|
setupLabel(settings.noteRelLabel.c_str(),noteRelLabel,3);
|
||||||
setupLabel(settings.macroRelLabel.c_str(),macroRelLabel,3);
|
setupLabel(settings.macroRelLabel.c_str(),macroRelLabel,3);
|
||||||
|
@ -3539,21 +3541,21 @@ void FurnaceGUI::applyUISettings(bool updateFonts) {
|
||||||
fc1.MergeMode=true;
|
fc1.MergeMode=true;
|
||||||
|
|
||||||
if (settings.mainFont==6) { // custom font
|
if (settings.mainFont==6) { // custom font
|
||||||
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(settings.mainFontPath.c_str(),e->getConfInt("mainFontSize",18)*dpiScale,NULL,fontRange))==NULL) {
|
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(settings.mainFontPath.c_str(),MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),NULL,fontRange))==NULL) {
|
||||||
logW("could not load UI font! reverting to default font");
|
logW("could not load UI font! reverting to default font");
|
||||||
settings.mainFont=0;
|
settings.mainFont=0;
|
||||||
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFont[settings.mainFont],builtinFontLen[settings.mainFont],e->getConfInt("mainFontSize",18)*dpiScale,NULL,fontRange))==NULL) {
|
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFont[settings.mainFont],builtinFontLen[settings.mainFont],MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),NULL,fontRange))==NULL) {
|
||||||
logE("could not load UI font! falling back to Proggy Clean.");
|
logE("could not load UI font! falling back to Proggy Clean.");
|
||||||
mainFont=ImGui::GetIO().Fonts->AddFontDefault();
|
mainFont=ImGui::GetIO().Fonts->AddFontDefault();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (settings.mainFont==5) { // system font
|
} else if (settings.mainFont==5) { // system font
|
||||||
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_FONT_PATH_1,e->getConfInt("mainFontSize",18)*dpiScale,NULL,fontRange))==NULL) {
|
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_FONT_PATH_1,MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),NULL,fontRange))==NULL) {
|
||||||
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_FONT_PATH_2,e->getConfInt("mainFontSize",18)*dpiScale,NULL,fontRange))==NULL) {
|
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_FONT_PATH_2,MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),NULL,fontRange))==NULL) {
|
||||||
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_FONT_PATH_3,e->getConfInt("mainFontSize",18)*dpiScale,NULL,fontRange))==NULL) {
|
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_FONT_PATH_3,MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),NULL,fontRange))==NULL) {
|
||||||
logW("could not load UI font! reverting to default font");
|
logW("could not load UI font! reverting to default font");
|
||||||
settings.mainFont=0;
|
settings.mainFont=0;
|
||||||
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFont[settings.mainFont],builtinFontLen[settings.mainFont],e->getConfInt("mainFontSize",18)*dpiScale,NULL,fontRange))==NULL) {
|
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFont[settings.mainFont],builtinFontLen[settings.mainFont],MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),NULL,fontRange))==NULL) {
|
||||||
logE("could not load UI font! falling back to Proggy Clean.");
|
logE("could not load UI font! falling back to Proggy Clean.");
|
||||||
mainFont=ImGui::GetIO().Fonts->AddFontDefault();
|
mainFont=ImGui::GetIO().Fonts->AddFontDefault();
|
||||||
}
|
}
|
||||||
|
@ -3561,21 +3563,21 @@ void FurnaceGUI::applyUISettings(bool updateFonts) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFont[settings.mainFont],builtinFontLen[settings.mainFont],e->getConfInt("mainFontSize",18)*dpiScale,NULL,fontRange))==NULL) {
|
if ((mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFont[settings.mainFont],builtinFontLen[settings.mainFont],MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),NULL,fontRange))==NULL) {
|
||||||
logE("could not load UI font! falling back to Proggy Clean.");
|
logE("could not load UI font! falling back to Proggy Clean.");
|
||||||
mainFont=ImGui::GetIO().Fonts->AddFontDefault();
|
mainFont=ImGui::GetIO().Fonts->AddFontDefault();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// two fallback fonts
|
// two fallback fonts
|
||||||
mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(font_liberationSans_compressed_data,font_liberationSans_compressed_size,e->getConfInt("mainFontSize",18)*dpiScale,&fc1,fontRange);
|
mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(font_liberationSans_compressed_data,font_liberationSans_compressed_size,MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),&fc1,fontRange);
|
||||||
mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(font_unifont_compressed_data,font_unifont_compressed_size,e->getConfInt("mainFontSize",18)*dpiScale,&fc1,fontRange);
|
mainFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(font_unifont_compressed_data,font_unifont_compressed_size,MAX(1,e->getConfInt("mainFontSize",18)*dpiScale),&fc1,fontRange);
|
||||||
|
|
||||||
ImFontConfig fc;
|
ImFontConfig fc;
|
||||||
fc.MergeMode=true;
|
fc.MergeMode=true;
|
||||||
fc.GlyphMinAdvanceX=e->getConfInt("iconSize",16)*dpiScale;
|
fc.GlyphMinAdvanceX=e->getConfInt("iconSize",16)*dpiScale;
|
||||||
static const ImWchar fontRangeIcon[]={ICON_MIN_FA,ICON_MAX_FA,0};
|
static const ImWchar fontRangeIcon[]={ICON_MIN_FA,ICON_MAX_FA,0};
|
||||||
if ((iconFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(iconFont_compressed_data,iconFont_compressed_size,e->getConfInt("iconSize",16)*dpiScale,&fc,fontRangeIcon))==NULL) {
|
if ((iconFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(iconFont_compressed_data,iconFont_compressed_size,MAX(1,e->getConfInt("iconSize",16)*dpiScale),&fc,fontRangeIcon))==NULL) {
|
||||||
logE("could not load icon font!");
|
logE("could not load icon font!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3584,21 +3586,21 @@ void FurnaceGUI::applyUISettings(bool updateFonts) {
|
||||||
patFont=mainFont;
|
patFont=mainFont;
|
||||||
} else {
|
} else {
|
||||||
if (settings.patFont==6) { // custom font
|
if (settings.patFont==6) { // custom font
|
||||||
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(settings.patFontPath.c_str(),e->getConfInt("patFontSize",18)*dpiScale,NULL,upTo800))==NULL) {
|
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(settings.patFontPath.c_str(),MAX(1,e->getConfInt("patFontSize",18)*dpiScale),NULL,upTo800))==NULL) {
|
||||||
logW("could not load pattern font! reverting to default font");
|
logW("could not load pattern font! reverting to default font");
|
||||||
settings.patFont=0;
|
settings.patFont=0;
|
||||||
if ((patFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFontM[settings.patFont],builtinFontMLen[settings.patFont],e->getConfInt("patFontSize",18)*dpiScale,NULL,upTo800))==NULL) {
|
if ((patFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFontM[settings.patFont],builtinFontMLen[settings.patFont],MAX(1,e->getConfInt("patFontSize",18)*dpiScale),NULL,upTo800))==NULL) {
|
||||||
logE("could not load pattern font! falling back to Proggy Clean.");
|
logE("could not load pattern font! falling back to Proggy Clean.");
|
||||||
patFont=ImGui::GetIO().Fonts->AddFontDefault();
|
patFont=ImGui::GetIO().Fonts->AddFontDefault();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (settings.patFont==5) { // system font
|
} else if (settings.patFont==5) { // system font
|
||||||
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_PAT_FONT_PATH_1,e->getConfInt("patFontSize",18)*dpiScale,NULL,upTo800))==NULL) {
|
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_PAT_FONT_PATH_1,MAX(1,e->getConfInt("patFontSize",18)*dpiScale),NULL,upTo800))==NULL) {
|
||||||
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_PAT_FONT_PATH_2,e->getConfInt("patFontSize",18)*dpiScale,NULL,upTo800))==NULL) {
|
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_PAT_FONT_PATH_2,MAX(1,e->getConfInt("patFontSize",18)*dpiScale),NULL,upTo800))==NULL) {
|
||||||
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_PAT_FONT_PATH_3,e->getConfInt("patFontSize",18)*dpiScale,NULL,upTo800))==NULL) {
|
if ((patFont=ImGui::GetIO().Fonts->AddFontFromFileTTF(SYSTEM_PAT_FONT_PATH_3,MAX(1,e->getConfInt("patFontSize",18)*dpiScale),NULL,upTo800))==NULL) {
|
||||||
logW("could not load pattern font! reverting to default font");
|
logW("could not load pattern font! reverting to default font");
|
||||||
settings.patFont=0;
|
settings.patFont=0;
|
||||||
if ((patFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFontM[settings.patFont],builtinFontMLen[settings.patFont],e->getConfInt("patFontSize",18)*dpiScale,NULL,upTo800))==NULL) {
|
if ((patFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFontM[settings.patFont],builtinFontMLen[settings.patFont],MAX(1,e->getConfInt("patFontSize",18)*dpiScale),NULL,upTo800))==NULL) {
|
||||||
logE("could not load pattern font! falling back to Proggy Clean.");
|
logE("could not load pattern font! falling back to Proggy Clean.");
|
||||||
patFont=ImGui::GetIO().Fonts->AddFontDefault();
|
patFont=ImGui::GetIO().Fonts->AddFontDefault();
|
||||||
}
|
}
|
||||||
|
@ -3606,7 +3608,7 @@ void FurnaceGUI::applyUISettings(bool updateFonts) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if ((patFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFontM[settings.patFont],builtinFontMLen[settings.patFont],e->getConfInt("patFontSize",18)*dpiScale,NULL,upTo800))==NULL) {
|
if ((patFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(builtinFontM[settings.patFont],builtinFontMLen[settings.patFont],MAX(1,e->getConfInt("patFontSize",18)*dpiScale),NULL,upTo800))==NULL) {
|
||||||
logE("could not load pattern font!");
|
logE("could not load pattern font!");
|
||||||
patFont=ImGui::GetIO().Fonts->AddFontDefault();
|
patFont=ImGui::GetIO().Fonts->AddFontDefault();
|
||||||
}
|
}
|
||||||
|
@ -3615,7 +3617,7 @@ void FurnaceGUI::applyUISettings(bool updateFonts) {
|
||||||
|
|
||||||
// 0x39B = Λ
|
// 0x39B = Λ
|
||||||
static const ImWchar bigFontRange[]={0x20,0xFF,0x39b,0x39b,0};
|
static const ImWchar bigFontRange[]={0x20,0xFF,0x39b,0x39b,0};
|
||||||
if ((bigFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(font_plexSans_compressed_data,font_plexSans_compressed_size,40*dpiScale,NULL,bigFontRange))==NULL) {
|
if ((bigFont=ImGui::GetIO().Fonts->AddFontFromMemoryCompressedTTF(font_plexSans_compressed_data,font_plexSans_compressed_size,MAX(1,40*dpiScale),NULL,bigFontRange))==NULL) {
|
||||||
logE("could not load big UI font!");
|
logE("could not load big UI font!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -29,7 +29,7 @@ void FurnaceGUI::drawSongInfo(bool asChild) {
|
||||||
nextWindow=GUI_WINDOW_NOTHING;
|
nextWindow=GUI_WINDOW_NOTHING;
|
||||||
}
|
}
|
||||||
if (!songInfoOpen && !asChild) return;
|
if (!songInfoOpen && !asChild) return;
|
||||||
bool began=asChild?ImGui::BeginChild("Song Information"):ImGui::Begin("Song Information",&songInfoOpen,globalWinFlags);
|
bool began=asChild?ImGui::BeginChild("Song Info##Song Information"):ImGui::Begin("Song Info##Song Information",&songInfoOpen,globalWinFlags);
|
||||||
if (began) {
|
if (began) {
|
||||||
if (ImGui::BeginTable("NameAuthor",2,ImGuiTableFlags_SizingStretchProp)) {
|
if (ImGui::BeginTable("NameAuthor",2,ImGuiTableFlags_SizingStretchProp)) {
|
||||||
ImGui::TableSetupColumn("c0",ImGuiTableColumnFlags_WidthFixed,0.0);
|
ImGui::TableSetupColumn("c0",ImGuiTableColumnFlags_WidthFixed,0.0);
|
||||||
|
@ -100,138 +100,16 @@ void FurnaceGUI::drawSongInfo(bool asChild) {
|
||||||
ImGui::EndTable();
|
ImGui::EndTable();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ImGui::BeginTable("OtherProps",3,ImGuiTableFlags_SizingStretchProp)) {
|
if (ImGui::BeginTable("OtherProps",2,ImGuiTableFlags_SizingStretchProp)) {
|
||||||
ImGui::TableSetupColumn("c0",ImGuiTableColumnFlags_WidthFixed,0.0);
|
ImGui::TableSetupColumn("c0",ImGuiTableColumnFlags_WidthFixed,0.0);
|
||||||
ImGui::TableSetupColumn("c1",ImGuiTableColumnFlags_WidthStretch,0.0);
|
ImGui::TableSetupColumn("c1",ImGuiTableColumnFlags_WidthStretch,0.0);
|
||||||
ImGui::TableSetupColumn("c2",ImGuiTableColumnFlags_WidthStretch,0.0);
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("TimeBase");
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
float avail=ImGui::GetContentRegionAvail().x;
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
unsigned char realTB=e->curSubSong->timeBase+1;
|
|
||||||
if (ImGui::InputScalar("##TimeBase",ImGuiDataType_U8,&realTB,&_ONE,&_THREE)) { MARK_MODIFIED
|
|
||||||
if (realTB<1) realTB=1;
|
|
||||||
if (realTB>16) realTB=16;
|
|
||||||
e->curSubSong->timeBase=realTB-1;
|
|
||||||
}
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("%.2f BPM",calcBPM(e->curSubSong->speed1,e->curSubSong->speed2,e->curSubSong->hz,e->curSubSong->virtualTempoN,e->curSubSong->virtualTempoD));
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("Speed");
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
if (ImGui::InputScalar("##Speed1",ImGuiDataType_U8,&e->curSubSong->speed1,&_ONE,&_THREE)) { MARK_MODIFIED
|
|
||||||
if (e->curSubSong->speed1<1) e->curSubSong->speed1=1;
|
|
||||||
if (e->isPlaying()) play();
|
|
||||||
}
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
if (ImGui::InputScalar("##Speed2",ImGuiDataType_U8,&e->curSubSong->speed2,&_ONE,&_THREE)) { MARK_MODIFIED
|
|
||||||
if (e->curSubSong->speed2<1) e->curSubSong->speed2=1;
|
|
||||||
if (e->isPlaying()) play();
|
|
||||||
}
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("Virtual Tempo");
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
if (ImGui::InputScalar("##VTempoN",ImGuiDataType_S16,&e->curSubSong->virtualTempoN,&_ONE,&_THREE)) { MARK_MODIFIED
|
|
||||||
if (e->curSubSong->virtualTempoN<1) e->curSubSong->virtualTempoN=1;
|
|
||||||
if (e->curSubSong->virtualTempoN>255) e->curSubSong->virtualTempoN=255;
|
|
||||||
}
|
|
||||||
if (ImGui::IsItemHovered()) {
|
|
||||||
ImGui::SetTooltip("Numerator");
|
|
||||||
}
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
if (ImGui::InputScalar("##VTempoD",ImGuiDataType_S16,&e->curSubSong->virtualTempoD,&_ONE,&_THREE)) { MARK_MODIFIED
|
|
||||||
if (e->curSubSong->virtualTempoD<1) e->curSubSong->virtualTempoD=1;
|
|
||||||
if (e->curSubSong->virtualTempoD>255) e->curSubSong->virtualTempoD=255;
|
|
||||||
}
|
|
||||||
if (ImGui::IsItemHovered()) {
|
|
||||||
ImGui::SetTooltip("Denominator (set to base tempo)");
|
|
||||||
}
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("Highlight");
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
if (ImGui::InputScalar("##Highlight1",ImGuiDataType_U8,&e->curSubSong->hilightA,&_ONE,&_THREE)) {
|
|
||||||
MARK_MODIFIED;
|
|
||||||
}
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
if (ImGui::InputScalar("##Highlight2",ImGuiDataType_U8,&e->curSubSong->hilightB,&_ONE,&_THREE)) {
|
|
||||||
MARK_MODIFIED;
|
|
||||||
}
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("Pattern Length");
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
int patLen=e->curSubSong->patLen;
|
|
||||||
if (ImGui::InputInt("##PatLength",&patLen,1,3)) { MARK_MODIFIED
|
|
||||||
if (patLen<1) patLen=1;
|
|
||||||
if (patLen>DIV_MAX_PATTERNS) patLen=DIV_MAX_PATTERNS;
|
|
||||||
e->curSubSong->patLen=patLen;
|
|
||||||
}
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("Song Length");
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
int ordLen=e->curSubSong->ordersLen;
|
|
||||||
if (ImGui::InputInt("##OrdLength",&ordLen,1,3)) { MARK_MODIFIED
|
|
||||||
if (ordLen<1) ordLen=1;
|
|
||||||
if (ordLen>DIV_MAX_PATTERNS) ordLen=DIV_MAX_PATTERNS;
|
|
||||||
e->curSubSong->ordersLen=ordLen;
|
|
||||||
if (curOrder>=ordLen) {
|
|
||||||
setOrder(ordLen-1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
if (ImGui::Selectable(tempoView?"Base Tempo##TempoOrHz":"Tick Rate##TempoOrHz")) {
|
|
||||||
tempoView=!tempoView;
|
|
||||||
}
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::SetNextItemWidth(avail);
|
|
||||||
float setHz=tempoView?e->curSubSong->hz*2.5:e->curSubSong->hz;
|
|
||||||
if (ImGui::InputFloat("##Rate",&setHz,1.0f,1.0f,"%g")) { MARK_MODIFIED
|
|
||||||
if (tempoView) setHz/=2.5;
|
|
||||||
if (setHz<1) setHz=1;
|
|
||||||
if (setHz>999) setHz=999;
|
|
||||||
e->setSongRate(setHz,setHz<52);
|
|
||||||
}
|
|
||||||
if (tempoView) {
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("= %gHz",e->curSubSong->hz);
|
|
||||||
} else {
|
|
||||||
if (e->curSubSong->hz>=49.98 && e->curSubSong->hz<=50.02) {
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("PAL");
|
|
||||||
}
|
|
||||||
if (e->curSubSong->hz>=59.9 && e->curSubSong->hz<=60.11) {
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("NTSC");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ImGui::TableNextRow();
|
ImGui::TableNextRow();
|
||||||
ImGui::TableNextColumn();
|
ImGui::TableNextColumn();
|
||||||
ImGui::Text("Tuning (A-4)");
|
ImGui::Text("Tuning (A-4)");
|
||||||
ImGui::TableNextColumn();
|
ImGui::TableNextColumn();
|
||||||
float tune=e->song.tuning;
|
float tune=e->song.tuning;
|
||||||
|
float avail=ImGui::GetContentRegionAvail().x;
|
||||||
ImGui::SetNextItemWidth(avail);
|
ImGui::SetNextItemWidth(avail);
|
||||||
if (ImGui::InputFloat("##Tuning",&tune,1.0f,3.0f,"%g")) { MARK_MODIFIED
|
if (ImGui::InputFloat("##Tuning",&tune,1.0f,3.0f,"%g")) { MARK_MODIFIED
|
||||||
if (tune<220.0f) tune=220.0f;
|
if (tune<220.0f) tune=220.0f;
|
||||||
|
|
179
src/gui/speed.cpp
Normal file
179
src/gui/speed.cpp
Normal file
|
@ -0,0 +1,179 @@
|
||||||
|
/**
|
||||||
|
* Furnace Tracker - multi-system chiptune tracker
|
||||||
|
* Copyright (C) 2021-2023 tildearrow and contributors
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "gui.h"
|
||||||
|
#include "imgui.h"
|
||||||
|
#include "misc/cpp/imgui_stdlib.h"
|
||||||
|
#include "intConst.h"
|
||||||
|
|
||||||
|
void FurnaceGUI::drawSpeed(bool asChild) {
|
||||||
|
if (nextWindow==GUI_WINDOW_SPEED) {
|
||||||
|
speedOpen=true;
|
||||||
|
ImGui::SetNextWindowFocus();
|
||||||
|
nextWindow=GUI_WINDOW_NOTHING;
|
||||||
|
}
|
||||||
|
if (!speedOpen && !asChild) return;
|
||||||
|
bool began=asChild?ImGui::BeginChild("Speed"):ImGui::Begin("Speed",&speedOpen,globalWinFlags);
|
||||||
|
if (began) {
|
||||||
|
if (ImGui::BeginTable("Props",3,ImGuiTableFlags_SizingStretchProp)) {
|
||||||
|
ImGui::TableSetupColumn("c0",ImGuiTableColumnFlags_WidthFixed,0.0);
|
||||||
|
ImGui::TableSetupColumn("c1",ImGuiTableColumnFlags_WidthStretch,0.0);
|
||||||
|
ImGui::TableSetupColumn("c2",ImGuiTableColumnFlags_WidthStretch,0.0);
|
||||||
|
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
if (ImGui::Selectable(tempoView?"Base Tempo##TempoOrHz":"Tick Rate##TempoOrHz")) {
|
||||||
|
tempoView=!tempoView;
|
||||||
|
}
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
float avail=ImGui::GetContentRegionAvail().x;
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
float setHz=tempoView?e->curSubSong->hz*2.5:e->curSubSong->hz;
|
||||||
|
if (ImGui::InputFloat("##Rate",&setHz,1.0f,1.0f,"%g")) { MARK_MODIFIED
|
||||||
|
if (tempoView) setHz/=2.5;
|
||||||
|
if (setHz<1) setHz=1;
|
||||||
|
if (setHz>999) setHz=999;
|
||||||
|
e->setSongRate(setHz,setHz<52);
|
||||||
|
}
|
||||||
|
if (tempoView) {
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("= %gHz",e->curSubSong->hz);
|
||||||
|
} else {
|
||||||
|
if (e->curSubSong->hz>=49.98 && e->curSubSong->hz<=50.02) {
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("PAL");
|
||||||
|
}
|
||||||
|
if (e->curSubSong->hz>=59.9 && e->curSubSong->hz<=60.11) {
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("NTSC");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Speed");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
if (ImGui::InputScalar("##Speed1",ImGuiDataType_U8,&e->curSubSong->speed1,&_ONE,&_THREE)) { MARK_MODIFIED
|
||||||
|
if (e->curSubSong->speed1<1) e->curSubSong->speed1=1;
|
||||||
|
if (e->isPlaying()) play();
|
||||||
|
}
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
if (ImGui::InputScalar("##Speed2",ImGuiDataType_U8,&e->curSubSong->speed2,&_ONE,&_THREE)) { MARK_MODIFIED
|
||||||
|
if (e->curSubSong->speed2<1) e->curSubSong->speed2=1;
|
||||||
|
if (e->isPlaying()) play();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Virtual Tempo");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
if (ImGui::InputScalar("##VTempoN",ImGuiDataType_S16,&e->curSubSong->virtualTempoN,&_ONE,&_THREE)) { MARK_MODIFIED
|
||||||
|
if (e->curSubSong->virtualTempoN<1) e->curSubSong->virtualTempoN=1;
|
||||||
|
if (e->curSubSong->virtualTempoN>255) e->curSubSong->virtualTempoN=255;
|
||||||
|
}
|
||||||
|
if (ImGui::IsItemHovered()) {
|
||||||
|
ImGui::SetTooltip("Numerator");
|
||||||
|
}
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
if (ImGui::InputScalar("##VTempoD",ImGuiDataType_S16,&e->curSubSong->virtualTempoD,&_ONE,&_THREE)) { MARK_MODIFIED
|
||||||
|
if (e->curSubSong->virtualTempoD<1) e->curSubSong->virtualTempoD=1;
|
||||||
|
if (e->curSubSong->virtualTempoD>255) e->curSubSong->virtualTempoD=255;
|
||||||
|
}
|
||||||
|
if (ImGui::IsItemHovered()) {
|
||||||
|
ImGui::SetTooltip("Denominator (set to base tempo)");
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("TimeBase");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
unsigned char realTB=e->curSubSong->timeBase+1;
|
||||||
|
if (ImGui::InputScalar("##TimeBase",ImGuiDataType_U8,&realTB,&_ONE,&_THREE)) { MARK_MODIFIED
|
||||||
|
if (realTB<1) realTB=1;
|
||||||
|
if (realTB>16) realTB=16;
|
||||||
|
e->curSubSong->timeBase=realTB-1;
|
||||||
|
}
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%.2f BPM",calcBPM(e->curSubSong->speed1,e->curSubSong->speed2,e->curSubSong->hz,e->curSubSong->virtualTempoN,e->curSubSong->virtualTempoD));
|
||||||
|
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Highlight");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
if (ImGui::InputScalar("##Highlight1",ImGuiDataType_U8,&e->curSubSong->hilightA,&_ONE,&_THREE)) {
|
||||||
|
MARK_MODIFIED;
|
||||||
|
}
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
if (ImGui::InputScalar("##Highlight2",ImGuiDataType_U8,&e->curSubSong->hilightB,&_ONE,&_THREE)) {
|
||||||
|
MARK_MODIFIED;
|
||||||
|
}
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
if (ImGui::BeginTable("Props2",3,ImGuiTableFlags_SizingStretchProp)) {
|
||||||
|
ImGui::TableSetupColumn("c0",ImGuiTableColumnFlags_WidthFixed,0.0);
|
||||||
|
ImGui::TableSetupColumn("c1",ImGuiTableColumnFlags_WidthStretch,0.0);
|
||||||
|
ImGui::TableSetupColumn("c2",ImGuiTableColumnFlags_WidthStretch,0.0);
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Pattern Length");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
float avail=ImGui::GetContentRegionAvail().x;
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
int patLen=e->curSubSong->patLen;
|
||||||
|
if (ImGui::InputInt("##PatLength",&patLen,1,3)) { MARK_MODIFIED
|
||||||
|
if (patLen<1) patLen=1;
|
||||||
|
if (patLen>DIV_MAX_PATTERNS) patLen=DIV_MAX_PATTERNS;
|
||||||
|
e->curSubSong->patLen=patLen;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Song Length");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::SetNextItemWidth(avail);
|
||||||
|
int ordLen=e->curSubSong->ordersLen;
|
||||||
|
if (ImGui::InputInt("##OrdLength",&ordLen,1,3)) { MARK_MODIFIED
|
||||||
|
if (ordLen<1) ordLen=1;
|
||||||
|
if (ordLen>DIV_MAX_PATTERNS) ordLen=DIV_MAX_PATTERNS;
|
||||||
|
e->curSubSong->ordersLen=ordLen;
|
||||||
|
if (curOrder>=ordLen) {
|
||||||
|
setOrder(ordLen-1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!asChild && ImGui::IsWindowFocused(ImGuiFocusedFlags_ChildWindows)) curWindow=GUI_WINDOW_SPEED;
|
||||||
|
if (asChild) {
|
||||||
|
ImGui::EndChild();
|
||||||
|
} else {
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue