File size: 8,264 Bytes
508b543
 
 
 
9a2a955
508b543
 
c1ba145
508b543
1f8010e
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
c1ba145
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
508b543
1f8010e
 
 
 
508b543
 
1f8010e
508b543
 
 
 
 
 
 
 
 
 
 
 
a4dbe50
cf01dad
 
 
 
a4dbe50
 
 
 
 
 
 
 
 
 
 
 
 
 
cf01dad
 
 
 
 
 
 
 
 
 
 
508b543
 
cf01dad
 
 
 
 
 
 
 
 
 
508b543
 
 
6156ecf
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
508b543
 
 
 
6fe938d
508b543
 
 
6fe938d
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
508b543
 
 
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
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
import gradio as gr
import os
import shutil
import subprocess
import sys
import tempfile
from zipfile import ZipFile
from pathlib import Path

# 程序根目录
BASE_DIR = os.path.dirname(__file__)

# 可执行文件路径:根据操作系统确定
if sys.platform.startswith('win'):
    BINARY_PATH = os.path.join(BASE_DIR, "PptxToPDF.exe")
else:
    BINARY_PATH = os.path.join(BASE_DIR, "PptxToPDF")

# 启动时确保二进制有执行权限(在 Hugging Face Spaces 里无法手动 chmod)
try:
    if not sys.platform.startswith('win'):
        os.chmod(BINARY_PATH, 0o755)
except Exception as e:
    # 如果权限设置失败,也无需中断,只要后续能调用即可
    print(f"⚠️ 警告:设置执行权限失败:{e}")

def install_fonts_from_repository():
    """

    将仓库中的 fonts 目录下的所有字体文件复制到 ~/.fonts 目录,并刷新字体缓存。

    """
    try:
        # 获取当前用户的主目录路径
        home_dir = Path.home()

        # 定义目标字体目录路径:~/.fonts
        fonts_dir = home_dir / ".fonts"

        # 如果 ~/.fonts 目录不存在,则创建该目录
        if not fonts_dir.exists():
            fonts_dir.mkdir(parents=True)
            print(f"已创建字体目录:{fonts_dir}")
        else:
            print(f"字体目录已存在:{fonts_dir}")

        # 定义仓库中的 fonts 目录路径
        repo_fonts_dir = Path(__file__).parent / "fonts"

        # 检查仓库中的 fonts 目录是否存在
        if not repo_fonts_dir.exists():
            print(f"错误:仓库中的 fonts 目录不存在:{repo_fonts_dir}")
            return

        # 支持的字体文件扩展名
        font_extensions = [".ttf", ".otf", ".ttc"]

        # 统计已复制的字体文件数量
        copied_count = 0

        # 遍历 fonts 目录中的所有文件
        for font_file in repo_fonts_dir.iterdir():
            if font_file.suffix.lower() in font_extensions:
                destination = fonts_dir / font_file.name
                try:
                    shutil.copy(font_file, destination)
                    print(f"已复制字体文件:{font_file.name}{destination}")
                    copied_count += 1
                except Exception as e:
                    print(f"复制字体文件时出错:{font_file.name},错误信息:{e}")

        if copied_count == 0:
            print("未找到任何可复制的字体文件。")
            return

        # 刷新字体缓存(仅在 Linux/Mac 下)
        if not sys.platform.startswith('win'):
            try:
                print("正在刷新字体缓存...")
                subprocess.run(["fc-cache", "-f", "-v"], check=True)
                print("字体缓存刷新完成。")
            except subprocess.CalledProcessError as e:
                print(f"刷新字体缓存时出错:{e}")
    except Exception as e:
        print(f"安装字体时出错:{e}")

# 启动时安装字体
install_fonts_from_repository()

def convert_pptx_to_pdf(pptx_path, output_dir):
    # 检查可执行文件是否存在
    if not os.path.exists(BINARY_PATH):
        raise FileNotFoundError(f"PptxToPDF 可执行文件未找到: {BINARY_PATH}")
    
    pdf_path = os.path.join(output_dir, os.path.splitext(os.path.basename(pptx_path))[0] + ".pdf")
    result = subprocess.run([
        BINARY_PATH,
        pptx_path
    ], cwd=output_dir, capture_output=True, text=True)
    if result.returncode != 0 or not os.path.exists(pdf_path):
        raise RuntimeError(f"Conversion failed: {result.stderr}")
    return pdf_path

