Caller/Callee UX: - lib/ringback.ts + assets/sounds/ringback_eu.mp3 (EU 425Hz Festnetz-Tone) - stores/call.ts: stopRingback bei connected, hangup-reasons, logCallToChat fix - locales: 'Wird angerufen…' statt 'Ruft an…' CallKit (iOS) + ConnectionService (Android): - lib/callkit.ts: setupCallKeep, displayIncomingCall, startOutgoingCall, reportConnected/Ended (appName 'ReBreak-Audio', includesCallsInRecents=false für DSGVO/DiGA) - hooks/useCallKeepEvents.ts: native answer/end/mute → useCallStore-Actions - stores/call.ts: CallKit-Aufrufe an allen lifecycle-Punkten - app.config.ts: @config-plugins/react-native-callkeep + UIBackgroundModes voip/audio + Android-Telecom-Perms VoIP-PushKit Backend: - services/voip-push.ts: @parse/node-apn Provider mit .p12 (Topic org.rebreak.app.voip) - services/push.ts sendCallRingPush: feuert beide Pfade (VoIP iOS + Expo Android/Fallback) - prisma: push_tokens.voip_token Column + Migration 20260604 - api/users/me/push-token: optional voipToken im Body - Env (Infisical): APNS_VOIP_P12_PATH/PASSWORD/TOPIC/PRODUCTION Push-tap routing + cold-start handling: - app/_layout.tsx: type:'call' Push → useCallStore.receiveIncoming + /call Docs: ops/CALLKIT_SETUP.md (Apple-Portal-Steps für VoIP-Cert)
63 lines
2.2 KiB
TypeScript
63 lines
2.2 KiB
TypeScript
/**
|
|
* Bridge zwischen CallKit/ConnectionService-Events und unserer Call-Store.
|
|
*
|
|
* Wenn der User in der nativen Call-UI Accept/Reject/Hangup tippt, kommt das
|
|
* NICHT über unser React-UI rein — sondern via RNCallKeep-Events. Wir
|
|
* übersetzen die in store-Actions.
|
|
*
|
|
* Wird einmal app-weit im _layout.tsx aufgerufen.
|
|
*/
|
|
import { useEffect } from 'react';
|
|
import { useRouter } from 'expo-router';
|
|
import RNCallKeep from 'react-native-callkeep';
|
|
import { useCallStore } from '../stores/call';
|
|
import { setupCallKeep } from '../lib/callkit';
|
|
|
|
export function useCallKeepEvents() {
|
|
const router = useRouter();
|
|
|
|
useEffect(() => {
|
|
void setupCallKeep();
|
|
|
|
// User tippt "Annehmen" in der CallKit-/ConnectionService-UI
|
|
const onAnswer = ({ callUUID }: { callUUID: string }) => {
|
|
console.log('[callkeep] answer', callUUID);
|
|
const st = useCallStore.getState();
|
|
if (st.status !== 'incoming') return;
|
|
// Call-Screen öffnen und Accept-Flow triggern.
|
|
router.push('/call');
|
|
void st.acceptCall();
|
|
};
|
|
|
|
// User tippt "Ablehnen" oder "Auflegen" in der nativen UI
|
|
const onEnd = ({ callUUID }: { callUUID: string }) => {
|
|
console.log('[callkeep] end', callUUID);
|
|
const st = useCallStore.getState();
|
|
if (st.status === 'idle' || st.status === 'ended') return;
|
|
if (st.status === 'incoming') {
|
|
st.declineCall();
|
|
} else {
|
|
st.hangup('ended');
|
|
}
|
|
};
|
|
|
|
// User mutet/unmutet über die native UI
|
|
const onMuted = ({ muted }: { muted: boolean; callUUID: string }) => {
|
|
const st = useCallStore.getState();
|
|
if (st.muted !== muted) st.toggleMute();
|
|
};
|
|
|
|
RNCallKeep.addEventListener('answerCall', onAnswer);
|
|
RNCallKeep.addEventListener('endCall', onEnd);
|
|
RNCallKeep.addEventListener('didPerformSetMutedCallAction', onMuted);
|
|
// didActivateAudioSession kommt nach CallKit-Audio-Activation — wir nutzen
|
|
// das (noch) nicht aktiv, weil WebRTC + InCallManager das selber regeln.
|
|
|
|
return () => {
|
|
RNCallKeep.removeEventListener('answerCall');
|
|
RNCallKeep.removeEventListener('endCall');
|
|
RNCallKeep.removeEventListener('didPerformSetMutedCallAction');
|
|
};
|
|
}, [router]);
|
|
}
|