-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathbmc.cpp
125 lines (108 loc) · 4.24 KB
/
bmc.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
// Copyright (C) 2011-2012 Mario Lang <[email protected]>
//
// Distributed under the terms of the GNU General Public License version 3.
// (see accompanying file LICENSE.txt or copy at
// http://www.gnu.org/licenses/gpl-3.0-standalone.html)
#include <string>
#include "config.hpp"
#include <fstream>
#include <boost/spirit/include/qi_parse.hpp>
#include <bmc/braille/text2braille.hpp>
#include "bmc/braille/parsing/grammar/score.hpp"
#include "bmc/braille/reformat.hpp"
#include "bmc/braille/semantic_analysis.hpp"
#include <boost/program_options.hpp>
#include "bmc/lilypond.hpp"
#include "bmc/musicxml.hpp"
#include <boost/locale/encoding_utf.hpp>
using boost::locale::conv::utf_to_utf;
namespace {
int bmc2ly( std::istream &istream
, bool lilypond, bool musicxml
, bool include_locations, std::string instrument, bool no_tagline
, ::bmc::braille::format_style const &style
) {
std::istreambuf_iterator<char> cin_begin(istream.rdbuf()), cin_end;
auto const utf8 = std::string(cin_begin, cin_end);
auto const source = utf_to_utf<wchar_t>(utf8);
typedef std::wstring::const_iterator iterator_type;
iterator_type iter = source.begin();
iterator_type const end = source.end();
typedef ::bmc::braille::error_handler<iterator_type> error_handler_type;
error_handler_type error_handler(iter, end);
typedef ::bmc::braille::score_grammar<iterator_type> parser_type;
parser_type parser(error_handler);
boost::spirit::traits::attribute_of<parser_type>::type score;
bool const success = parse(iter, end, parser, score);
if (success && iter == end) {
::bmc::braille::compiler<error_handler_type> compile(error_handler);
if (compile(score)) {
std::wcerr << error_handler;
if (lilypond) {
::bmc::lilypond::generator generate(std::cout, true, true, include_locations);
if (!instrument.empty()) generate.instrument(instrument);
if (no_tagline) generate.remove_tagline();
generate(score);
} else if (musicxml) {
::bmc::musicxml(std::cout, score);
} else {
std::cout << ::bmc::braille::reformat(score, style);
}
return EXIT_SUCCESS;
}
std::wcerr << "Failed to compile:" << std::endl << error_handler << std::endl;
} else {
std::wcerr << "Failed to Parse:" << std::endl << source << std::endl;
}
return EXIT_FAILURE;
}
} // namespace
int main(int argc, char const *argv[])
{
std::locale::global(std::locale(""));
bmc::braille::set_default_table_from_locale();
using namespace boost::program_options;
std::string instrument;
bool locations;
bool no_tagline = false;
::bmc::braille::format_style style;
std::vector<std::string> input_files;
options_description desc("Allowed options");
desc.add_options()
("help,h", "print usage message")
("locale", value(&bmc::braille::default_table)->default_value(bmc::braille::default_table), "Braille locale")
("input-file", value(&input_files), "input file")
("instrument,i", value(&instrument), "default MIDI instrument")
("lilypond", "Produce LilyPond output.")
("musicxml", "Produce MusicXML output.")
("locations,l", bool_switch(&locations), "Include braille locations in LilyPond output")
("no-tagline", bool_switch(&no_tagline)->default_value(false), "Supress LilyPond default tagline")
("width,w", value(&style.columns), "Line width for reformatting")
;
positional_options_description positional_desc;
positional_desc.add("input-file", -1);
variables_map vm;
try {
store(command_line_parser(argc, argv)
.options(desc).positional(positional_desc).run(), vm);
notify(vm);
} catch (unknown_option) {
std::cerr << "Unknown option" << std::endl << desc << std::endl;
return EXIT_FAILURE;
}
if (vm.count("help")) {
std::cout << desc << std::endl;
return 0;
}
int status = EXIT_SUCCESS;
bool const do_lilypond { bool(vm.count("lilypond")) }
, do_musicxml { bool(vm.count("musicxml")) };
for (auto const &file: input_files) {
if (file == "-") status = bmc2ly(std::cin, do_lilypond, do_musicxml, locations, instrument, no_tagline, style);
else {
std::ifstream f(file);
if (f.good()) status = bmc2ly(f, do_lilypond, do_musicxml, locations, instrument, no_tagline, style);
}
}
return status;
}