def make_zip_file(files, zip_path):
    with ZipFile(zip_path, 'w') as zipf:
        for f in files:
            zipf.write(f, os.path.basename(f))
    return zip_path

def pptx2pdf_web(pptx_file): # pptx_file is the Gradio file object
    # Create a temporary directory that won't be auto-deleted
    tmpdir = tempfile.mkdtemp()
    
    try:
        gradio_temp_file_path = pptx_file.name # Path to the file uploaded by Gradio
        
        # Create a path for our copy of the PPTX inside our own tmpdir
        # Use the original basename of the uploaded file
        target_pptx_filename = os.path.basename(gradio_temp_file_path)
        our_copy_of_pptx_path = os.path.join(tmpdir, target_pptx_filename)

        # Copy the uploaded PPTX from Gradio's temp location to our temp location
        shutil.copyfile(gradio_temp_file_path, our_copy_of_pptx_path)
        
        # Now, our_copy_of_pptx_path is the path to the PPTX file within tmpdir.
        # convert_pptx_to_pdf will create the PDF also within tmpdir.
        pdf_path = convert_pptx_to_pdf(our_copy_of_pptx_path, tmpdir)
        
        # Create persistent copies of the output files for Gradio
        import tempfile as tf
        
        # Create persistent PDF file
        pdf_fd, persistent_pdf_path = tf.mkstemp(suffix='.pdf', prefix='converted_')
        os.close(pdf_fd)
        shutil.copy2(pdf_path, persistent_pdf_path)
        
        # Create persistent ZIP file
        zip_fd, persistent_zip_path = tf.mkstemp(suffix='.zip', prefix='converted_')
        os.close(zip_fd)
        zip_path = os.path.join(tmpdir, 'converted.zip')
        make_zip_file([pdf_path], zip_path)
        shutil.copy2(zip_path, persistent_zip_path)
        
        return persistent_pdf_path, persistent_zip_path
        
    finally:
        # Clean up our temporary directory
        try:
            shutil.rmtree(tmpdir)
        except:
            pass  # Ignore cleanup errors

with gr.Blocks(title="PPTX to PDF Converter - Easily Convert PowerPoint to PDF") as demo:
    gr.Markdown("# PPTX to PDF Converter") # Added main title here
    
    # Application Introduction and Instructions - 直接显示,不使用手风琴
    gr.Markdown("""

    ## Application Introduction and Instructions

    

    Welcome to the PPTX to PDF Converter! This tool allows you to easily convert your PowerPoint (PPTX) files into PDF documents.

    

    **Features:**

    - **Simple Upload:** Easily upload your PPTX file through the web interface.

    - **One-Click Conversion:** Convert your file to PDF with a single click.

    - **Direct PDF Download:** Download the converted PDF file directly.

    - **Zipped Archive:** Optionally, download a ZIP file containing the converted PDF for convenience.

    

    **How to Use:**

    1.  **Upload PPTX File:** Click on the "Upload PPTX File" area or drag and drop your `.pptx` file.

    2.  **Click Convert:** Press the "Convert" button to start the conversion process.

    3.  **Download Files:** Once the conversion is complete, download links for the PDF file and a ZIP archive (containing the PDF) will appear. Click on your preferred option to download.

    """)
    
    gr.Markdown("---")
    with gr.Row():
        with gr.Column():
            pptx_input = gr.File(label="Upload PPTX File", file_types=[".pptx"])
            convert_btn = gr.Button("Convert", variant="primary", size="lg")
        with gr.Column():
            pdf_output = gr.File(label="Download PDF File")
            zip_output = gr.File(label="Download ZIP Archive")
            # 添加漂亮的下载按钮
            with gr.Row():
                download_btn = gr.DownloadButton(
                    label="📦 Download ZIP Package", 
                    variant="secondary",
                    size="lg",
                    visible=False
                )
    
    def handle_conversion(pptx_file):
        if pptx_file is None:
            return None, None, gr.update(visible=False)
        
        pdf_path, zip_path = pptx2pdf_web(pptx_file)
        
        # 更新下载按钮,使其可见并设置下载文件
        return pdf_path, zip_path, gr.update(visible=True, value=zip_path)
    
    convert_btn.click(
        handle_conversion, 
        inputs=[pptx_input], 
        outputs=[pdf_output, zip_output, download_btn]
    )

demo.queue(default_concurrency_limit=10, max_size=50)
demo.launch(show_api=False)