-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathMerge Media Files
More file actions
334 lines (297 loc) · 10.9 KB
/
Merge Media Files
File metadata and controls
334 lines (297 loc) · 10.9 KB
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
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
#!/usr/bin/env bash
# Merge Media Files — universal Nautilus/Nemo/Caja/Thunar/Dolphin/PCManFM wrapper
# - Erkennt Dateiauswahl aus Nautilus-Variablen ODER aus Argumenten (%F)
# - Funktioniert somit ohne install.sh in vielen Dateimanagern
# - Beibehaltung aller bisherigen Features (Re-Encode-Abfrage, Fortschrittsfenster, Logging)
set -o errexit -o nounset -o pipefail
# Soll der Ausgabedatei-Zeitstempel an den der ersten Quelldatei angeglichen werden?
PRESERVE_SOURCE_TIMESTAMP=true
LOGFILE="/tmp/catmerge.log"
exec >>"$LOGFILE" 2>&1
echo "---- $(date) start ----"
# -------------------- kleine Utils --------------------
bin_exists() { command -v "$1" >/dev/null 2>&1; }
print_ffmpeg_cmd() {
local -a args=("$@")
printf 'ffmpeg'
for arg in "${args[@]}"; do
printf ' %q' "$arg"
done
printf '\n'
}
# Verfügbares GUI-Tool ermitteln (yad bevorzugt, sonst zenity)
GUI_TOOL=""
if bin_exists yad; then
GUI_TOOL=yad
elif bin_exists zenity; then
GUI_TOOL=zenity
fi
# Prozent-Decoding (aus file:// URIs) – bevorzugt python3, Fallback: printf-basierte Dekodierung
uridecode() {
if bin_exists python3; then
python3 - "$@" <<'PY'
import sys, urllib.parse
for s in sys.argv[1:]:
if s.startswith('file://'):
s = urllib.parse.unquote(s[7:])
print(s)
PY
else
# Fallback: 'file://' entfernen und Prozent-Codes via printf dekodieren
for s in "$@"; do
s="${s#file://}"
# Ersetze %HH durch \xHH und lasse printf sie dekodieren
printf '%b\n' "${s//%/\\x}"
done
fi
}
# -------------------- Abhängigkeiten prüfen --------------------
need_cmds=("ffmpeg")
missing=()
for c in "${need_cmds[@]}"; do
bin_exists "$c" || missing+=("$c")
done
if ((${#missing[@]})); then
rpmfusion_hint=""
if printf '%s\n' "${missing[@]}" | grep -q '^ffmpeg$'; then
rpmfusion_hint=$'\nHinweis (Fedora): ffmpeg ist i. d. R. in RPM Fusion.\n sudo dnf install -y https://mirrors.rpmfusion.org/free/fedora/rpmfusion-free-release-$(rpm -E %fedora).noarch.rpm\n sudo dnf install -y ffmpeg'
fi
msg=$'Fehlende Abhängigkeiten: '"${missing[*]}"$'\n\nUbuntu/Debian: sudo apt install ffmpeg\nFedora: sudo dnf install ffmpeg\n (ffmpeg über RPM Fusion)\n'"${rpmfusion_hint}"
case "$GUI_TOOL" in
yad)
yad --title="Abhängigkeiten fehlen" --text="$msg" --button="OK":0 --center || true
;;
zenity)
zenity --info --title="Abhängigkeiten fehlen" --text="$msg" || true
;;
*)
echo "$msg" >&2
;;
esac
exit 1
fi
# -------------------- Dateiauswahl universell einsammeln --------------------
# Priorität:
# 1) NAUTILUS_SCRIPT_SELECTED_FILE_PATHS (Nautilus/Nemo/Caja)
# 2) NAUTILUS_SCRIPT_SELECTED_URIS (als file:// URIs)
# 3) Argumente (%F) aus Thunar/Dolphin/PCManFM(-Qt)
SELECTED_LINES=""
if [[ -n "${NAUTILUS_SCRIPT_SELECTED_FILE_PATHS:-}" ]]; then
SELECTED_LINES="$(printf '%s\n' "${NAUTILUS_SCRIPT_SELECTED_FILE_PATHS}" | sed '/^$/d')"
elif [[ -n "${NAUTILUS_SCRIPT_SELECTED_URIS:-}" ]]; then
# in Pfade umwandeln
mapfile -t paths < <(uridecode ${NAUTILUS_SCRIPT_SELECTED_URIS})
SELECTED_LINES="$(printf '%s\n' "${paths[@]}" | sed '/^$/d')"
elif (($#>0)); then
# Argumente als Pfade
mapfile -t paths < <(printf '%s\n' "$@")
SELECTED_LINES="$(printf '%s\n' "${paths[@]}" | sed '/^$/d')"
else
case "$GUI_TOOL" in
yad)
yad --image=dialog-error --text="Keine Dateien ausgewählt/übergeben." --button="OK":0 --center || true
;;
zenity)
zenity --error --text="Keine Dateien ausgewählt/übergeben." || true
;;
*)
echo "Keine Dateien ausgewählt/übergeben." >&2
;;
esac
exit 1
fi
# Sicherstellen, dass alle Pfade existieren (nur Dateien)
FILES_RAW="$(printf '%s\n' "${SELECTED_LINES}" | while IFS= read -r p; do [[ -f "$p" ]] && printf '%s\n' "$p"; done)"
if [[ -z "${FILES_RAW}" ]]; then
echo "Keine gültigen Dateien gefunden." >&2
exit 1
fi
# -------------------- Ausgangswerte/Metadaten --------------------
FIRST_FILE="$(printf '%s\n' "${FILES_RAW}" | head -n 1)"
DIRNAME="$(dirname "${FIRST_FILE}")"
EXTENSION_RAW="${FIRST_FILE##*.}"
EXTENSION="${EXTENSION_RAW,,}" # in Kleinbuchstaben
BASENAME="$(basename "${FIRST_FILE}" | sed -E 's/_[0-9]{3}\.[^.]+$//')"
TIMESTAMP="$(stat -c %y "${FIRST_FILE}" | cut -d' ' -f1)"
OUTPUT_FILE="${BASENAME}_${TIMESTAMP}.${EXTENSION}"
echo "PRESERVE_SOURCE_TIMESTAMP=$PRESERVE_SOURCE_TIMESTAMP"
# Sortierung (natürlich aufsteigend), nur existierende Dateien
SORTED_FILES="$(printf '%s\n' "${FILES_RAW}" | sort -V)"
echo "FIRST_FILE=$FIRST_FILE"
echo "DIRNAME=$DIRNAME"
echo "EXTENSION=$EXTENSION"
echo "OUTPUT_FILE=$OUTPUT_FILE"
# --- Anzeige der Reihenfolge (vertikale Liste, nur Basename, dynamische Höhe) ---
get_screen_height() {
if command -v xrandr >/dev/null 2>&1; then
# Beispielausgaben: "1920x1080" oder "connected primary 1920x1080+..."
xrandr | awk '/*/{if($1=="*"||$2=="*"){print $1; exit}}' 2>/dev/null | cut -d'x' -f2 || \
xrandr | awk '/current/{print $8; exit}' 2>/dev/null
elif command -v xdpyinfo >/dev/null 2>&1; then
xdpyinfo | awk '/dimensions:/{split($2,a,"x"); print a[2]; exit}' 2>/dev/null
else
echo 900
fi
}
SCREEN_H="$(get_screen_height)"
: "${SCREEN_H:=500}"
MAX_H="$(( SCREEN_H * 75 / 100 ))"
MIN_H=300
HEIGHT=$(( MAX_H > MIN_H ? MAX_H : MIN_H ))
# Nur die Basisnamen vorbereiten (robust gegen Leerzeichen)
TMP_NAMES="$(mktemp)"
while IFS= read -r f; do
[ -n "$f" ] || continue
printf '%s\n' "$(basename "$f")" >> "$TMP_NAMES"
done <<< "$SORTED_FILES"
# Liste per STDIN an GUI geben: eine Zeile = eine Tabellenzeile
case "$GUI_TOOL" in
yad)
yad --title="Zusammenfügen bestätigen" \
--text="Die folgenden Dateien werden in dieser Reihenfolge zusammengefügt:" \
--list --no-headers --no-click --scroll \
--column="Dateien" \
--width=500 --height="$HEIGHT" --center \
--button="Abbrechen":1 --button="OK":0 \
< "$TMP_NAMES" || { rm -f "$TMP_NAMES"; exit 1; }
;;
zenity)
zenity --list --title="Zusammenfügen bestätigen" \
--text="Die folgenden Dateien werden in dieser Reihenfolge zusammengefügt:" \
--column="Dateien" --hide-header \
--width=500 --height="$HEIGHT" \
--cancel-label="Abbrechen" --ok-label="OK" \
< "$TMP_NAMES" || { rm -f "$TMP_NAMES"; exit 1; }
;;
esac
rm -f "$TMP_NAMES"
# -------------------- Speicherplatz prüfen --------------------
REQUIRED_SPACE=0
while IFS= read -r f; do
sz=$(du -k -- "$f" | awk '{print $1}')
REQUIRED_SPACE=$((REQUIRED_SPACE + sz))
done <<< "${SORTED_FILES}"
AVAILABLE_SPACE=$(df -Pk -- "${DIRNAME}" | awk 'END{print $4}')
echo "REQUIRED_SPACE=${REQUIRED_SPACE} KB, AVAILABLE_SPACE=${AVAILABLE_SPACE} KB"
if (( AVAILABLE_SPACE < REQUIRED_SPACE )); then
case "$GUI_TOOL" in
yad)
yad --image=dialog-error --button=gtk-ok:0 --center --text="Nicht genügend Speicherplatz verfügbar!" || true
;;
zenity)
zenity --error --text="Nicht genügend Speicherplatz verfügbar!" || true
;;
*)
echo "Nicht genügend Speicherplatz." >&2
;;
esac
exit 1
fi
cd "${DIRNAME}"
# -------------------- ffmpeg-Listendatei (concat demuxer) --------------------
LIST_FILE="$(mktemp)"
while IFS= read -r line; do
[[ -n "$line" ]] || continue
printf "file '%s'\n" "$line" >> "${LIST_FILE}"
done <<< "${SORTED_FILES}"
# Gemeinsame ffmpeg-Argumente vorbereiten (concat-Demuxer)
FFMPEG_INPUT_ARGS=(-hide_banner -loglevel error -f concat -safe 0 -i "${LIST_FILE}")
# -------------------- Copy-Test (kurz) --------------------
NEED_REENCODE=0
if ffmpeg "${FFMPEG_INPUT_ARGS[@]}" -c copy -t 0.1 -f null - </dev/null 2>/dev/null; then
NEED_REENCODE=0
else
NEED_REENCODE=1
fi
echo "NEED_REENCODE=$NEED_REENCODE"
# Nachfrage bei Re-Encode
if (( NEED_REENCODE == 1 )); then
case "$GUI_TOOL" in
yad)
yad --title="Neukodierung erforderlich" --text="Die Dateien müssen neukodiert werden. Fortfahren?" --button="Nein":1 --button="Ja":0 || { rm -f "${LIST_FILE}"; exit 1; }
;;
zenity)
zenity --question --title="Neukodierung erforderlich" --text="Die Dateien müssen neukodiert werden. Fortfahren?" --cancel-label="Nein" --ok-label="Ja" || { rm -f "${LIST_FILE}"; exit 1; }
;;
esac
fi
# -------------------- Fortschritts-UI via FIFO (immer sichtbar) --------------------
PROG_FIFO="$(mktemp)"
trap 'rm -f "$PROG_FIFO"' EXIT
rm -f "$PROG_FIFO"
mkfifo "$PROG_FIFO"
case "$GUI_TOOL" in
yad)
yad --progress --pulsate --no-cancel --auto-close --title="Zusammenfügen läuft" --text="# Bitte warten..." < "$PROG_FIFO" &
GUI_PID=$!
echo "GUI_PID=$GUI_PID"
;;
zenity)
zenity --progress --pulsate --no-cancel --auto-close --title="Zusammenfügen läuft" --text="# Bitte warten..." < "$PROG_FIFO" &
GUI_PID=$!
echo "GUI_PID=$GUI_PID"
;;
esac
set +e
FF_ERR=0
if (( NEED_REENCODE == 1 )); then
case "${EXTENSION}" in
mp4|mov|mkv)
print_ffmpeg_cmd -y "${FFMPEG_INPUT_ARGS[@]}" -c:v libx264 -preset fast -crf 23 -c:a aac "${OUTPUT_FILE}"
ffmpeg -y "${FFMPEG_INPUT_ARGS[@]}" -c:v libx264 -preset fast -crf 23 -c:a aac "${OUTPUT_FILE}" || FF_ERR=$?
;;
mp3|aac|ogg)
print_ffmpeg_cmd -y "${FFMPEG_INPUT_ARGS[@]}" -c:a libmp3lame "${OUTPUT_FILE}"
ffmpeg -y "${FFMPEG_INPUT_ARGS[@]}" -c:a libmp3lame "${OUTPUT_FILE}" || FF_ERR=$?
;;
wav|flac)
print_ffmpeg_cmd -y "${FFMPEG_INPUT_ARGS[@]}" -c:a flac "${OUTPUT_FILE}"
ffmpeg -y "${FFMPEG_INPUT_ARGS[@]}" -c:a flac "${OUTPUT_FILE}" || FF_ERR=$?
;;
*)
print_ffmpeg_cmd -y "${FFMPEG_INPUT_ARGS[@]}" -c copy "${OUTPUT_FILE}"
ffmpeg -y "${FFMPEG_INPUT_ARGS[@]}" -c copy "${OUTPUT_FILE}" || FF_ERR=$?
;;
esac
else
print_ffmpeg_cmd -y "${FFMPEG_INPUT_ARGS[@]}" -c copy "${OUTPUT_FILE}"
ffmpeg -y "${FFMPEG_INPUT_ARGS[@]}" -c copy "${OUTPUT_FILE}" || FF_ERR=$?
fi
set -e
# Fortschritt schließen / Meldung
{
echo 100
echo "# Fertig."
} > "$PROG_FIFO"
rm -f "$PROG_FIFO" "${LIST_FILE}"
if (( FF_ERR != 0 )); then
echo "ffmpeg exit code: $FF_ERR"
case "$GUI_TOOL" in
yad)
yad --image=dialog-error --text="Fehler beim Zusammenfügen (Code $FF_ERR).\nSiehe Log: $LOGFILE" --button="OK":0 --center || true
;;
zenity)
zenity --error --text="Fehler beim Zusammenfügen (Code $FF_ERR).\nSiehe Log: $LOGFILE" || true
;;
*)
echo "Fehler beim Zusammenfügen (Code $FF_ERR). Siehe Log: $LOGFILE" >&2
;;
esac
exit "$FF_ERR"
else
case "$GUI_TOOL" in
yad)
yad --image=dialog-information --text="Erfolg: ${OUTPUT_FILE}" --button="OK":0 --center || true
;;
zenity)
zenity --info --text="Erfolg: ${OUTPUT_FILE}" || true
;;
*)
echo "Erfolg: ${OUTPUT_FILE}"
;;
esac
fi
if [[ "${PRESERVE_SOURCE_TIMESTAMP}" == "true" ]] && [[ -f "${FIRST_FILE}" ]] && [[ -f "${OUTPUT_FILE}" ]]; then
touch -r "${FIRST_FILE}" "${OUTPUT_FILE}" || echo "Warnung: Konnte Zeitstempel nicht übernehmen" >&2
fi
echo "---- $(date) end ----"