Я хочу включить звук в презентацию в проектор, и я хочу, чтобы этот звук начинался на определенном слайде и длился несколько наложений или несколько слайдов. Я пытался сделать это с помощью media9
или multimedia
(оба должны, согласно документации, работать). Но в обоих случаях (см. MWE ниже) звук начинается, когда он должен начинаться, но останавливается, когда отображается элемент 2. Как мне сделать так, чтобы звук длился более одного слайда (для одного из пакетов media9
или multimedia
, мне все равно, какой)?
МВЭ для media9
:
\documentclass{beamer}
\usepackage{media9}
\begin{document}
\begin{frame}{1}
1
\end{frame}
\begin{frame}{2}
\includemedia[label=my_sound, width=1ex, height=1ex, transparent, activate=pageopen, deactivate=onclick, addresource=./Sounds/MITM.mp3,
flashvars={
source=./Sounds/MITM.mp3
&autoPlay=true
},]{}{APlayer.swf}
\begin{itemize}
\item<2-> 2
\item<3-> 3
\end{itemize}
\end{frame}
\begin{frame}{3}
3
\end{frame}
\begin{frame}{4}
4
\end{frame}
\end{document}
MWE для multimedia
: (здесь эта autostart
опция также не работает, и мне приходится вручную запускать звук, а без этой опции вообще нет звука inlinesound
)
\documentclass{beamer}
\usepackage{multimedia}
\begin{document}
\begin{frame}{1}
1
\end{frame}
\begin{frame}{2}
\sound[autostart, inlinesound, bitspersample=16, channels=2, encoding=Signed, samplingrate=48000]{A}{./Sounds/MITM.au}
\begin{itemize}
\item<2-> 2
\item<3-> 3
\end{itemize}
\end{frame}
\begin{frame}{3}
3
\end{frame}
\begin{frame}{4}
4
\end{frame}
\end{document}
Если это важно: я использую Adobe Reader (последней версии) на Windows 11.
решение1
Насколько мне известно, ни один из распространенных просмотрщиков PDF в настоящее время не поддерживает воспроизведение звука на нескольких слайдах. Однако довольно просто создать собственный просмотрщик, который может это поддерживать.
Сохраните следующее в файле с именем viewer.html
(или <anything>.html
, если вам так больше нравится):
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="
width=device-width,
initial-scale=1,
maximum-scale=1
">
<style>
/* Make the viewer take up the full screen */
* {
margin: 0;
padding: 0;
border: 0;
outline: 0;
}
#viewer-container {
overflow: clip;
position: absolute;
width: 100%;
height: 100%;
}
/* Center the page */
.page {
margin: 0 auto 0;
}
</style>
<script type="module">
import "https://cdnjs.cloudflare.com/ajax/libs/pdf.js/4.1.392/pdf.mjs"
import "https://cdnjs.cloudflare.com/ajax/libs/pdf.js/4.1.392/pdf_viewer.mjs"
pdfjsLib.GlobalWorkerOptions.workerSrc =
"https://cdnjs.cloudflare.com/ajax/libs/pdf.js/4.1.392/pdf.worker.mjs"
const bus = new pdfjsViewer.EventBus()
const container = document.getElementById("viewer-container")
const viewer = new pdfjsViewer.PDFSinglePageViewer({
container,
eventBus: bus,
textLayerMode: 0,
annotationMode: 0,
})
bus.on("pagechanging", () => {
viewer.currentScaleValue = "page-fit"
})
const file_element = document.getElementById("file")
async function open(event) {
let file
if (event instanceof window.DragEvent) {
event.dataTransfer.dropEffect = "copy"
file = event.dataTransfer.files[0]
} else if (event.target instanceof window.HTMLInputElement) {
file = event.target.files[0]
}
file_element.remove()
const doc = await pdfjsLib.getDocument(
await file.arrayBuffer()
).promise
viewer.setDocument(doc)
const attatchments = await doc.getAttachments()
let audio
async function sound() {
viewer.currentScaleValue = "page-fit"
const page = await doc.getPage(viewer.currentPageNumber)
const annotations = await page.getAnnotations()
for (const annotation of annotations) {
const directive = annotation?.contentsObj?.str
if (!directive) continue
const [cmd, file] = directive.split(" ")
if (["play", "stop"].includes(cmd) && audio) {
audio.pause()
audio = null
}
if (cmd === "play") {
const blob = new Blob([attatchments[file].content])
audio = new Audio(URL.createObjectURL(blob))
audio.play()
}
}
}
function move(direction) {
if (!document.fullscreenElement) {
document.documentElement.requestFullscreen()
} else {
if (direction === "next") {
viewer.nextPage()
} else if (direction === "previous") {
viewer.previousPage()
}
}
sound()
}
document.addEventListener("keydown", (event) => {
switch (event.key) {
case "ArrowRight":
case "ArrowDown":
case " ":
case "PageDown":
case "Enter":
event.preventDefault()
move("next")
break
case "ArrowLeft":
case "ArrowUp":
case "PageUp":
case "Backspace":
event.preventDefault()
move("previous")
break
}
})
document.addEventListener("click", (event) => {
event.preventDefault()
switch (event.button) {
case 0:
event.preventDefault()
move("next")
break
case 2:
event.preventDefault()
move("previous")
break
}
})
}
file_element.addEventListener("change", open)
document.addEventListener("drop", open)
document.addEventListener("dragover", event => event.preventDefault())
</script>
</head>
<body>
<input type="file" id="file" accept="application/pdf">
<div id="viewer-container">
<div id="viewer"></div>
</div>
</body>
</html>
Затем скомпилируйте следующий документ с помощью pdfLaTeX или LuaLaTeX:
\documentclass{beamer}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%% Load required packages %%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% Needed to embed the audio files
\usepackage{embedfile}
%% Needed for \pdfannot to work with LuaTeX
\ifdefined\directlua
\usepackage{luatex85}
\fi
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%% Variable Declarations %%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\ExplSyntaxOn
\seq_new:N \g__example_files_seq %% Store all used files
\tl_new:N \l__example_lastfile_tl %% The most recent file loaded
\tl_new:N \g__example_currentfile_tl %% The currently playing file
%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%% Command Definitions %%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% Embeds a file if it isn't already embedded, and stores its attachment
%% name in "\l__example_lastfile_tl".
\cs_new_protected:Nn \__example_add_file:n {
\tl_set:Ne \l__example_lastfile_tl { \str_mdfive_hash:n { #1 } }
\seq_if_in:NVF \g__example_files_seq \l__example_lastfile_tl {
\seq_gput_right:NV \g__example_files_seq \l__example_lastfile_tl
\embedfile [ filespec = \l__example_lastfile_tl ] { #1 }
}
}
%% Writes to the PDF the given command and parameter.
\cs_new_protected:Nn \__example_pdfcmd:nn {
\pdfannot width 0pt~ height 0pt~ depth 0pt~ {%
/Subtype~ /Text
/Contents~ (#1~ #2)
/F~ 1
}
}
\cs_generate_variant:Nn \__example_pdfcmd:nn { nV }
%% Plays the given audio file.
\cs_new_protected:Nn \__example_play:n {
\__example_add_file:n { #1 }
\tl_if_eq:NNF \g__example_currentfile_tl \l__example_lastfile_tl {
\tl_gset:NV \g__example_currentfile_tl \l__example_lastfile_tl
\__example_pdfcmd:nV { play } \l__example_lastfile_tl
}
}
%% Stops the currently playing audio file.
\cs_new_protected:Nn \__example_stop: {
\__example_pdfcmd:nn { stop } { }
\tl_gclear:N \g__example_currentfile_tl
}
%% Expose the commands publicly
\cs_set_eq:NN \playaudio \__example_play:n
\cs_set_eq:NN \stopaudio \__example_stop:
\ExplSyntaxOff
%%%%%%%%%%%%%%%%%%%%%
%%% Demonstration %%%
%%%%%%%%%%%%%%%%%%%%%
%% There are no audio files in $TEXMFDIST, so we need to lookup the
%% (system-dependent) path to a sample audio file.
\ExplSyntaxOn
\sys_get_shell:nnN { kpsewhich~ --format=source~ bird.mp3 } { } \l_tmpa_tl
\iow_term:e {
\iow_newline:
``bird.mp3''~ path:~ \tl_to_str:N \l_tmpa_tl
\iow_newline:
}
\ExplSyntaxOff
\begin{document}
\begin{frame}{Title One}
Body One
\end{frame}
\begin{frame}{Title Two}
%% Replace this path with the path output to the terminal earlier
\playaudio{/usr/local/texlive/2024/texmf-dist/source/latex/media9/files/bird.mp3}
Body Two A
\begin{itemize}
\item<2-> Body Two B
\item<3-> Body Two C
\end{itemize}
\end{frame}
\begin{frame}{Title Three}
\stopaudio
Body Three
\end{frame}
\begin{frame}{Title Four}
Body Four
\end{frame}
\end{document}
Наконец, откройте viewer.html
веб-браузер, нажмите «Обзор» и откройте файл PDF, который вы скомпилировали выше. Щелчок в любом месте развернет презентацию на весь экран, любой из [ Left Click, →, Page ↓, Space, Enter] переместит вас на один слайд вперед, а любой из [ ←, Page ↑, Backspace] переместит вас на один слайд назад.
Как это работает
На стороне TeX мы добавляем звуковой файл как «вложение» PDF-файла, затем помещаем команду воспроизведения/остановки внутри скрытой «аннотации» PDF-файла.
Что касается HTML, мы импортируем pdf.js, чтобы использовать его в качестве основы для нашего средства просмотра, затем настраиваем его для анализа аннотаций на каждой странице и используем их для принятия решения о том, когда воспроизводить/останавливать любые прикрепленные аудиофайлы.
Зритель - этооченьbarebones и не очень надежный, но в целом его должно быть достаточно для простого слайд-шоу. Добавление новых функций и повышение надежности оставлено в качестве упражнения для читателя :)
решение2
Это не ответ, а мое решение проблемы. Я столкнулся с описанной проблемой и после довольно долгих попыток пришел к выводу, что она на самом деле не входит в область просмотра PDF в настоящее время.
Поэтому я перешел с PDF-презентаций на основе бимера на веб-презентации с использованием фреймворков вроде reveal.js. Это имеет недостатки, поскольку типичные движки рендеринга Latex на основе JS предлагают только подмножество Latex, поэтому время производства слайдов увеличилось.
Главный недостаток нового подхода в том, что он более трудоемкий. То есть, если мне нужны определенные латексные рендеры, не поддерживаемые KaTex или аналогичными латексными веб-рендерами, я сначала рендерю стандартный латекс в файл PNG, вставляю как изображение на слайд, где это необходимо. Большим преимуществом является то, что добавление аудио- и видеоконтента стало простым, а добавление интерактивного контента (в виде программ javascript) — вполне выполнимая, но индивидуальная работа. Моя работа часто включает в себя алгоритмические реализации, поэтому приятно иметь возможность бесшовно показывать работу в презентации (без alt-tab и т. д.). Reveal.js имеет библиотеки, которые даже позволяют делать разметку и рисовать поверх анимированного и интерактивного контента. Я надеюсь, что главный недостаток этого подхода будет преодолен со временем, оставив вариант, который предлагает лучшее из всех миров.