File size: 3,976 Bytes
8b13692
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
125
126
127
128
129
130
131
132
133
import os
import sys
import fitz
import requests
import subprocess
from PIL import Image
from music21 import converter


def download(url: str, directory: str, filename: str):
    if directory != "" and not os.path.exists(directory):
        os.makedirs(directory)
    # Create the full path for the file to be saved
    file_path = os.path.join(directory, filename)
    # Send a GET request to the URL
    response = requests.get(url, stream=True)
    # Check if the request was successful
    if response.status_code == 200:
        # Open the file in write-binary mode
        with open(file_path, "wb") as file:
            # Write the contents of the response to the file
            for chunk in response.iter_content(chunk_size=1024):
                if chunk:  # Filter out keep-alive new chunks
                    file.write(chunk)

        print(f"The file has been downloaded and saved to {file_path}")

    else:
        print(f"Failed to download the file. Status code: {response.status_code}")

    return os.path.join(directory, filename)


if sys.platform.startswith("linux"):
    apkname = "MuseScore.AppImage"
    extra_dir = "squashfs-root"
    if not os.path.exists(apkname):
        download(
            url="https://www.modelscope.cn/studio/Genius-Society/piano_transcription/resolve/master/MuseScore.AppImage",
            directory="./",
            filename=apkname,
        )

    if not os.path.exists(extra_dir):
        subprocess.run(["chmod", "+x", f"./{apkname}"])
        subprocess.run([f"./{apkname}", "--appimage-extract"])

    MSCORE = f"./{extra_dir}/AppRun"
    os.environ["QT_QPA_PLATFORM"] = "offscreen"

else:
    MSCORE = os.getenv("mscore")


def add_title_to_xml(xml_path: str, title: str):
    midi_data = converter.parse(xml_path)
    # 将标题添加到 MIDI 文件中
    midi_data.metadata.movementName = title
    midi_data.metadata.composer = "Transcripted by AI"
    # 保存修改后的 MIDI 文件
    midi_data.write("musicxml", fp=xml_path)


def xml2abc(xml_path: str):
    result = subprocess.run(
        ["python", "xml2abc.py", xml_path], stdout=subprocess.PIPE, text=True
    )
    if result.returncode == 0:
        return result.stdout

    return ""


def xml2mxl(xml_path: str):
    mxl_file = xml_path.replace(".musicxml", ".mxl")
    command = [MSCORE, "-o", mxl_file, xml_path]
    result = subprocess.run(command)
    print(result)
    return mxl_file


def midi2xml(mid_file: str, title: str):
    xml_file = mid_file.replace(".mid", ".musicxml")
    command = [MSCORE, "-o", xml_file, mid_file]
    result = subprocess.run(command)
    add_title_to_xml(xml_file, title)
    print(result)
    return xml_file


def xml2midi(xml_file: str):
    midi_file = xml_file.replace(".musicxml", ".mid")
    command = [MSCORE, "-o", midi_file, xml_file]
    result = subprocess.run(command)
    print(result)
    return midi_file


def pdf2img(pdf_path: str):
    output_path = pdf_path.replace(".pdf", ".jpg")
    doc = fitz.open(pdf_path)
    # 创建一个图像列表
    images = []
    for page_number in range(doc.page_count):
        page = doc[page_number]
        # 将页面渲染为图像
        image = page.get_pixmap()
        # 将图像添加到列表
        images.append(
            Image.frombytes("RGB", [image.width, image.height], image.samples)
        )
    # 竖向合并图像
    merged_image = Image.new(
        "RGB", (images[0].width, sum(image.height for image in images))
    )
    y_offset = 0
    for image in images:
        merged_image.paste(image, (0, y_offset))
        y_offset += image.height
    # 保存合并后的图像为JPG
    merged_image.save(output_path, "JPEG")
    # 关闭PDF文档
    doc.close()
    return output_path


def xml2jpg(xml_file: str):
    pdf_score = xml_file.replace(".musicxml", ".pdf")
    command = [MSCORE, "-o", pdf_score, xml_file]
    result = subprocess.run(command)
    print(result)
    return pdf_score, pdf2img(pdf_score)