protected DPCMSample CreateUniquelyNamedSampleFromDmcData(string baseName, byte[] data) { string name = baseName; var j = 2; while (!project.IsDPCMSampleNameUnique(name)) name = baseName + "-" + j++; return project.CreateDPCMSampleFromDmcData(name, data); }
public Project Load(string filename) { #if !DEBUG try #endif { var lines = File.ReadAllLines(filename); var parameters = new Dictionary <string, string>(); var project = (Project)null; var instrument = (Instrument)null; var arpeggio = (Arpeggio)null; var song = (Song)null; var channel = (Channel)null; var pattern = (Pattern)null; SetInvariantCulture(); foreach (var line in lines) { var cmd = SplitLine(line.Trim(), ref parameters); switch (cmd) { case "Project": { project = new Project(); parameters.TryGetValue("Version", out var version); if (parameters.TryGetValue("Name", out var name)) { project.Name = name; } if (parameters.TryGetValue("Author", out var author)) { project.Author = author; } if (parameters.TryGetValue("Copyright", out var copyright)) { project.Copyright = copyright; } if (parameters.TryGetValue("TempoMode", out var tempoMode)) { project.TempoMode = TempoType.GetValueForName(tempoMode); } if (parameters.TryGetValue("PAL", out var pal)) { project.PalMode = bool.Parse(pal); } if (parameters.TryGetValue("Expansions", out var expansions)) { var expansionMask = 0; var expansionStrings = expansions.Split(','); foreach (var s in expansionStrings) { var exp = ExpansionType.GetValueForShortName(s.Trim()); expansionMask |= ExpansionType.GetMaskFromValue(exp); } var numN163Channels = 1; if ((expansionMask & ExpansionType.N163Mask) != 0 && parameters.TryGetValue("NumN163Channels", out var numN163ChannelsStr)) { numN163Channels = int.Parse(numN163ChannelsStr); } project.SetExpansionAudioMask(expansionMask, numN163Channels); } if (!version.StartsWith("3.2")) { Log.LogMessage(LogSeverity.Error, "File was created with an incompatible version of FamiStudio. The text format is only compatible with the current version."); return(null); } break; } case "DPCMSample": { var str = parameters["Data"]; var data = new byte[str.Length / 2]; for (int i = 0; i < data.Length; i++) { data[i] = Convert.ToByte(str.Substring(i * 2, 2), 16); } var sample = project.CreateDPCMSampleFromDmcData(parameters["Name"], data); break; } case "DPCMMapping": { var pitch = 15; var loop = false; if (parameters.TryGetValue("Pitch", out var pitchStr)) { pitch = int.Parse(pitchStr); } if (parameters.TryGetValue("Loop", out var loopStr)) { loop = bool.Parse(loopStr); } project.MapDPCMSample(Note.FromFriendlyName(parameters["Note"]), project.GetSample(parameters["Sample"]), pitch, loop); break; } case "Instrument": { var instrumentExp = ExpansionType.None; if (parameters.TryGetValue("Expansion", out var instrumentExpStr)) { instrumentExp = ExpansionType.GetValueForShortName(instrumentExpStr); } instrument = project.CreateInstrument(instrumentExp, parameters["Name"]); if (instrument.IsFdsInstrument) { if (parameters.TryGetValue("FdsWavePreset", out var wavPresetStr)) { instrument.FdsWavePreset = (byte)WavePresetType.GetValueForName(wavPresetStr); } if (parameters.TryGetValue("FdsModPreset", out var modPresetStr)) { instrument.FdsWavePreset = (byte)WavePresetType.GetValueForName(modPresetStr); } if (parameters.TryGetValue("FdsMasterVolume", out var masterVolumeStr)) { instrument.FdsMasterVolume = byte.Parse(masterVolumeStr); } if (parameters.TryGetValue("FdsModSpeed", out var fdsModSpeedStr)) { instrument.FdsModSpeed = ushort.Parse(fdsModSpeedStr); } if (parameters.TryGetValue("FdsModDepth", out var fdsModDepthStr)) { instrument.FdsModDepth = byte.Parse(fdsModDepthStr); } if (parameters.TryGetValue("FdsModDelay", out var fdsModDelayStr)) { instrument.FdsModDelay = byte.Parse(fdsModDelayStr); } } else if (instrument.IsN163Instrument) { if (parameters.TryGetValue("N163WavePreset", out var wavPresetStr)) { instrument.N163WavePreset = (byte)WavePresetType.GetValueForName(wavPresetStr); } if (parameters.TryGetValue("N163WaveSize", out var n163WavSizeStr)) { instrument.N163WaveSize = byte.Parse(n163WavSizeStr); } if (parameters.TryGetValue("N163WavePos", out var n163WavPosStr)) { instrument.N163WavePos = byte.Parse(n163WavPosStr); } } else if (instrument.IsVrc6Instrument) { if (parameters.TryGetValue("Vrc6SawMasterVolume", out var vrc6SawVolumeStr)) { instrument.Vrc6SawMasterVolume = (byte)Vrc6SawMasterVolumeType.GetValueForName(vrc6SawVolumeStr); } } else if (instrument.IsVrc7Instrument) { if (parameters.TryGetValue("Vrc7Patch", out var vrc7PatchStr)) { instrument.Vrc7Patch = byte.Parse(vrc7PatchStr); } if (instrument.Vrc7Patch == Vrc7InstrumentPatch.Custom) { for (int i = 0; i < 8; i++) { if (parameters.TryGetValue($"Vrc7Reg{i}", out var regStr)) { instrument.Vrc7PatchRegs[i] = byte.Parse(regStr); } } } } break; } case "Arpeggio": { arpeggio = project.CreateArpeggio(parameters["Name"]); arpeggio.Envelope.Length = int.Parse(parameters["Length"]); if (parameters.TryGetValue("Loop", out var loopStr)) { arpeggio.Envelope.Loop = int.Parse(loopStr); } var values = parameters["Values"].Split(','); for (int j = 0; j < values.Length; j++) { arpeggio.Envelope.Values[j] = sbyte.Parse(values[j]); } break; } case "Envelope": { var env = instrument.Envelopes[EnvelopeType.GetValueForShortName(parameters["Type"])]; if (env != null) { if (env.CanResize) { env.Length = int.Parse(parameters["Length"]); } if (parameters.TryGetValue("Loop", out var loopStr)) { env.Loop = int.Parse(loopStr); } if (parameters.TryGetValue("Release", out var releaseStr)) { env.Release = int.Parse(releaseStr); } if (parameters.TryGetValue("Relative", out var relativeStr)) { env.Relative = bool.Parse(relativeStr); } var values = parameters["Values"].Split(','); for (int j = 0; j < values.Length; j++) { env.Values[j] = sbyte.Parse(values[j]); } } break; } case "Song": { song = project.CreateSong(parameters["Name"]); song.SetLength(int.Parse(parameters["Length"])); song.SetBeatLength(int.Parse(parameters["BeatLength"])); song.SetLoopPoint(int.Parse(parameters["LoopPoint"])); if (song.UsesFamiTrackerTempo) { song.SetDefaultPatternLength(int.Parse(parameters["PatternLength"])); song.FamitrackerTempo = int.Parse(parameters["FamiTrackerTempo"]); song.FamitrackerSpeed = int.Parse(parameters["FamiTrackerSpeed"]); } else { var noteLength = int.Parse(parameters["NoteLength"]); var groove = parameters["Groove"].Split('-').Select(Int32.Parse).ToArray(); var groovePaddingMode = GroovePaddingType.GetValueForName(parameters["GroovePaddingMode"]); if (!FamiStudioTempoUtils.ValidateGroove(groove) || Utils.Min(groove) != noteLength) { Log.LogMessage(LogSeverity.Error, "Invalid tempo settings."); return(null); } song.ChangeFamiStudioTempoGroove(groove, false); song.SetBeatLength(song.BeatLength * noteLength); song.SetDefaultPatternLength(int.Parse(parameters["PatternLength"]) * noteLength); song.SetGroovePaddingMode(groovePaddingMode); } break; } case "PatternCustomSettings": { if (project.UsesFamiTrackerTempo) { var beatLength = song.BeatLength; if (parameters.TryGetValue("BeatLength", out var beatLengthStr)) { beatLength = int.Parse(beatLengthStr); } song.SetPatternCustomSettings(int.Parse(parameters["Time"]), int.Parse(parameters["Length"]), beatLength); } else { var patternLength = int.Parse(parameters["Length"]); var noteLength = int.Parse(parameters["NoteLength"]); var beatLength = int.Parse(parameters["BeatLength"]); var groove = parameters["Groove"].Split('-').Select(Int32.Parse).ToArray(); var groovePaddingMode = GroovePaddingType.GetValueForName(parameters["GroovePaddingMode"]); if (!FamiStudioTempoUtils.ValidateGroove(groove) || Utils.Min(groove) != noteLength) { Log.LogMessage(LogSeverity.Error, "Invalid tempo settings."); return(null); } song.SetPatternCustomSettings(int.Parse(parameters["Time"]), patternLength * noteLength, beatLength * noteLength, groove, groovePaddingMode); } break; } case "Channel": { var channelType = ChannelType.GetValueForShortName(parameters["Type"]); channel = song.GetChannelByType(channelType); break; } case "Pattern": { pattern = channel.CreatePattern(parameters["Name"]); break; } case "Note": { var time = int.Parse(parameters["Time"]); var note = pattern.GetOrCreateNoteAt(time); if (parameters.TryGetValue("Value", out var valueStr)) { note.Value = (byte)Note.FromFriendlyName(valueStr); } if (note.IsMusical && parameters.TryGetValue("Duration", out var durationStr)) { note.Duration = int.Parse(durationStr); } else if (note.IsStop) { note.Duration = 1; } if (note.IsMusical && parameters.TryGetValue("Release", out var releaseStr)) { note.Release = int.Parse(releaseStr); } if (note.IsMusical && parameters.TryGetValue("Instrument", out var instStr) && channel.SupportsInstrument(project.GetInstrument(instStr))) { note.Instrument = project.GetInstrument(instStr); } if (note.IsMusical && parameters.TryGetValue("Arpeggio", out var arpStr) && channel.SupportsArpeggios) { note.Arpeggio = project.GetArpeggio(arpStr); } if (note.IsMusical && parameters.TryGetValue("SlideTarget", out var slideStr) && channel.SupportsSlideNotes) { note.SlideNoteTarget = (byte)Note.FromFriendlyName(slideStr); } if (note.IsMusical && parameters.TryGetValue("Attack", out var attackStr)) { note.HasAttack = bool.Parse(attackStr); } if (parameters.TryGetValue("Volume", out var volumeStr) && channel.SupportsEffect(Note.EffectVolume)) { note.Volume = byte.Parse(volumeStr); } if (parameters.TryGetValue("VolumeSlideTarget", out var volumeSlideStr) && channel.SupportsEffect(Note.EffectVolumeSlide)) { note.VolumeSlideTarget = byte.Parse(volumeSlideStr); } if (parameters.TryGetValue("VibratoSpeed", out var vibSpeedStr) && channel.SupportsEffect(Note.EffectVibratoSpeed)) { note.VibratoSpeed = byte.Parse(vibSpeedStr); } if (parameters.TryGetValue("VibratoDepth", out var vibDepthStr) && channel.SupportsEffect(Note.EffectVibratoDepth)) { note.VibratoDepth = byte.Parse(vibDepthStr); } if (parameters.TryGetValue("Speed", out var speedStr) && channel.SupportsEffect(Note.EffectSpeed)) { note.Speed = byte.Parse(speedStr); } if (parameters.TryGetValue("FinePitch", out var finePitchStr) && channel.SupportsEffect(Note.EffectFinePitch)) { note.FinePitch = sbyte.Parse(finePitchStr); } if (parameters.TryGetValue("FdsModSpeed", out var modSpeedStr) && channel.SupportsEffect(Note.EffectFdsModSpeed)) { note.FdsModSpeed = ushort.Parse(modSpeedStr); } if (parameters.TryGetValue("FdsModDepth", out var modDepthStr) && channel.SupportsEffect(Note.EffectFdsModDepth)) { note.FdsModDepth = byte.Parse(modDepthStr); } if (parameters.TryGetValue("DutyCycle", out var dutyCycleStr) && channel.SupportsEffect(Note.EffectDutyCycle)) { note.DutyCycle = byte.Parse(dutyCycleStr); } if (parameters.TryGetValue("NoteDelay", out var noteDelayStr) && channel.SupportsEffect(Note.EffectNoteDelay)) { note.NoteDelay = byte.Parse(noteDelayStr); } if (parameters.TryGetValue("CutDelay", out var cutDelayStr) && channel.SupportsEffect(Note.EffectCutDelay)) { note.CutDelay = byte.Parse(cutDelayStr); } break; } case "PatternInstance": { var time = int.Parse(parameters["Time"]); channel.PatternInstances[time] = channel.GetPattern(parameters["Pattern"]); break; } } } project.SortEverything(false); ResetCulture(); return(project); } #if !DEBUG catch (Exception e) { Log.LogMessage(LogSeverity.Error, "Please contact the developer on GitHub!"); Log.LogMessage(LogSeverity.Error, e.Message); Log.LogMessage(LogSeverity.Error, e.StackTrace); ResetCulture(); return(null); } #endif }
public Instrument CreateFromFile(Project project, string filename) { var bytes = System.IO.File.ReadAllBytes(filename); if (!bytes.Skip(0).Take(3).SequenceEqual(Encoding.ASCII.GetBytes("FTI"))) { Log.LogMessage(LogSeverity.Error, "Incompatible file."); return(null); } if (!bytes.Skip(3).Take(3).SequenceEqual(Encoding.ASCII.GetBytes("2.4"))) { Log.LogMessage(LogSeverity.Error, "Incompatible FTI version."); return(null); } var instType = InstrumentTypeLookup[bytes[6]]; // Needs to match the current expansion audio. Our enum happens to match (-1) for now. if (instType != ExpansionType.None && instType != project.ExpansionAudio) { Log.LogMessage(LogSeverity.Error, "Audio expansion does not match the current project expansion."); return(null); } var offset = 7; var nameLen = BitConverter.ToInt32(bytes, offset); offset += 4; var name = Encoding.ASCII.GetString(bytes, offset, nameLen); offset += nameLen; if (project.GetInstrument(name) != null) { Log.LogMessage(LogSeverity.Error, $"An instrument named '{name}' already exists."); return(null); } var instrument = project.CreateInstrument(instType, name); if (instType == ExpansionType.Fds) { var wavEnv = instrument.Envelopes[EnvelopeType.FdsWaveform]; for (int i = 0; i < wavEnv.Length; i++) { wavEnv.Values[i] = (sbyte)bytes[offset++]; } var modEnv = instrument.Envelopes[EnvelopeType.FdsModulation]; for (int i = 0; i < modEnv.Length; i++) { modEnv.Values[i] = (sbyte)bytes[offset++]; } instrument.FdsWavePreset = WavePresetType.Custom; instrument.FdsModPreset = WavePresetType.Custom; modEnv.ConvertFdsModulationToAbsolute(); // Skip mod speed/depth/delay. offset += sizeof(int) * 3; ReadEnvelope(bytes, ref offset, instrument, EnvelopeType.Volume); ReadEnvelope(bytes, ref offset, instrument, EnvelopeType.Arpeggio); ReadEnvelope(bytes, ref offset, instrument, EnvelopeType.Pitch); } else if (instType == ExpansionType.None || instType == ExpansionType.N163) { var seqCount = bytes[offset++]; if (seqCount != SEQ_COUNT) { Log.LogMessage(LogSeverity.Error, $"Unexpected number of envelopes ({seqCount})."); return(null); } // Envelopes for (int i = 0; i < SEQ_COUNT; i++) { if (bytes[offset++] == 1) { ReadEnvelope(bytes, ref offset, instrument, EnvelopeTypeLookup[i]); } } } else if (instType == ExpansionType.Vrc7) { instrument.Vrc7Patch = (byte)BitConverter.ToInt32(bytes, offset); offset += 4; if (instrument.Vrc7Patch == 0) { for (int i = 0; i < 8; ++i) { instrument.Vrc7PatchRegs[i] = bytes[offset++]; } } } if (instType == ExpansionType.N163) { int waveSize = BitConverter.ToInt32(bytes, offset); offset += 4; int wavePos = BitConverter.ToInt32(bytes, offset); offset += 4; int waveCount = BitConverter.ToInt32(bytes, offset); offset += 4; instrument.N163WavePreset = WavePresetType.Custom; instrument.N163WaveSize = (byte)waveSize; instrument.N163WavePos = (byte)wavePos; var wavEnv = instrument.Envelopes[EnvelopeType.N163Waveform]; // Only read the first wave for now. for (int j = 0; j < waveSize; j++) { wavEnv.Values[j] = (sbyte)bytes[offset++]; } if (waveCount > 1) { Log.LogMessage(LogSeverity.Warning, $"Multiple N163 waveforms detected, only loading the first one."); } } // Samples if (instType == ExpansionType.None) { // Skip over the sample mappings for now, we will load them after the actual sample data. var assignedCount = BitConverter.ToInt32(bytes, offset); offset += 4; var mappingOffset = offset; offset += assignedCount * 4; var sampleCount = BitConverter.ToInt32(bytes, offset); offset += 4; var sampleMap = new DPCMSample[sampleCount]; for (int i = 0; i < sampleCount; i++) { var sampleIdx = BitConverter.ToInt32(bytes, offset); offset += 4; var sampleNameLen = BitConverter.ToInt32(bytes, offset); offset += 4; var sampleName = Encoding.ASCII.GetString(bytes, offset, sampleNameLen); offset += sampleNameLen; var sampleSize = BitConverter.ToInt32(bytes, offset); offset += 4; var sampleData = new byte[sampleSize]; Array.Copy(bytes, offset, sampleData, 0, sampleSize); offset += sampleSize; sampleMap[sampleIdx] = project.CreateDPCMSampleFromDmcData(sampleName, sampleData); } for (int i = 0; i < assignedCount; i++) { byte idx = bytes[mappingOffset++]; byte sample = bytes[mappingOffset++]; byte pitch = bytes[mappingOffset++]; byte delta = bytes[mappingOffset++]; if (project.NoteSupportsDPCM(idx + 1)) { project.MapDPCMSample(idx + 1, sampleMap[sample - 1], pitch & 0x0f, (pitch & 0x80) != 0); } } } project.SortInstruments(); return(instrument); }
public Project Load(string filename) { #if !DEBUG try #endif { var lines = File.ReadAllLines(filename); var parameters = new Dictionary <string, string>(); var project = (Project)null; var instrument = (Instrument)null; var arpeggio = (Arpeggio)null; var song = (Song)null; var channel = (Channel)null; var pattern = (Pattern)null; var version = "0.0.0"; var isVersion230OrNewer = false; var isVersion240OrNewer = false; var beatLengthAttributeName = "BeatLength"; foreach (var line in lines) { var cmd = SplitLine(line.Trim(), ref parameters); switch (cmd) { case "Project": { project = new Project(); parameters.TryGetValue("Version", out version); if (parameters.TryGetValue("Name", out var name)) { project.Name = name; } if (parameters.TryGetValue("Author", out var author)) { project.Author = author; } if (parameters.TryGetValue("Copyright", out var copyright)) { project.Copyright = copyright; } if (parameters.TryGetValue("Expansion", out var expansion)) { project.SetExpansionAudio(ExpansionType.GetValueForShortName(expansion)); } if (parameters.TryGetValue("TempoMode", out var tempoMode)) { project.TempoMode = TempoType.GetValueForName(tempoMode); } if (parameters.TryGetValue("PAL", out var pal)) { project.PalMode = bool.Parse(pal); } isVersion230OrNewer = string.CompareOrdinal(version, "2.3.0") >= 0; isVersion240OrNewer = string.CompareOrdinal(version, "2.4.0") >= 0; if (!isVersion230OrNewer) { beatLengthAttributeName = "BarLength"; } break; } case "DPCMSample": { var str = parameters["Data"]; var data = new byte[str.Length / 2]; for (int i = 0; i < data.Length; i++) { data[i] = Convert.ToByte(str.Substring(i * 2, 2), 16); } var sample = project.CreateDPCMSampleFromDmcData(parameters["Name"], data); // Any DPCM processing options are no longer supported in 2.4.0 and newer. We just import/export the processed DMC data. if (!isVersion240OrNewer && parameters.TryGetValue("ReverseBits", out var reverseBitStr)) { sample.ReverseBits = bool.Parse(reverseBitStr); sample.Process(); } break; } case "DPCMMapping": { var pitch = 15; var loop = false; if (parameters.TryGetValue("Pitch", out var pitchStr)) { pitch = int.Parse(pitchStr); } if (parameters.TryGetValue("Loop", out var loopStr)) { loop = bool.Parse(loopStr); } project.MapDPCMSample(Note.FromFriendlyName(parameters["Note"]), project.GetSample(parameters["Sample"]), pitch, loop); break; } case "Instrument": { instrument = project.CreateInstrument(parameters.TryGetValue("Expansion", out _) ? project.ExpansionAudio : ExpansionType.None, parameters["Name"]); if (instrument.ExpansionType == ExpansionType.Fds) { if (parameters.TryGetValue("FdsWavePreset", out var wavPresetStr)) { instrument.FdsWavePreset = (byte)WavePresetType.GetValueForName(wavPresetStr); } if (parameters.TryGetValue("FdsModPreset", out var modPresetStr)) { instrument.FdsWavePreset = (byte)WavePresetType.GetValueForName(modPresetStr); } if (parameters.TryGetValue("FdsMasterVolume", out var masterVolumeStr)) { instrument.FdsMasterVolume = byte.Parse(masterVolumeStr); } if (parameters.TryGetValue("FdsModSpeed", out var fdsModSpeedStr)) { instrument.FdsModSpeed = ushort.Parse(fdsModSpeedStr); } if (parameters.TryGetValue("FdsModDepth", out var fdsModDepthStr)) { instrument.FdsModDepth = byte.Parse(fdsModDepthStr); } if (parameters.TryGetValue("FdsModDelay", out var fdsModDelayStr)) { instrument.FdsModDelay = byte.Parse(fdsModDelayStr); } } else if (instrument.ExpansionType == ExpansionType.N163) { if (parameters.TryGetValue("N163WavePreset", out var wavPresetStr)) { instrument.N163WavePreset = (byte)WavePresetType.GetValueForName(wavPresetStr); } if (parameters.TryGetValue("N163WaveSize", out var n163WavSizeStr)) { instrument.N163WaveSize = byte.Parse(n163WavSizeStr); } if (parameters.TryGetValue("N163WavePos", out var n163WavPosStr)) { instrument.N163WavePos = byte.Parse(n163WavPosStr); } } else if (instrument.ExpansionType == ExpansionType.Vrc7) { if (parameters.TryGetValue("Vrc7Patch", out var vrc7PatchStr)) { instrument.Vrc7Patch = byte.Parse(vrc7PatchStr); } if (instrument.Vrc7Patch == Vrc7InstrumentPatch.Custom) { for (int i = 0; i < 8; i++) { if (parameters.TryGetValue($"Vrc7Reg{i}", out var regStr)) { instrument.Vrc7PatchRegs[i] = byte.Parse(regStr); } } } } break; } case "Arpeggio": { arpeggio = project.CreateArpeggio(parameters["Name"]); arpeggio.Envelope.Length = int.Parse(parameters["Length"]); if (parameters.TryGetValue("Loop", out var loopStr)) { arpeggio.Envelope.Loop = int.Parse(loopStr); } var values = parameters["Values"].Split(','); for (int j = 0; j < values.Length; j++) { arpeggio.Envelope.Values[j] = sbyte.Parse(values[j]); } break; } case "Envelope": { var env = instrument.Envelopes[EnvelopeType.GetValueForShortName(parameters["Type"])]; if (env != null) { if (env.CanResize) { env.Length = int.Parse(parameters["Length"]); } if (parameters.TryGetValue("Loop", out var loopStr)) { env.Loop = int.Parse(loopStr); } if (parameters.TryGetValue("Release", out var releaseStr)) { env.Release = int.Parse(releaseStr); } if (parameters.TryGetValue("Relative", out var relativeStr)) { env.Relative = bool.Parse(relativeStr); } var values = parameters["Values"].Split(','); for (int j = 0; j < values.Length; j++) { env.Values[j] = sbyte.Parse(values[j]); } } break; } case "Song": { song = project.CreateSong(parameters["Name"]); song.SetLength(int.Parse(parameters["Length"])); song.SetBeatLength(int.Parse(parameters[beatLengthAttributeName])); song.SetLoopPoint(int.Parse(parameters["LoopPoint"])); if (song.UsesFamiTrackerTempo) { song.SetDefaultPatternLength(int.Parse(parameters["PatternLength"])); song.FamitrackerTempo = int.Parse(parameters["FamiTrackerTempo"]); song.FamitrackerSpeed = int.Parse(parameters["FamiTrackerSpeed"]); } else { var noteLength = int.Parse(parameters["NoteLength"]); song.ResizeNotes(noteLength, false); song.SetBeatLength(song.BeatLength * noteLength); song.SetDefaultPatternLength(int.Parse(parameters["PatternLength"]) * noteLength); } break; } case "PatternCustomSettings": { if (project.UsesFamiTrackerTempo) { var beatLength = song.BeatLength; if (parameters.TryGetValue(beatLengthAttributeName, out var beatLengthStr)) { beatLength = int.Parse(beatLengthStr); } song.SetPatternCustomSettings(int.Parse(parameters["Time"]), int.Parse(parameters["Length"]), beatLength); } else { var patternLength = int.Parse(parameters["Length"]); var noteLength = int.Parse(parameters["NoteLength"]); var beatLength = int.Parse(parameters[beatLengthAttributeName]); song.SetPatternCustomSettings(int.Parse(parameters["Time"]), patternLength * noteLength, beatLength * noteLength, noteLength); } break; } case "Channel": { var channelType = ChannelType.GetValueForShortName(parameters["Type"]); channel = song.GetChannelByType(channelType); break; } case "Pattern": { pattern = channel.CreatePattern(parameters["Name"]); break; } case "Note": { var time = int.Parse(parameters["Time"]); var note = pattern.GetOrCreateNoteAt(time); if (parameters.TryGetValue("Value", out var valueStr)) { note.Value = (byte)Note.FromFriendlyName(valueStr); } if (parameters.TryGetValue("Instrument", out var instStr) && channel.SupportsInstrument(project.GetInstrument(instStr))) { note.Instrument = project.GetInstrument(instStr); } if (parameters.TryGetValue("Arpeggio", out var arpStr) && channel.SupportsArpeggios) { note.Arpeggio = project.GetArpeggio(arpStr); } if (parameters.TryGetValue("SlideTarget", out var slideStr) && channel.SupportsSlideNotes) { note.SlideNoteTarget = (byte)Note.FromFriendlyName(slideStr); } if (parameters.TryGetValue("Attack", out var attackStr)) { note.HasAttack = bool.Parse(attackStr); } if (parameters.TryGetValue("Volume", out var volumeStr) && channel.SupportsEffect(Note.EffectVolume)) { note.Volume = byte.Parse(volumeStr); } if (parameters.TryGetValue("VibratoSpeed", out var vibSpeedStr) && channel.SupportsEffect(Note.EffectVibratoSpeed)) { note.VibratoSpeed = byte.Parse(vibSpeedStr); } if (parameters.TryGetValue("VibratoDepth", out var vibDepthStr) && channel.SupportsEffect(Note.EffectVibratoDepth)) { note.VibratoDepth = byte.Parse(vibDepthStr); } if (parameters.TryGetValue("Speed", out var speedStr) && channel.SupportsEffect(Note.EffectSpeed)) { note.Speed = byte.Parse(speedStr); } if (parameters.TryGetValue("FinePitch", out var finePitchStr) && channel.SupportsEffect(Note.EffectFinePitch)) { note.FinePitch = sbyte.Parse(finePitchStr); } if (parameters.TryGetValue("FdsModSpeed", out var modSpeedStr) && channel.SupportsEffect(Note.EffectFdsModSpeed)) { note.FdsModSpeed = ushort.Parse(modSpeedStr); } if (parameters.TryGetValue("FdsModDepth", out var modDepthStr) && channel.SupportsEffect(Note.EffectFdsModDepth)) { note.FdsModDepth = byte.Parse(modDepthStr); } if (parameters.TryGetValue("DutyCycle", out var dutyCycleStr) && channel.SupportsEffect(Note.EffectDutyCycle)) { note.DutyCycle = byte.Parse(dutyCycleStr); } if (parameters.TryGetValue("NoteDelay", out var noteDelayStr) && channel.SupportsEffect(Note.EffectNoteDelay)) { note.NoteDelay = byte.Parse(noteDelayStr); } if (parameters.TryGetValue("CutDelay", out var cutDelayStr) && channel.SupportsEffect(Note.EffectCutDelay)) { note.CutDelay = byte.Parse(cutDelayStr); } break; } case "PatternInstance": { var time = int.Parse(parameters["Time"]); channel.PatternInstances[time] = channel.GetPattern(parameters["Pattern"]); break; } } } project.SortEverything(); return(project); } #if !DEBUG catch (Exception e) { Log.LogMessage(LogSeverity.Error, "Please contact the developer on GitHub!"); Log.LogMessage(LogSeverity.Error, e.Message); Log.LogMessage(LogSeverity.Error, e.StackTrace); return(null); } #endif }