TIC-80-guile/src/music.c

1676 lines
37 KiB
C
Raw Normal View History

2017-09-26 08:59:34 +02:00
// MIT License
// Copyright (c) 2017 Vadim Grigoruk @nesbox // grigoruk@gmail.com
// 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 "music.h"
#include "history.h"
#define TRACKER_ROWS (MUSIC_PATTERN_ROWS / 4)
#define CHANNEL_COLS 8
#define TRACKER_COLS (TIC_SOUND_CHANNELS * CHANNEL_COLS)
enum
{
ColumnNote = 0,
ColumnSemitone,
ColumnOctave,
ColumnSfxHi,
ColumnSfxLow,
ColumnVolume,
ColumnEffect,
ColumnParameter,
};
static void undo(Music* music)
{
history_undo(music->history);
}
static void redo(Music* music)
{
history_redo(music->history);
}
const tic_music_pos* getMusicPos(Music* music)
{
return &music->tic->ram.music_pos;
}
2017-12-25 21:57:58 +01:00
static void drawDownBorder(Music* music, s32 x, s32 y, s32 w, s32 h)
{
tic_mem* tic = music->tic;
tic->api.rect(tic, x, y-1, w, 1, tic_color_dark_gray);
tic->api.rect(tic, x-1, y, 1, h, tic_color_dark_gray);
tic->api.rect(tic, x, y+h, w, 1, tic_color_light_blue);
tic->api.rect(tic, x+w, y, 1, h, tic_color_light_blue);
}
2017-09-26 08:59:34 +02:00
static void drawEditbox(Music* music, s32 x, s32 y, s32 value, void(*set)(Music*, s32, s32 channel), s32 channel)
{
static const u8 LeftArrow[] =
{
2017-12-25 21:57:58 +01:00
0b00100000,
0b01100000,
0b11100000,
0b01100000,
0b00100000,
2017-09-26 08:59:34 +02:00
0b00000000,
0b00000000,
0b00000000,
};
static const u8 RightArrow[] =
{
2017-12-25 21:57:58 +01:00
0b00100000,
0b00110000,
0b00111000,
0b00110000,
0b00100000,
2017-09-26 08:59:34 +02:00
0b00000000,
0b00000000,
0b00000000,
};
{
tic_rect rect = { x, y, TIC_FONT_WIDTH, TIC_FONT_HEIGHT };
2017-09-26 08:59:34 +02:00
bool over = false;
2017-12-25 21:57:58 +01:00
bool down = false;
2017-09-26 08:59:34 +02:00
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
2018-02-01 16:32:31 +01:00
if (checkMouseDown(&rect, tic_mouse_left))
2017-12-25 21:57:58 +01:00
down = true;
2018-02-01 16:32:31 +01:00
if (checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
set(music, -1, channel);
}
2017-12-25 21:57:58 +01:00
drawBitIcon(rect.x, rect.y + (down ? 1 : 0), LeftArrow, (over ? tic_color_light_blue : tic_color_dark_gray));
2017-09-26 08:59:34 +02:00
}
{
x += TIC_FONT_WIDTH;
2017-09-26 08:59:34 +02:00
tic_rect rect = { x-1, y-1, TIC_FONT_WIDTH*2+1, TIC_FONT_HEIGHT+1 };
2017-09-26 08:59:34 +02:00
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
2018-02-01 16:32:31 +01:00
if (checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
music->tracker.row = -1;
music->tracker.col = channel * CHANNEL_COLS;
s32 mx = getMouseX() - rect.x;
music->tracker.patternCol = mx / TIC_FONT_WIDTH;
2017-09-26 08:59:34 +02:00
}
}
2017-10-20 09:37:30 +02:00
music->tic->api.rect(music->tic, rect.x, rect.y, rect.w, rect.h, (tic_color_black));
2017-12-25 21:57:58 +01:00
drawDownBorder(music, rect.x, rect.y, rect.w, rect.h);
2017-09-26 08:59:34 +02:00
if(music->tracker.row == -1 && music->tracker.col / CHANNEL_COLS == channel)
{
music->tic->api.rect(music->tic, x - 1 + music->tracker.patternCol * TIC_FONT_WIDTH, y - 1, TIC_FONT_WIDTH + 1, TIC_FONT_HEIGHT + 1, (tic_color_red));
2017-09-26 08:59:34 +02:00
}
char val[] = "99";
sprintf(val, "%02i", value);
music->tic->api.fixed_text(music->tic, val, x, y, (tic_color_white), false);
2017-09-26 08:59:34 +02:00
}
{
x += 2*TIC_FONT_WIDTH;
2017-09-26 08:59:34 +02:00
tic_rect rect = { x, y, TIC_FONT_WIDTH, TIC_FONT_HEIGHT };
2017-09-26 08:59:34 +02:00
bool over = false;
2017-12-25 21:57:58 +01:00
bool down = false;
2017-09-26 08:59:34 +02:00
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
2018-02-01 16:32:31 +01:00
if (checkMouseDown(&rect, tic_mouse_left))
2017-12-25 21:57:58 +01:00
down = true;
2018-02-01 16:32:31 +01:00
if (checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
set(music, +1, channel);
}
2017-12-25 21:57:58 +01:00
drawBitIcon(rect.x, rect.y + (down ? 1 : 0), RightArrow, (over ? tic_color_light_blue : tic_color_dark_gray));
2017-09-26 08:59:34 +02:00
}
}
static void drawSwitch(Music* music, s32 x, s32 y, const char* label, s32 value, void(*set)(Music*, s32, void* data), void* data)
{
static const u8 LeftArrow[] =
{
0b00010000,
0b00110000,
0b01110000,
0b00110000,
0b00010000,
2017-09-26 08:59:34 +02:00
0b00000000,
0b00000000,
0b00000000,
};
static const u8 RightArrow[] =
{
0b01000000,
0b01100000,
0b01110000,
0b01100000,
0b01000000,
2017-09-26 08:59:34 +02:00
0b00000000,
0b00000000,
0b00000000,
};
music->tic->api.text(music->tic, label, x, y+1, (tic_color_black), false);
music->tic->api.text(music->tic, label, x, y, (tic_color_white), false);
2017-09-26 08:59:34 +02:00
{
x += (s32)strlen(label)*TIC_FONT_WIDTH;
2017-09-26 08:59:34 +02:00
tic_rect rect = { x, y, TIC_FONT_WIDTH, TIC_FONT_HEIGHT };
2017-09-26 08:59:34 +02:00
2017-12-25 21:57:58 +01:00
bool over = false;
bool down = false;
2017-09-26 08:59:34 +02:00
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
2017-12-25 21:57:58 +01:00
over = true;
2018-02-01 16:32:31 +01:00
if (checkMouseDown(&rect, tic_mouse_left))
2017-12-25 21:57:58 +01:00
down = true;
2018-02-01 16:32:31 +01:00
if (checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
set(music, -1, data);
}
2017-12-25 21:57:58 +01:00
drawBitIcon(rect.x, rect.y + (down ? 1 : 0), LeftArrow, over ? tic_color_light_blue : tic_color_dark_gray);
2017-09-26 08:59:34 +02:00
}
{
char val[] = "999";
sprintf(val, "%02i", value);
music->tic->api.fixed_text(music->tic, val, x + TIC_FONT_WIDTH, y+1, (tic_color_black), false);
music->tic->api.fixed_text(music->tic, val, x += TIC_FONT_WIDTH, y, (tic_color_white), false);
2017-09-26 08:59:34 +02:00
}
{
x += (value > 99 ? 3 : 2)*TIC_FONT_WIDTH;
2017-09-26 08:59:34 +02:00
tic_rect rect = { x, y, TIC_FONT_WIDTH, TIC_FONT_HEIGHT };
2017-09-26 08:59:34 +02:00
2017-12-25 21:57:58 +01:00
bool over = false;
bool down = false;
2017-09-26 08:59:34 +02:00
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
2017-12-25 21:57:58 +01:00
over = true;
2018-02-01 16:32:31 +01:00
if (checkMouseDown(&rect, tic_mouse_left))
2017-12-25 21:57:58 +01:00
down = true;
2018-02-01 16:32:31 +01:00
if (checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
set(music, +1, data);
}
2017-12-25 21:57:58 +01:00
drawBitIcon(rect.x, rect.y + (down ? 1 : 0), RightArrow, over ? tic_color_light_blue : tic_color_dark_gray);
2017-09-26 08:59:34 +02:00
}
}
static tic_track* getTrack(Music* music)
{
2017-12-14 15:08:04 +01:00
return &music->src->tracks.data[music->track];
2017-09-26 08:59:34 +02:00
}
static s32 getRows(Music* music)
{
tic_track* track = getTrack(music);
return MUSIC_PATTERN_ROWS - track->rows;
}
static void updateScroll(Music* music)
{
s32 rows = getRows(music);
if (music->tracker.scroll < 0)
music->tracker.scroll = 0;
if (music->tracker.scroll > rows - TRACKER_ROWS)
music->tracker.scroll = rows - TRACKER_ROWS;
}
static void updateTracker(Music* music)
{
s32 row = music->tracker.row;
if (row < music->tracker.scroll) music->tracker.scroll = row;
else if (row >= music->tracker.scroll + TRACKER_ROWS)
music->tracker.scroll = row - TRACKER_ROWS + 1;
{
s32 rows = getRows(music);
if (music->tracker.row >= rows) music->tracker.row = rows - 1;
}
updateScroll(music);
}
static void upRow(Music* music)
{
if (music->tracker.row > -1)
{
music->tracker.row--;
updateTracker(music);
}
}
static void downRow(Music* music)
{
const tic_music_pos* pos = getMusicPos(music);
if(pos->track == music->track && music->tracker.follow) return;
if (music->tracker.row < getRows(music) - 1)
{
music->tracker.row++;
updateTracker(music);
}
}
static void leftCol(Music* music)
{
if (music->tracker.col > 0)
{
music->tracker.col--;
updateTracker(music);
}
}
static void rightCol(Music* music)
{
if (music->tracker.col < TRACKER_COLS - 1)
{
music->tracker.col++;
updateTracker(music);
}
}
static void goHome(Music* music)
{
music->tracker.col -= music->tracker.col % CHANNEL_COLS;
}
static void goEnd(Music* music)
{
music->tracker.col -= music->tracker.col % CHANNEL_COLS;
music->tracker.col += CHANNEL_COLS-1;
}
static void pageUp(Music* music)
{
music->tracker.row -= TRACKER_ROWS;
if(music->tracker.row < 0)
music->tracker.row = 0;
updateTracker(music);
}
static void pageDown(Music* music)
{
if (music->tracker.row < getRows(music) - 1)
music->tracker.row += TRACKER_ROWS;
s32 rows = getRows(music);
if(music->tracker.row >= rows)
music->tracker.row = rows-1;
updateTracker(music);
}
static void doTab(Music* music)
{
s32 channel = (music->tracker.col / CHANNEL_COLS + 1) % TIC_SOUND_CHANNELS;
music->tracker.col = channel * CHANNEL_COLS + music->tracker.col % CHANNEL_COLS;
updateTracker(music);
2017-09-26 08:59:34 +02:00
}
static void upFrame(Music* music)
{
music->tracker.frame--;
if(music->tracker.frame < 0)
music->tracker.frame = 0;
}
static void downFrame(Music* music)
{
music->tracker.frame++;
if(music->tracker.frame >= MUSIC_FRAMES)
music->tracker.frame = MUSIC_FRAMES-1;
}
static bool checkPlayFrame(Music* music, s32 frame)
{
const tic_music_pos* pos = getMusicPos(music);
return pos->track == music->track &&
pos->frame == frame;
}
static bool checkPlayRow(Music* music, s32 row)
{
const tic_music_pos* pos = getMusicPos(music);
return checkPlayFrame(music, music->tracker.frame) && pos->row == row;
}
static tic_track_pattern* getPattern(Music* music, s32 channel)
{
s32 patternId = tic_tool_get_pattern_id(getTrack(music), music->tracker.frame, channel);
2017-12-14 15:08:04 +01:00
return patternId ? &music->src->patterns.data[patternId - PATTERN_START] : NULL;
2017-09-26 08:59:34 +02:00
}
static tic_track_pattern* getChannelPattern(Music* music)
{
s32 channel = music->tracker.col / CHANNEL_COLS;
return getPattern(music, channel);
}
static s32 getNote(Music* music)
{
tic_track_pattern* pattern = getChannelPattern(music);
return pattern->rows[music->tracker.row].note - NoteStart;
}
static s32 getOctave(Music* music)
{
tic_track_pattern* pattern = getChannelPattern(music);
return pattern->rows[music->tracker.row].octave;
}
static s32 getVolume(Music* music)
{
tic_track_pattern* pattern = getChannelPattern(music);
return pattern->rows[music->tracker.row].volume;
}
static s32 getSfxId(const tic_track_pattern* pattern, s32 row)
{
return (pattern->rows[row].sfxhi << MUSIC_SFXID_LOW_BITS) | pattern->rows[row].sfxlow;
}
static void setSfxId(tic_track_pattern* pattern, s32 row, s32 sfx)
{
if(sfx >= SFX_COUNT) sfx = SFX_COUNT-1;
pattern->rows[row].sfxhi = (sfx & 0b00100000) >> MUSIC_SFXID_LOW_BITS;
pattern->rows[row].sfxlow = sfx & 0b00011111;
}
static s32 getSfx(Music* music)
{
tic_track_pattern* pattern = getChannelPattern(music);
return getSfxId(pattern, music->tracker.row);
}
static void setSfx(Music* music, s32 sfx)
{
tic_track_pattern* pattern = getChannelPattern(music);
setSfxId(pattern, music->tracker.row, sfx);
music->tracker.last.sfx = getSfxId(pattern, music->tracker.row);
}
static void playNote(Music* music)
{
if (getChannelPattern(music))
{
s32 note = getNote(music);
if (note >= 0 && music->tracker.note != note)
{
music->tracker.note = note;
s32 channel = music->tracker.col / CHANNEL_COLS;
music->tic->api.sfx_stop(music->tic, channel);
music->tic->api.sfx_ex(music->tic, getSfx(music), note, getOctave(music), -1, channel, MAX_VOLUME - getVolume(music), 0);
}
}
}
static void setStopNote(Music* music)
{
tic_track_pattern* pattern = getChannelPattern(music);
pattern->rows[music->tracker.row].note = NoteStop;
pattern->rows[music->tracker.row].octave = 0;
}
static void setNote(Music* music, s32 note, s32 octave, s32 volume, s32 sfx)
2017-09-26 08:59:34 +02:00
{
tic_track_pattern* pattern = getChannelPattern(music);
pattern->rows[music->tracker.row].note = note + NoteStart;
pattern->rows[music->tracker.row].octave = octave;
pattern->rows[music->tracker.row].volume = volume;
setSfxId(pattern, music->tracker.row, sfx);
2017-09-26 08:59:34 +02:00
playNote(music);
}
static void setOctave(Music* music, s32 octave)
{
tic_track_pattern* pattern = getChannelPattern(music);
pattern->rows[music->tracker.row].octave = octave;
music->tracker.last.octave = octave;
playNote(music);
}
static void setVolume(Music* music, s32 volume)
{
tic_track_pattern* pattern = getChannelPattern(music);
pattern->rows[music->tracker.row].volume = volume;
music->tracker.last.volume = volume;
playNote(music);
}
static void playFrameRow(Music* music)
{
music->tic->api.music_frame(music->tic, music->track, music->tracker.frame, music->tracker.row, true);
}
static void playFrame(Music* music)
{
music->tic->api.music_frame(music->tic, music->track, music->tracker.frame, -1, true);
}
static void playTrack(Music* music)
{
music->tic->api.music(music->tic, music->track, -1, -1, true);
}
static void stopTrack(Music* music)
{
music->tic->api.music(music->tic, -1, -1, -1, false);
}
2017-10-03 12:35:00 +02:00
static void resetSelection(Music* music)
2017-09-26 08:59:34 +02:00
{
2018-02-01 17:34:42 +01:00
music->tracker.select.start = (tic_point){-1, -1};
music->tracker.select.rect = (tic_rect){0, 0, 0, 0};
2017-09-26 08:59:34 +02:00
}
2017-10-03 12:35:00 +02:00
static void deleteSelection(Music* music)
2017-09-26 08:59:34 +02:00
{
tic_track_pattern* pattern = getChannelPattern(music);
if(pattern)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = music->tracker.select.rect;
2017-09-26 08:59:34 +02:00
2017-10-03 12:35:00 +02:00
if(rect.h <= 0)
{
rect.y = music->tracker.row;
rect.h = 1;
}
enum{RowSize = sizeof(tic_track_pattern) / MUSIC_PATTERN_ROWS};
2018-02-06 20:15:56 +01:00
memset(&pattern->rows[rect.y], 0, RowSize * rect.h);
2017-09-26 08:59:34 +02:00
}
}
2017-10-03 12:35:00 +02:00
typedef struct
{
u8 size;
} ClipboardHeader;
static void copyToClipboard(Music* music, bool cut)
2017-09-26 08:59:34 +02:00
{
2017-10-03 12:35:00 +02:00
tic_track_pattern* pattern = getChannelPattern(music);
if(pattern)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = music->tracker.select.rect;
2017-10-03 12:35:00 +02:00
if(rect.h <= 0)
{
rect.y = music->tracker.row;
rect.h = 1;
}
ClipboardHeader header = {rect.h};
enum{RowSize = sizeof(tic_track_pattern) / MUSIC_PATTERN_ROWS, HeaderSize = sizeof(ClipboardHeader)};
s32 size = rect.h * RowSize + HeaderSize;
2018-02-06 20:15:56 +01:00
u8* data = malloc(size);
2017-10-03 12:35:00 +02:00
if(data)
{
2018-02-06 20:15:56 +01:00
memcpy(data, &header, HeaderSize);
memcpy(data + HeaderSize, &pattern->rows[rect.y], RowSize * rect.h);
2017-10-03 12:35:00 +02:00
toClipboard(data, size, true);
2018-02-06 20:15:56 +01:00
free(data);
2017-10-03 12:35:00 +02:00
if(cut)
{
deleteSelection(music);
history_add(music->history);
}
resetSelection(music);
}
}
2017-09-26 08:59:34 +02:00
}
static void copyFromClipboard(Music* music)
{
tic_track_pattern* pattern = getChannelPattern(music);
2018-02-14 12:52:10 +01:00
if(pattern && getSystem()->hasClipboardText())
2017-10-03 12:35:00 +02:00
{
2018-02-14 12:52:10 +01:00
char* clipboard = getSystem()->getClipboardText();
2017-10-03 12:35:00 +02:00
if(clipboard)
{
s32 size = strlen(clipboard)/2;
enum{RowSize = sizeof(tic_track_pattern) / MUSIC_PATTERN_ROWS, HeaderSize = sizeof(ClipboardHeader)};
if(size > HeaderSize)
{
2018-02-06 20:15:56 +01:00
u8* data = malloc(size);
2017-10-03 12:35:00 +02:00
2017-11-11 12:44:41 +01:00
str2buf(clipboard, strlen(clipboard), data, true);
2017-10-03 12:35:00 +02:00
ClipboardHeader header = {0};
2018-02-06 20:15:56 +01:00
memcpy(&header, data, HeaderSize);
2017-10-03 12:35:00 +02:00
if(header.size * RowSize == size - HeaderSize)
{
if(header.size + music->tracker.row > MUSIC_PATTERN_ROWS)
header.size = MUSIC_PATTERN_ROWS - music->tracker.row;
2018-02-06 20:15:56 +01:00
memcpy(&pattern->rows[music->tracker.row], data + HeaderSize, header.size * RowSize);
2017-10-03 12:35:00 +02:00
history_add(music->history);
}
2018-02-06 20:15:56 +01:00
free(data);
2017-10-03 12:35:00 +02:00
}
getSystem()->freeClipboardText(clipboard);
2017-10-03 12:35:00 +02:00
}
}
2017-09-26 08:59:34 +02:00
}
static void setChannelPatternValue(Music* music, s32 patternId, s32 channel)
{
tic_track* track = getTrack(music);
s32 frame = music->tracker.frame;
u32 patternData = 0;
for(s32 b = 0; b < TRACK_PATTERNS_SIZE; b++)
patternData |= track->data[frame * TRACK_PATTERNS_SIZE + b] << (BITS_IN_BYTE * b);
s32 shift = channel * TRACK_PATTERN_BITS;
if(patternId < 0) patternId = MUSIC_PATTERNS;
if(patternId > MUSIC_PATTERNS) patternId = 0;
patternData &= ~(TRACK_PATTERN_MASK << shift);
patternData |= patternId << shift;
for(s32 b = 0; b < TRACK_PATTERNS_SIZE; b++)
track->data[frame * TRACK_PATTERNS_SIZE + b] = (patternData >> (b * BITS_IN_BYTE)) & 0xff;
history_add(music->history);
}
static void prevPattern(Music* music)
{
s32 channel = music->tracker.col / CHANNEL_COLS;
if (channel > 0)
{
music->tracker.col = (channel-1) * CHANNEL_COLS;
music->tracker.patternCol = 1;
}
}
static void nextPattern(Music* music)
{
s32 channel = music->tracker.col / CHANNEL_COLS;
if (channel < TIC_SOUND_CHANNELS-1)
{
music->tracker.col = (channel+1) * CHANNEL_COLS;
music->tracker.patternCol = 0;
}
}
static void patternColLeft(Music* music)
{
if(music->tracker.patternCol > 0)
music->tracker.patternCol--;
else prevPattern(music);
}
static void patternColRight(Music* music)
{
if(music->tracker.patternCol < 1)
music->tracker.patternCol++;
else nextPattern(music);
}
2017-10-02 20:13:51 +02:00
static void checkSelection(Music* music)
{
if(music->tracker.select.start.x < 0 || music->tracker.select.start.y < 0)
{
music->tracker.select.start.x = music->tracker.col;
music->tracker.select.start.y = music->tracker.row;
}
}
static void updateSelection(Music* music)
{
2018-02-06 20:15:56 +01:00
s32 rl = MIN(music->tracker.col, music->tracker.select.start.x);
s32 rt = MIN(music->tracker.row, music->tracker.select.start.y);
s32 rr = MAX(music->tracker.col, music->tracker.select.start.x);
s32 rb = MAX(music->tracker.row, music->tracker.select.start.y);
2017-10-02 20:13:51 +02:00
2018-02-01 17:34:42 +01:00
tic_rect* rect = &music->tracker.select.rect;
*rect = (tic_rect){rl, rt, rr - rl + 1, rb - rt + 1};
2017-10-02 20:13:51 +02:00
if(rect->x % CHANNEL_COLS + rect->w > CHANNEL_COLS)
resetSelection(music);
}
2018-02-12 20:05:30 +01:00
static void processTrackerKeyboard(Music* music)
{
tic_mem* tic = music->tic;
if(tic->ram.input.keyboard.data == 0)
{
music->tracker.note = -1;
s32 channel = music->tracker.col / CHANNEL_COLS;
music->tic->api.sfx_stop(music->tic, channel);
return;
}
bool shift = tic->api.key(tic, tic_key_shift);
if(shift)
{
2018-02-13 11:29:33 +01:00
if(keyWasPressed(tic_key_up)
|| keyWasPressed(tic_key_down)
|| keyWasPressed(tic_key_left)
|| keyWasPressed(tic_key_right)
|| keyWasPressed(tic_key_home)
|| keyWasPressed(tic_key_end)
|| keyWasPressed(tic_key_pageup)
|| keyWasPressed(tic_key_pagedown)
|| keyWasPressed(tic_key_tab))
2018-02-12 20:05:30 +01:00
{
checkSelection(music);
}
}
2018-02-13 11:29:33 +01:00
if(keyWasPressed(tic_key_up)) upRow(music);
else if(keyWasPressed(tic_key_down)) downRow(music);
else if(keyWasPressed(tic_key_left)) leftCol(music);
else if(keyWasPressed(tic_key_right)) rightCol(music);
else if(keyWasPressed(tic_key_home)) goHome(music);
else if(keyWasPressed(tic_key_end)) goEnd(music);
else if(keyWasPressed(tic_key_pageup)) pageUp(music);
else if(keyWasPressed(tic_key_pagedown)) pageDown(music);
else if(keyWasPressed(tic_key_tab)) doTab(music);
else if(keyWasPressed(tic_key_delete))
2018-02-12 20:05:30 +01:00
{
deleteSelection(music);
history_add(music->history);
downRow(music);
}
2018-02-13 11:29:33 +01:00
else if(keyWasPressed(tic_key_space)) playNote(music);
else if(keyWasPressed(tic_key_return))
2018-02-12 20:05:30 +01:00
{
const tic_music_pos* pos = getMusicPos(music);
pos->track < 0
? (shift ? playFrameRow(music) : playFrame(music))
: stopTrack(music);
}
if(shift)
{
2018-02-13 11:29:33 +01:00
if(keyWasPressed(tic_key_up)
|| keyWasPressed(tic_key_down)
|| keyWasPressed(tic_key_left)
|| keyWasPressed(tic_key_right)
|| keyWasPressed(tic_key_home)
|| keyWasPressed(tic_key_end)
|| keyWasPressed(tic_key_pageup)
|| keyWasPressed(tic_key_pagedown)
|| keyWasPressed(tic_key_tab))
2018-02-12 20:05:30 +01:00
{
updateSelection(music);
}
}
else resetSelection(music);
static const tic_keycode Piano[] =
{
tic_key_z,
tic_key_s,
tic_key_x,
tic_key_d,
tic_key_c,
tic_key_v,
tic_key_g,
tic_key_b,
tic_key_h,
tic_key_n,
tic_key_j,
tic_key_m,
// octave +1
tic_key_q,
tic_key_2,
tic_key_w,
tic_key_3,
tic_key_e,
tic_key_r,
tic_key_5,
tic_key_t,
tic_key_6,
tic_key_y,
tic_key_7,
tic_key_u,
};
if (getChannelPattern(music))
{
s32 col = music->tracker.col % CHANNEL_COLS;
switch (col)
{
case ColumnNote:
case ColumnSemitone:
2018-02-13 11:29:33 +01:00
if (keyWasPressed(tic_key_1) || keyWasPressed(tic_key_a))
2018-02-12 20:05:30 +01:00
{
setStopNote(music);
downRow(music);
}
else
{
tic_track_pattern* pattern = getChannelPattern(music);
for (s32 i = 0; i < COUNT_OF(Piano); i++)
{
2018-02-13 11:29:33 +01:00
if (keyWasPressed(Piano[i]))
2018-02-12 20:05:30 +01:00
{
s32 note = i % NOTES;
if(pattern->rows[music->tracker.row].note > NoteNone)
{
pattern->rows[music->tracker.row].note = note + NoteStart;
playNote(music);
}
else
{
s32 octave = i / NOTES + music->tracker.last.octave;
s32 volume = music->tracker.last.volume;
s32 sfx = music->tracker.last.sfx;
setNote(music, note, octave, volume, sfx);
}
downRow(music);
break;
}
}
}
break;
case ColumnOctave:
if(getNote(music) >= 0)
{
s32 octave = -1;
char sym = getKeyboardText();
if(sym >= '1' && sym <= '8') octave = sym - '1';
if(octave >= 0)
{
setOctave(music, octave);
downRow(music);
}
}
break;
case ColumnSfxHi:
case ColumnSfxLow:
if(getNote(music) >= 0)
{
s32 val = -1;
char sym = getKeyboardText();
2017-09-26 08:59:34 +02:00
2018-02-12 20:05:30 +01:00
if (sym >= '0' && sym <= '9') val = sym - '0';
if(val >= 0)
{
enum {Base = 10};
s32 sfx = getSfx(music);
sfx = col == 3
? val * Base + sfx % Base
: sfx / Base * Base + val % Base;
setSfx(music, sfx);
if(col == 3) rightCol(music);
else downRow(music), leftCol(music);
}
}
break;
case ColumnVolume:
if (getNote(music) >= 0)
{
s32 val = -1;
2017-09-26 08:59:34 +02:00
2018-02-12 20:05:30 +01:00
char sym = getKeyboardText();
if(sym >= '0' && sym <= '9') val = sym - '0';
if(sym >= 'a' && sym <= 'f') val = sym - 'a' + 10;
if(val >= 0)
{
setVolume(music, MAX_VOLUME - val);
downRow(music);
}
}
break;
}
history_add(music->history);
}
}
static void processPatternKeyboard(Music* music)
{
s32 channel = music->tracker.col / CHANNEL_COLS;
2018-02-13 11:29:33 +01:00
if(keyWasPressed(tic_key_delete)) setChannelPatternValue(music, 0, channel);
else if(keyWasPressed(tic_key_tab)) nextPattern(music);
else if(keyWasPressed(tic_key_left)) patternColLeft(music);
else if(keyWasPressed(tic_key_right)) patternColRight(music);
else if(keyWasPressed(tic_key_down)
|| keyWasPressed(tic_key_return))
2018-02-12 20:05:30 +01:00
music->tracker.row = music->tracker.scroll;
else
{
s32 val = -1;
char sym = getKeyboardText();
if(sym >= '0' && sym <= '9') val = sym - '0';
if(val >= 0)
{
enum {Base = 10};
s32 patternId = tic_tool_get_pattern_id(getTrack(music), music->tracker.frame, channel);
patternId = music->tracker.patternCol == 0
? val * Base + patternId % Base
: patternId / Base * Base + val % Base;
if(patternId <= MUSIC_PATTERNS)
{
setChannelPatternValue(music, patternId, channel);
if(music->tracker.patternCol == 0)
patternColRight(music);
}
}
}
}
2017-09-26 08:59:34 +02:00
2017-10-02 20:13:51 +02:00
static void selectAll(Music* music)
{
resetSelection(music);
s32 col = music->tracker.col - music->tracker.col % CHANNEL_COLS;
2018-02-01 17:34:42 +01:00
music->tracker.select.start = (tic_point){col, 0};
2017-10-02 20:13:51 +02:00
music->tracker.col = col + CHANNEL_COLS-1;
music->tracker.row = MUSIC_PATTERN_ROWS-1;
updateSelection(music);
}
2018-02-12 20:05:30 +01:00
static void processKeyboard(Music* music)
{
tic_mem* tic = music->tic;
switch(getClipboardEvent())
{
case TIC_CLIPBOARD_CUT: copyToClipboard(music, true); break;
case TIC_CLIPBOARD_COPY: copyToClipboard(music, false); break;
case TIC_CLIPBOARD_PASTE: copyFromClipboard(music); break;
default: break;
}
bool ctrl = tic->api.key(tic, tic_key_ctrl);
if (ctrl)
{
2018-02-13 11:29:33 +01:00
if(keyWasPressed(tic_key_a)) selectAll(music);
else if(keyWasPressed(tic_key_z)) undo(music);
else if(keyWasPressed(tic_key_y)) redo(music);
else if(keyWasPressed(tic_key_up)) upFrame(music);
else if(keyWasPressed(tic_key_down)) downFrame(music);
2018-02-12 20:05:30 +01:00
}
else
{
music->tracker.row >= 0
? processTrackerKeyboard(music)
: processPatternKeyboard(music);
}
}
2017-09-26 08:59:34 +02:00
static void setIndex(Music* music, s32 delta, void* data)
{
music->track += delta;
}
static void setTempo(Music* music, s32 delta, void* data)
{
enum
{
Step = 10,
Min = 40-DEFAULT_TEMPO,
Max = 250-DEFAULT_TEMPO,
};
tic_track* track = getTrack(music);
s32 tempo = track->tempo;
tempo += delta * Step;
if (tempo > Max) tempo = Max;
if (tempo < Min) tempo = Min;
track->tempo = tempo;
history_add(music->history);
}
static void setSpeed(Music* music, s32 delta, void* data)
{
enum
{
Step = 1,
Min = 1-DEFAULT_SPEED,
Max = 31-DEFAULT_SPEED,
};
tic_track* track = getTrack(music);
s32 speed = track->speed;
speed += delta * Step;
if (speed > Max) speed = Max;
if (speed < Min) speed = Min;
track->speed = speed;
history_add(music->history);
}
static void setRows(Music* music, s32 delta, void* data)
{
enum
{
Step = 1,
Min = 0,
Max = MUSIC_PATTERN_ROWS - TRACKER_ROWS,
};
tic_track* track = getTrack(music);
s32 rows = track->rows;
rows -= delta * Step;
if (rows < Min) rows = Min;
if (rows > Max) rows = Max;
track->rows = rows;
updateTracker(music);
history_add(music->history);
}
static void drawTopPanel(Music* music, s32 x, s32 y)
{
tic_track* track = getTrack(music);
drawSwitch(music, x, y, "TRACK", music->track, setIndex, NULL);
drawSwitch(music, x += TIC_FONT_WIDTH * 10, y, "TEMPO", track->tempo + DEFAULT_TEMPO, setTempo, NULL);
drawSwitch(music, x += TIC_FONT_WIDTH * 11, y, "SPD", track->speed + DEFAULT_SPEED, setSpeed, NULL);
drawSwitch(music, x += TIC_FONT_WIDTH * 8, y, "ROWS", MUSIC_PATTERN_ROWS - track->rows, setRows, NULL);
2017-09-26 08:59:34 +02:00
}
static void drawTrackerFrames(Music* music, s32 x, s32 y)
{
enum
{
Border = 1,
Width = TIC_FONT_WIDTH * 2 + Border,
2017-09-26 08:59:34 +02:00
};
{
tic_rect rect = { x - Border, y - Border, Width, MUSIC_FRAMES * TIC_FONT_HEIGHT + Border };
2017-09-26 08:59:34 +02:00
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
2018-02-01 16:32:31 +01:00
if (checkMouseDown(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
s32 my = getMouseY() - rect.y - Border;
music->tracker.frame = my / TIC_FONT_HEIGHT;
2017-09-26 08:59:34 +02:00
}
}
2017-10-20 09:37:30 +02:00
music->tic->api.rect(music->tic, rect.x, rect.y, rect.w, rect.h, (tic_color_black));
2017-12-25 21:57:58 +01:00
drawDownBorder(music, rect.x, rect.y, rect.w, rect.h);
2017-09-26 08:59:34 +02:00
}
for (s32 i = 0; i < MUSIC_FRAMES; i++)
{
if (checkPlayFrame(music, i))
{
static const u8 Icon[] =
{
0b00000000,
0b01000000,
0b01100000,
0b01110000,
0b01100000,
0b01000000,
0b00000000,
0b00000000,
};
drawBitIcon(x - TIC_FONT_WIDTH-1, y + i*TIC_FONT_HEIGHT, Icon, tic_color_black);
drawBitIcon(x - TIC_FONT_WIDTH-1, y - 1 + i*TIC_FONT_HEIGHT, Icon, tic_color_white);
2017-09-26 08:59:34 +02:00
}
if (i == music->tracker.frame)
{
music->tic->api.rect(music->tic, x - 1, y - 1 + i*TIC_FONT_HEIGHT, Width, TIC_FONT_HEIGHT + 1, (tic_color_white));
2017-09-26 08:59:34 +02:00
}
char buf[] = "99";
sprintf(buf, "%02i", i);
music->tic->api.fixed_text(music->tic, buf, x, y + i*TIC_FONT_HEIGHT, (tic_color_dark_gray), false);
2017-09-26 08:59:34 +02:00
}
if(music->tracker.row >= 0)
{
char buf[] = "99";
sprintf(buf, "%02i", music->tracker.row);
music->tic->api.fixed_text(music->tic, buf, x, y - 10, (tic_color_black), false);
music->tic->api.fixed_text(music->tic, buf, x, y - 11, (tic_color_white), false);
2017-09-26 08:59:34 +02:00
}
}
static void setChannelPattern(Music* music, s32 delta, s32 channel)
{
tic_track* track = getTrack(music);
s32 frame = music->tracker.frame;
u32 patternData = 0;
for(s32 b = 0; b < TRACK_PATTERNS_SIZE; b++)
patternData |= track->data[frame * TRACK_PATTERNS_SIZE + b] << (BITS_IN_BYTE * b);
s32 shift = channel * TRACK_PATTERN_BITS;
s32 patternId = (patternData >> shift) & TRACK_PATTERN_MASK;
setChannelPatternValue(music, patternId + delta, channel);
}
static void drawTrackerChannel(Music* music, s32 x, s32 y, s32 channel)
{
2017-10-02 17:27:50 +02:00
tic_mem* tic = music->tic;
2017-09-26 08:59:34 +02:00
enum
{
Border = 1,
Rows = TRACKER_ROWS,
Width = TIC_FONT_WIDTH * 8 + Border,
2017-09-26 08:59:34 +02:00
};
tic_rect rect = {x - Border, y - Border, Width, Rows*TIC_FONT_HEIGHT + Border};
2017-09-26 08:59:34 +02:00
if(checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
2018-02-01 16:32:31 +01:00
if(checkMouseDown(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
s32 mx = getMouseX() - rect.x - Border;
s32 my = getMouseY() - rect.y - Border;
s32 col = music->tracker.col = channel * CHANNEL_COLS + mx / TIC_FONT_WIDTH;
s32 row = music->tracker.row = my / TIC_FONT_HEIGHT + music->tracker.scroll;
2017-10-02 17:27:50 +02:00
if(music->tracker.select.drag)
{
2017-10-02 20:13:51 +02:00
updateSelection(music);
2017-10-02 17:27:50 +02:00
}
else
{
2017-10-02 20:13:51 +02:00
resetSelection(music);
2018-02-01 17:34:42 +01:00
music->tracker.select.start = (tic_point){col, row};
2017-10-02 17:27:50 +02:00
music->tracker.select.drag = true;
}
}
}
if(music->tracker.select.drag)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {0, 0, TIC80_WIDTH, TIC80_HEIGHT};
2018-02-01 16:32:31 +01:00
if(!checkMouseDown(&rect, tic_mouse_left))
2017-10-02 17:27:50 +02:00
{
music->tracker.select.drag = false;
2017-09-26 08:59:34 +02:00
}
}
2017-10-20 09:37:30 +02:00
music->tic->api.rect(music->tic, rect.x, rect.y, rect.w, rect.h, (tic_color_black));
2017-12-25 21:57:58 +01:00
drawDownBorder(music, rect.x, rect.y, rect.w, rect.h);
2017-09-26 08:59:34 +02:00
s32 start = music->tracker.scroll;
s32 end = start + Rows;
2017-10-02 17:27:50 +02:00
bool selectedChannel = music->tracker.select.rect.x / CHANNEL_COLS == channel;
2017-09-26 08:59:34 +02:00
tic_track_pattern* pattern = getPattern(music, channel);
for (s32 i = start, pos = 0; i < end; i++, pos++)
{
s32 rowy = y + pos*TIC_FONT_HEIGHT;
2017-09-26 08:59:34 +02:00
if (i == music->tracker.row)
{
music->tic->api.rect(music->tic, x - 1, rowy - 1, Width, TIC_FONT_HEIGHT + 1, (tic_color_dark_red));
2017-09-26 08:59:34 +02:00
}
2017-10-02 17:27:50 +02:00
// draw selection
if (selectedChannel)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = music->tracker.select.rect;
2017-10-03 12:35:00 +02:00
if (rect.h > 1 && i >= rect.y && i < rect.y + rect.h)
2017-10-02 17:27:50 +02:00
{
2017-10-03 12:35:00 +02:00
s32 sx = x - 1;
tic->api.rect(tic, sx, rowy - 1, CHANNEL_COLS * TIC_FONT_WIDTH + 1, TIC_FONT_HEIGHT + 1, (tic_color_yellow));
2017-10-02 17:27:50 +02:00
}
}
2017-09-26 08:59:34 +02:00
if (checkPlayRow(music, i))
{
music->tic->api.rect(music->tic, x - 1, rowy - 1, Width, TIC_FONT_HEIGHT + 1, (tic_color_white));
2017-09-26 08:59:34 +02:00
}
char rowStr[] = "--------";
if (pattern)
{
static const char* Notes[] = SFX_NOTES;
s32 note = pattern->rows[i].note;
s32 octave = pattern->rows[i].octave;
s32 sfx = getSfxId(pattern, i);
s32 volume = MAX_VOLUME - pattern->rows[i].volume;
if (note == NoteStop)
{
strcpy(rowStr, " ");
}
else
{
if (note >= NoteStart)
sprintf(rowStr, "%s%i%02i%01X--", Notes[note - NoteStart], octave + 1, sfx, volume & 0xf);
2017-10-20 09:37:30 +02:00
const u8 Colors[] = { (tic_color_light_green), (tic_color_orange), (tic_color_blue), (tic_color_gray) };
const u8 DarkColors[] = { (tic_color_green), (tic_color_brown), (tic_color_dark_blue), (tic_color_dark_gray) };
2017-09-26 08:59:34 +02:00
static u8 ColorIndexes[] = { 0, 0, 0, 1, 1, 2, 3, 3 };
bool beetRow = i % NOTES_PER_BEET == 0;
for (s32 c = 0, colx = x; c < sizeof rowStr - 1; c++, colx += TIC_FONT_WIDTH)
2017-09-26 08:59:34 +02:00
{
char sym = rowStr[c];
const u8* colors = beetRow || sym != '-' ? Colors : DarkColors;
music->tic->api.draw_char(music->tic, sym, colx, rowy, colors[ColorIndexes[c]], false);
2017-09-26 08:59:34 +02:00
}
}
}
else music->tic->api.fixed_text(music->tic, rowStr, x, rowy, (tic_color_dark_gray), false);
2017-09-26 08:59:34 +02:00
if (i == music->tracker.row)
{
if (music->tracker.col / CHANNEL_COLS == channel)
{
s32 col = music->tracker.col % CHANNEL_COLS;
s32 colx = x - 1 + col * TIC_FONT_WIDTH;
music->tic->api.rect(music->tic, colx, rowy - 1, TIC_FONT_WIDTH + 1, TIC_FONT_HEIGHT + 1, (tic_color_red));
music->tic->api.draw_char(music->tic, rowStr[col], colx + 1, rowy, (tic_color_black), false);
2017-09-26 08:59:34 +02:00
}
}
if (i % NOTES_PER_BEET == 0)
music->tic->api.pixel(music->tic, x - 4, y + pos*TIC_FONT_HEIGHT + 2, (tic_color_black));
2017-09-26 08:59:34 +02:00
}
}
static void drawTumbler(Music* music, s32 x, s32 y, s32 index)
2017-09-28 17:52:52 +02:00
{
tic_mem* tic = music->tic;
enum{On=36, Off = 52, Size=5, Chroma=14};
2018-02-01 17:34:42 +01:00
tic_rect rect = {x, y, Size, Size};
2017-09-28 17:52:52 +02:00
if(checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-28 17:52:52 +02:00
showTooltip("on/off channel");
2018-02-01 16:32:31 +01:00
if(checkMouseClick(&rect, tic_mouse_left))
{
2018-08-18 16:40:28 +02:00
if (tic->api.key(tic, tic_key_ctrl))
{
for (s32 i = 0; i < TIC_SOUND_CHANNELS; i++)
music->tracker.patterns[i] = i == index;
}
else music->tracker.patterns[index] = !music->tracker.patterns[index];
}
2017-09-28 17:52:52 +02:00
}
u8 color = Chroma;
2018-03-12 07:40:48 +01:00
tic->api.sprite(tic, &getConfig()->cart->bank0.tiles, music->tracker.patterns[index] ? On : Off, x, y, &color, 1);
2017-09-28 17:52:52 +02:00
}
2017-09-26 08:59:34 +02:00
static void drawTracker(Music* music, s32 x, s32 y)
{
drawTrackerFrames(music, x, y);
x += TIC_FONT_WIDTH * 3;
2017-09-26 08:59:34 +02:00
enum{ChannelWidth = TIC_FONT_WIDTH * 9};
2017-09-28 17:52:52 +02:00
2017-09-26 08:59:34 +02:00
for (s32 i = 0; i < TIC_SOUND_CHANNELS; i++)
{
s32 patternId = tic_tool_get_pattern_id(getTrack(music), music->tracker.frame, i);
drawEditbox(music, x + ChannelWidth * i + 2*TIC_FONT_WIDTH, y - 11, patternId, setChannelPattern, i);
drawTumbler(music, x + ChannelWidth * i + 7*TIC_FONT_WIDTH, y - 11, i);
2017-09-26 08:59:34 +02:00
}
for (s32 i = 0; i < TIC_SOUND_CHANNELS; i++)
drawTrackerChannel(music, x + ChannelWidth * i, y, i);
2017-09-26 08:59:34 +02:00
}
static void enableFollowMode(Music* music)
{
music->tracker.follow = !music->tracker.follow;
}
static void drawPlayButtons(Music* music)
{
static const u8 Icons[] =
{
0b00000000,
0b01110000,
0b11111000,
0b11111000,
0b11111000,
0b01110000,
0b00000000,
0b00000000,
0b00000000,
0b01010000,
0b01011000,
0b01011100,
0b01011000,
0b01010000,
0b00000000,
0b00000000,
0b00000000,
0b00100000,
0b00110000,
0b00111000,
0b00110000,
0b00100000,
0b00000000,
0b00000000,
0b00000000,
0b01111100,
0b01111100,
0b01111100,
0b01111100,
0b01111100,
0b00000000,
0b00000000,
};
enum { Offset = TIC80_WIDTH - 52, Width = 7, Height = 7, Rows = 8, Count = sizeof Icons / Rows };
for (s32 i = 0; i < Count; i++)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = { Offset + Width * i, 0, Width, Height };
2017-09-26 08:59:34 +02:00
bool over = false;
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
static const char* Tooltips[] = { "RECORD MUSIC", "PLAY FRAME [enter]", "PLAY TRACK", "STOP [enter]" };
showTooltip(Tooltips[i]);
static void(*const Handlers[])(Music*) = { enableFollowMode, playFrame, playTrack, stopTrack };
2018-02-01 16:32:31 +01:00
if (checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
Handlers[i](music);
}
if(i == 0 && music->tracker.follow)
2017-10-20 09:37:30 +02:00
drawBitIcon(rect.x, rect.y, Icons + i*Rows, over ? (tic_color_peach) : (tic_color_red));
2017-09-26 08:59:34 +02:00
else
2017-10-20 09:37:30 +02:00
drawBitIcon(rect.x, rect.y, Icons + i*Rows, over ? (tic_color_dark_gray) : (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
}
}
static void drawModeTabs(Music* music)
{
static const u8 Icons[] =
{
0b00000000,
0b01111111,
0b01010101,
0b01010101,
0b01000001,
0b01111111,
0b00000000,
0b00000000,
0b00000000,
0b01011011,
0b00000000,
0b01011011,
0b00000000,
0b01011011,
0b00000000,
0b00000000,
};
enum { Width = 9, Height = 7, Rows = 8, Count = sizeof Icons / Rows };
for (s32 i = 0; i < Count; i++)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = { TIC80_WIDTH - Width * (Count - i), 0, Width, Height };
2017-09-26 08:59:34 +02:00
static const s32 Tabs[] = { MUSIC_PIANO_TAB, MUSIC_TRACKER_TAB };
2017-12-25 21:57:58 +01:00
bool over = false;
2017-09-26 08:59:34 +02:00
if (checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
static const char* Tooltips[] = { "PIANO MODE", "TRACKER MODE" };
showTooltip(Tooltips[i]);
2018-02-01 16:32:31 +01:00
if (checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
music->tab = Tabs[i];
}
if (music->tab == Tabs[i])
2017-12-25 21:57:58 +01:00
{
2017-10-20 09:37:30 +02:00
music->tic->api.rect(music->tic, rect.x, rect.y, rect.w, rect.h, (tic_color_gray));
2017-12-25 21:57:58 +01:00
drawBitIcon(rect.x, rect.y + 1, Icons + i*Rows, tic_color_black);
}
2017-09-26 08:59:34 +02:00
2017-10-20 09:37:30 +02:00
drawBitIcon(rect.x, rect.y, Icons + i*Rows, music->tab == Tabs[i] ? (tic_color_white) : over ? (tic_color_dark_gray) : (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
}
}
static void drawMusicToolbar(Music* music)
{
2017-10-20 09:37:30 +02:00
music->tic->api.rect(music->tic, 0, 0, TIC80_WIDTH, TOOLBAR_SIZE, (tic_color_white));
2017-09-26 08:59:34 +02:00
drawPlayButtons(music);
drawModeTabs(music);
}
static void drawPianoLayout(Music* music)
{
2017-10-20 09:37:30 +02:00
music->tic->api.clear(music->tic, (tic_color_gray));
2017-09-26 08:59:34 +02:00
static const char Wip[] = "PIANO MODE - WORK IN PROGRESS...";
music->tic->api.fixed_text(music->tic, Wip, (TIC80_WIDTH - (sizeof Wip - 1) * TIC_FONT_WIDTH) / 2, TIC80_HEIGHT / 2, (tic_color_white), false);
2017-09-26 08:59:34 +02:00
}
2017-10-04 11:19:27 +02:00
static void scrollNotes(Music* music, s32 delta)
{
tic_track_pattern* pattern = getChannelPattern(music);
if(pattern)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = music->tracker.select.rect;
2017-10-04 11:19:27 +02:00
if(rect.h <= 0)
{
rect.y = music->tracker.row;
rect.h = 1;
}
for(s32 i = rect.y; i < rect.y + rect.h; i++)
{
s32 note = pattern->rows[i].note + pattern->rows[i].octave * NOTES - NoteStart;
note += delta;
if(note >= 0 && note < NOTES*OCTAVES)
{
pattern->rows[i].note = note % NOTES + NoteStart;
pattern->rows[i].octave = note / NOTES;
}
}
history_add(music->history);
}
}
2017-09-26 08:59:34 +02:00
static void drawTrackerLayout(Music* music)
{
2018-02-12 11:31:52 +01:00
tic_mem* tic = music->tic;
// process scroll
{
tic80_input* input = &tic->ram.input;
if(input->mouse.scrolly)
{
if(tic->api.key(tic, tic_key_ctrl))
{
scrollNotes(music, input->mouse.scrolly > 0 ? 1 : -1);
}
else
{
enum{Scroll = NOTES_PER_BEET};
s32 delta = input->mouse.scrolly > 0 ? -Scroll : Scroll;
music->tracker.scroll += delta;
updateScroll(music);
}
}
}
2018-02-12 20:05:30 +01:00
processKeyboard(music);
2017-09-26 08:59:34 +02:00
if(music->tracker.follow)
{
const tic_music_pos* pos = getMusicPos(music);
if(pos->track == music->track &&
music->tracker.row >= 0 &&
pos->row >= 0)
{
music->tracker.frame = pos->frame;
music->tracker.row = pos->row;
updateTracker(music);
}
}
2017-10-20 09:37:30 +02:00
music->tic->api.clear(music->tic, (tic_color_gray));
2017-09-26 08:59:34 +02:00
2017-12-25 21:57:58 +01:00
drawTopPanel(music, 7, TOOLBAR_SIZE + 4);
drawTracker(music, 7, 35);
2017-09-26 08:59:34 +02:00
}
static void tick(Music* music)
{
2017-09-28 17:52:52 +02:00
tic_mem* tic = music->tic;
for (s32 i = 0; i < TIC_SOUND_CHANNELS; i++)
if(!music->tracker.patterns[i])
tic->ram.registers[i].volume = 0;
2017-09-26 08:59:34 +02:00
switch (music->tab)
{
case MUSIC_TRACKER_TAB: drawTrackerLayout(music); break;
case MUSIC_PIANO_TAB: drawPianoLayout(music); break;
}
drawMusicToolbar(music);
2017-10-20 09:37:30 +02:00
drawToolbar(music->tic, (tic_color_gray), false);
2017-09-26 08:59:34 +02:00
}
static void onStudioEvent(Music* music, StudioEvent event)
{
switch (event)
{
2017-10-03 12:35:00 +02:00
case TIC_TOOLBAR_CUT: copyToClipboard(music, true); break;
case TIC_TOOLBAR_COPY: copyToClipboard(music, false); break;
2017-09-26 08:59:34 +02:00
case TIC_TOOLBAR_PASTE: copyFromClipboard(music); break;
case TIC_TOOLBAR_UNDO: undo(music); break;
case TIC_TOOLBAR_REDO: redo(music); break;
default: break;
}
}
2017-12-14 15:08:04 +01:00
void initMusic(Music* music, tic_mem* tic, tic_music* src)
2017-09-26 08:59:34 +02:00
{
if (music->history) history_delete(music->history);
*music = (Music)
{
.tic = tic,
.tick = tick,
2017-12-14 15:08:04 +01:00
.src = src,
2017-09-26 08:59:34 +02:00
.track = 0,
.tracker =
{
.follow = false,
.frame = 0,
.col = 0,
.row = 0,
.scroll = 0,
.note = -1,
.last =
{
.octave = 3,
.sfx = 0,
.volume = 0,
},
2017-09-28 17:52:52 +02:00
.patterns = {true, true, true, true},
2017-10-02 17:27:50 +02:00
.select =
{
.start = {0, 0},
.rect = {0, 0, 0, 0},
.drag = false,
},
2017-09-26 08:59:34 +02:00
},
.tab = MUSIC_TRACKER_TAB,
2017-12-14 15:08:04 +01:00
.history = history_create(src, sizeof(tic_music)),
2017-09-26 08:59:34 +02:00
.event = onStudioEvent,
};
2017-10-02 20:13:51 +02:00
resetSelection(music);
2017-09-26 08:59:34 +02:00
}