/* Copyright 2013-2019 Matt Tytel
 *
 * vital is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * vital 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 vital.  If not, see <http://www.gnu.org/licenses/>.
 */

#include "JuceHeader.h"
#include "load_save.h"
#include "tuning.h"
#include "synth_base.h"

String getArgumentValue(int argc, const char* argv[], const String& flag, const String& full_flag) {
  for (int i = 0; i < argc - 1; ++i) {
    std::string arg = argv[i];
    if (arg == flag || arg == full_flag)
      return argv[i + 1];
  }
  
  return "";
}

bool hasFlag(int argc, const char* argv[], const String& flag, const String& full_flag) {
  for (int i = 0; i < argc - 1; ++i) {
    std::string arg = argv[i];
    if (arg == flag || arg == full_flag)
      return true;
  }

  return false;
}

float getRenderLength(int argc, const char* argv[]) {
  static constexpr float kDefaultRenderLength = 5.0f;
  static constexpr float kMaxRenderLength = 15.0f;
  
  String string_length = getArgumentValue(argc, argv, "-l", "--length");
  float length = kDefaultRenderLength;
  if (string_length.isEmpty())
    return kDefaultRenderLength;
  
  float float_val = string_length.getFloatValue();
  if (float_val > 0.0f)
    length = std::min(float_val, kMaxRenderLength);
  
  return length;
}

std::vector<int> getRenderMidiNotes(int argc, const char* argv[]) {
  static constexpr int kDefaultMidiNote = 48;
  
  String string_midi = getArgumentValue(argc, argv, "-m", "--midi");
  std::vector<int> midi_notes;
  if (!string_midi.isEmpty()) {
    StringArray midi_tokens;
    midi_tokens.addTokens(string_midi, ",", "");
    
    for (const String& midi_token : midi_tokens) {
      int midi = Tuning::noteToMidiKey(midi_token);
      if (midi >= 0)
        midi_notes.push_back(midi);
    }
  }
  
  if (midi_notes.empty())
    midi_notes.push_back(kDefaultMidiNote);
  
  return midi_notes;
}

float getRenderBpm(int argc, const char* argv[]) {
  static constexpr float kDefaultBpm = 120.0f;
  static constexpr float kMinBpm = 5.0f;
  static constexpr float kMaxBpm = 900.0f;

  String string_length = getArgumentValue(argc, argv, "-b", "--bpm");
  float bpm = kDefaultBpm;
  if (string_length.isEmpty())
    return kDefaultBpm;

  bpm = std::min(string_length.getFloatValue(), kMaxBpm);
  return std::max(bpm, kMinBpm);
}

void doRenderToFile(HeadlessSynth& headless_synth, int argc, const char* argv[]) {
  String string_output_file = getArgumentValue(argc, argv, "-o", "--output");
  bool render_images = hasFlag(argc, argv, "-i", "--render-images");

  if (string_output_file.isEmpty())
    return;
  
  if (!string_output_file.startsWith("/"))
    string_output_file = "./" + string_output_file;
  
  File output_file(string_output_file);
  if (!output_file.hasWriteAccess()) {
    std::cout << "Error: Don't have permission to write output file." << newLine;
    return;
  }

  float length = getRenderLength(argc, argv);
  float bpm = getRenderBpm(argc, argv);
  std::vector<int> midi_notes = getRenderMidiNotes(argc, argv);
  
  headless_synth.renderAudioToFile(output_file, length, bpm, midi_notes, render_images);
}

bool loadFromCommandLine(HeadlessSynth& synth, const String& command_line) {
  String file_path = command_line;
  if (file_path[0] == '"' && file_path[file_path.length() - 1] == '"')
    file_path = command_line.substring(1, command_line.length() - 1);
  File file = File::getCurrentWorkingDirectory().getChildFile(file_path);
  if (!file.exists())
    return false;
  
  std::string error;
  synth.loadFromFile(file, error);
  return true;
}

int main(int argc, const char* argv[]) {
  HeadlessSynth headless_synth;
  
  bool last_arg_was_option = false;
  for (int i = 1; i < argc; ++i) {
    std::string arg = argv[i];
    if (arg != "" && arg[0] != '-' && !last_arg_was_option && loadFromCommandLine(headless_synth, arg))
      break;

    last_arg_was_option = arg[0] == '-' && arg != "--headless";
  }
  
  doRenderToFile(headless_synth, argc, argv);
}
