Skip to content

Commit ebf8cfb

Browse files
committed
Add Web Share Target and Protocol Handler #9
1 parent 13e6571 commit ebf8cfb

File tree

6 files changed

+136
-4
lines changed

6 files changed

+136
-4
lines changed

public/manifest.json

+27-1
Original file line numberDiff line numberDiff line change
@@ -55,5 +55,31 @@
5555
"form_factor": "narrow",
5656
"label": "Transcribe App"
5757
}
58-
]
58+
],
59+
"protocol_handlers": [
60+
{
61+
"protocol": "web+audio",
62+
"url": "/transcribe/?url=%s"
63+
}
64+
],
65+
"share_target": {
66+
"action": "/transcribe/",
67+
"method": "POST",
68+
"enctype": "multipart/form-data",
69+
"params": {
70+
"files": [
71+
{
72+
"name": "audio_files",
73+
"accept": [
74+
"audio/*",
75+
".mp3",
76+
".wav",
77+
".ogg",
78+
".flac",
79+
".m4a"
80+
]
81+
}
82+
]
83+
}
84+
}
5985
}

src/components/AudioManager.tsx

+15-1
Original file line numberDiff line numberDiff line change
@@ -10,8 +10,10 @@ import { AnchorIcon, FolderIcon, MicrophoneIcon } from "./Icons";
1010
import { Modal } from "./Modal";
1111
import { Switch } from "./Switch";
1212
import { UrlInput } from "./UrlInput";
13-
import { Transcriber } from "../hooks/useTranscriber";
1413
import { titleCase } from "../utils/StringUtils";
14+
import { Transcriber } from "../hooks/useTranscriber";
15+
import { useProtocolHandler } from "../hooks/useProtocolHandler";
16+
import { useAudioFileReceiver } from "../hooks/useAudioFileReceiver";
1517
import { SAMPLING_RATE, DEFAULT_AUDIO_URL, LANGUAGES, MODELS } from "../config";
1618

1719
export enum AudioSource {
@@ -116,6 +118,18 @@ export const AudioManager: React.FC<AudiomanagerProps> = ({ transcriber }) => {
116118
}
117119
};
118120

121+
// Handle requests to http://localhost:5173/transcribe/?url=
122+
useProtocolHandler("web+transcribe", (url: URL) => {
123+
const incomingUrl = url.toString();
124+
setAudioDownloadUrl(incomingUrl);
125+
});
126+
127+
useAudioFileReceiver(async (file) => {
128+
const arrayBuffer = await file.arrayBuffer();
129+
const mimeType = file.type;
130+
await setAudioFromDownload(arrayBuffer, mimeType);
131+
});
132+
119133
// When URL changes, download audio
120134
useEffect(() => {
121135
if (audioDownloadUrl) {

src/components/Transcript.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -165,7 +165,7 @@ export const Transcript: React.FC<TranscriptProps> = ({ transcriber }) => {
165165
className='flex-1'
166166
>
167167
<DownloadIcon className='size-5 fill-slate-100' />{" "}
168-
TXT
168+
Text
169169
</Button>
170170
<Button
171171
aria-label='Download text as SRT file'

src/hooks/useAudioFileReceiver.ts

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
import { useEffect } from "react";
2+
3+
type AudioFileHandler = (file: File) => void;
4+
5+
export const useAudioFileReceiver = (handleFile: AudioFileHandler) => {
6+
useEffect(() => {
7+
const handleFetchEvent = async (event: FetchEvent) => {
8+
if (event.request.method === "POST") {
9+
const url = new URL(event.request.url);
10+
console.log(event.request);
11+
console.log(url);
12+
13+
event.respondWith(
14+
(async () => {
15+
try {
16+
const formData = await event.request.formData();
17+
const audioFile = formData.get(
18+
"audio_files",
19+
) as File;
20+
21+
if (
22+
audioFile &&
23+
audioFile.type.startsWith("audio/")
24+
) {
25+
handleFile(audioFile);
26+
return new Response("Audio file received", {
27+
status: 200,
28+
});
29+
} else {
30+
return new Response(
31+
"No audio file received or invalid file type",
32+
{ status: 400 },
33+
);
34+
}
35+
} catch (error) {
36+
console.error("Error handling fetch event:", error);
37+
return new Response("Error handling audio file", {
38+
status: 500,
39+
});
40+
}
41+
})(),
42+
);
43+
}
44+
};
45+
46+
// @ts-ignore
47+
self.addEventListener("fetch", handleFetchEvent);
48+
49+
return () => {
50+
// @ts-ignore
51+
self.removeEventListener("fetch", handleFetchEvent);
52+
};
53+
}, [handleFile]);
54+
};

src/hooks/useProtocolHandler.ts

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
import { useEffect, useRef } from "react";
2+
3+
type ProtocolHandler = (url: URL) => void;
4+
5+
export const useProtocolHandler = (
6+
protocol: string,
7+
handler: ProtocolHandler,
8+
) => {
9+
const lastUrlRef = useRef<string | null>(null);
10+
11+
useEffect(() => {
12+
if ("registerProtocolHandler" in navigator) {
13+
try {
14+
navigator.registerProtocolHandler(
15+
protocol,
16+
`${window.location.origin}/?url=%s`,
17+
);
18+
} catch (error) {
19+
console.warn("Failed to register protocol handler:", error);
20+
}
21+
}
22+
}, [protocol]);
23+
24+
useEffect(() => {
25+
const urlParams = new URLSearchParams(window.location.search);
26+
const incomingUrl = urlParams.get("url");
27+
28+
if (incomingUrl && incomingUrl !== lastUrlRef.current) {
29+
try {
30+
const parsedUrl = new URL(incomingUrl);
31+
handler(parsedUrl);
32+
lastUrlRef.current = incomingUrl; // Update the last URL
33+
} catch (error) {
34+
console.error("Failed to parse incoming URL:", error);
35+
}
36+
}
37+
}, [handler]);
38+
};

tsconfig.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"compilerOptions": {
33
"target": "ESNext",
44
"useDefineForClassFields": true,
5-
"lib": ["DOM", "DOM.Iterable", "ESNext"],
5+
"lib": ["ESNext", "DOM", "DOM.Iterable", "webworker"],
66
"allowJs": false,
77
"skipLibCheck": true,
88
"esModuleInterop": false,

0 commit comments

Comments
 (0)