mirror of
https://github.com/tapframe/NuvioStreaming.git
synced 2026-04-21 00:32:04 +00:00
fixed tmdb ratings
This commit is contained in:
parent
12c591216e
commit
d914e01346
5 changed files with 146 additions and 670 deletions
|
|
@ -12,7 +12,7 @@ import Animated, {
|
||||||
FadeIn,
|
FadeIn,
|
||||||
} from 'react-native-reanimated';
|
} from 'react-native-reanimated';
|
||||||
import { useTheme } from '../../contexts/ThemeContext';
|
import { useTheme } from '../../contexts/ThemeContext';
|
||||||
import MetadataSourceSelector from './MetadataSourceSelector';
|
// MetadataSourceSelector removed
|
||||||
|
|
||||||
interface MetadataDetailsProps {
|
interface MetadataDetailsProps {
|
||||||
metadata: any;
|
metadata: any;
|
||||||
|
|
@ -20,8 +20,7 @@ interface MetadataDetailsProps {
|
||||||
type: 'movie' | 'series';
|
type: 'movie' | 'series';
|
||||||
renderRatings?: () => React.ReactNode;
|
renderRatings?: () => React.ReactNode;
|
||||||
contentId: string;
|
contentId: string;
|
||||||
currentMetadataSource?: string;
|
// Source switching removed
|
||||||
onMetadataSourceChange?: (sourceId: string, sourceType: 'addon' | 'tmdb') => void;
|
|
||||||
loadingMetadata?: boolean;
|
loadingMetadata?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -31,8 +30,6 @@ const MetadataDetails: React.FC<MetadataDetailsProps> = ({
|
||||||
type,
|
type,
|
||||||
renderRatings,
|
renderRatings,
|
||||||
contentId,
|
contentId,
|
||||||
currentMetadataSource,
|
|
||||||
onMetadataSourceChange,
|
|
||||||
loadingMetadata = false,
|
loadingMetadata = false,
|
||||||
}) => {
|
}) => {
|
||||||
const { currentTheme } = useTheme();
|
const { currentTheme } = useTheme();
|
||||||
|
|
@ -40,23 +37,7 @@ const MetadataDetails: React.FC<MetadataDetailsProps> = ({
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{/* Metadata Source Selector */}
|
{/* Metadata Source Selector removed */}
|
||||||
{onMetadataSourceChange && (
|
|
||||||
<View style={styles.sourceSelectorContainer}>
|
|
||||||
<MetadataSourceSelector
|
|
||||||
currentSource={currentMetadataSource}
|
|
||||||
contentId={contentId}
|
|
||||||
contentType={type}
|
|
||||||
onSourceChange={onMetadataSourceChange}
|
|
||||||
disabled={loadingMetadata}
|
|
||||||
/>
|
|
||||||
{currentMetadataSource && currentMetadataSource !== 'auto' && (
|
|
||||||
<Text style={[styles.sourceIndicator, { color: currentTheme.colors.textMuted }]}>
|
|
||||||
Currently showing metadata from: {currentMetadataSource === 'tmdb' ? 'TMDB' : currentMetadataSource}
|
|
||||||
</Text>
|
|
||||||
)}
|
|
||||||
</View>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Loading indicator when switching sources */}
|
{/* Loading indicator when switching sources */}
|
||||||
{loadingMetadata && (
|
{loadingMetadata && (
|
||||||
|
|
@ -144,15 +125,7 @@ const MetadataDetails: React.FC<MetadataDetailsProps> = ({
|
||||||
};
|
};
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
sourceSelectorContainer: {
|
// Removed source selector styles
|
||||||
paddingHorizontal: 16,
|
|
||||||
marginBottom: 16,
|
|
||||||
},
|
|
||||||
sourceIndicator: {
|
|
||||||
fontSize: 12,
|
|
||||||
marginTop: 4,
|
|
||||||
fontStyle: 'italic',
|
|
||||||
},
|
|
||||||
loadingContainer: {
|
loadingContainer: {
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
|
|
|
||||||
|
|
@ -1,523 +0,0 @@
|
||||||
import React, { useState, useEffect, useCallback, useMemo, useRef } from 'react';
|
|
||||||
import {
|
|
||||||
View,
|
|
||||||
Text,
|
|
||||||
StyleSheet,
|
|
||||||
TouchableOpacity,
|
|
||||||
Modal,
|
|
||||||
ScrollView,
|
|
||||||
ActivityIndicator,
|
|
||||||
Animated,
|
|
||||||
} from 'react-native';
|
|
||||||
import { MaterialIcons } from '@expo/vector-icons';
|
|
||||||
import * as Haptics from 'expo-haptics';
|
|
||||||
import { useTheme } from '../../contexts/ThemeContext';
|
|
||||||
import { stremioService } from '../../services/stremioService';
|
|
||||||
import { tmdbService } from '../../services/tmdbService';
|
|
||||||
import { catalogService } from '../../services/catalogService';
|
|
||||||
import { logger } from '../../utils/logger';
|
|
||||||
|
|
||||||
interface MetadataSource {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
type: 'addon' | 'tmdb';
|
|
||||||
hasMetaSupport?: boolean;
|
|
||||||
icon?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface MetadataSourceSelectorProps {
|
|
||||||
currentSource?: string;
|
|
||||||
contentId: string;
|
|
||||||
contentType: string;
|
|
||||||
onSourceChange: (sourceId: string, sourceType: 'addon' | 'tmdb') => void;
|
|
||||||
disabled?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const MetadataSourceSelector: React.FC<MetadataSourceSelectorProps> = ({
|
|
||||||
currentSource,
|
|
||||||
contentId,
|
|
||||||
contentType,
|
|
||||||
onSourceChange,
|
|
||||||
disabled = false,
|
|
||||||
}) => {
|
|
||||||
const { currentTheme } = useTheme();
|
|
||||||
const [isVisible, setIsVisible] = useState(false);
|
|
||||||
const [sources, setSources] = useState<MetadataSource[]>([]);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [selectedSource, setSelectedSource] = useState<string>(currentSource || 'auto');
|
|
||||||
const scaleAnim = useRef(new Animated.Value(0.8)).current;
|
|
||||||
const opacityAnim = useRef(new Animated.Value(0)).current;
|
|
||||||
|
|
||||||
// Load available metadata sources
|
|
||||||
const loadMetadataSources = useCallback(async () => {
|
|
||||||
setLoading(true);
|
|
||||||
try {
|
|
||||||
const sources: MetadataSource[] = [];
|
|
||||||
|
|
||||||
// Add auto-select option
|
|
||||||
sources.push({
|
|
||||||
id: 'auto',
|
|
||||||
name: 'Auto (Best Available)',
|
|
||||||
type: 'addon',
|
|
||||||
icon: 'auto-fix-high',
|
|
||||||
});
|
|
||||||
|
|
||||||
// Add TMDB as a source
|
|
||||||
sources.push({
|
|
||||||
id: 'tmdb',
|
|
||||||
name: 'The Movie Database (TMDB)',
|
|
||||||
type: 'tmdb',
|
|
||||||
icon: 'movie',
|
|
||||||
});
|
|
||||||
|
|
||||||
// Get installed addons with meta support
|
|
||||||
const addons = await stremioService.getInstalledAddonsAsync();
|
|
||||||
logger.log(`[MetadataSourceSelector] Checking ${addons.length} installed addons for meta support`);
|
|
||||||
|
|
||||||
for (const addon of addons) {
|
|
||||||
logger.log(`[MetadataSourceSelector] Checking addon: ${addon.name} (${addon.id})`);
|
|
||||||
logger.log(`[MetadataSourceSelector] Addon resources:`, addon.resources);
|
|
||||||
|
|
||||||
// Check if addon supports meta resource
|
|
||||||
let hasMetaSupport = false;
|
|
||||||
|
|
||||||
if (addon.resources) {
|
|
||||||
// Handle both array of strings and array of objects
|
|
||||||
hasMetaSupport = addon.resources.some((resource: any) => {
|
|
||||||
if (typeof resource === 'string') {
|
|
||||||
// Simple string format like ["catalog", "meta"]
|
|
||||||
return resource === 'meta';
|
|
||||||
} else if (resource && typeof resource === 'object') {
|
|
||||||
// Object format like { name: 'meta', types: ['movie', 'series'] }
|
|
||||||
const supportsType = !resource.types || resource.types.includes(contentType);
|
|
||||||
logger.log(`[MetadataSourceSelector] Resource ${resource.name}: types=${resource.types}, supportsType=${supportsType}`);
|
|
||||||
return resource.name === 'meta' && supportsType;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.log(`[MetadataSourceSelector] Addon ${addon.name} has meta support: ${hasMetaSupport}`);
|
|
||||||
|
|
||||||
if (hasMetaSupport) {
|
|
||||||
sources.push({
|
|
||||||
id: addon.id,
|
|
||||||
name: addon.name,
|
|
||||||
type: 'addon',
|
|
||||||
hasMetaSupport: true,
|
|
||||||
icon: 'extension',
|
|
||||||
});
|
|
||||||
logger.log(`[MetadataSourceSelector] Added addon ${addon.name} as metadata source`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sort sources: auto first, then TMDB, then addons alphabetically
|
|
||||||
sources.sort((a, b) => {
|
|
||||||
if (a.id === 'auto') return -1;
|
|
||||||
if (b.id === 'auto') return 1;
|
|
||||||
if (a.id === 'tmdb') return -1;
|
|
||||||
if (b.id === 'tmdb') return 1;
|
|
||||||
return a.name.localeCompare(b.name);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Always include at least auto and TMDB for comparison
|
|
||||||
if (sources.length < 2) {
|
|
||||||
logger.warn('[MetadataSourceSelector] Less than 2 sources found, this should not happen');
|
|
||||||
}
|
|
||||||
|
|
||||||
setSources(sources);
|
|
||||||
logger.log(`[MetadataSourceSelector] Found ${sources.length} metadata sources:`, sources.map(s => s.name));
|
|
||||||
} catch (error) {
|
|
||||||
logger.error('[MetadataSourceSelector] Failed to load metadata sources:', error);
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
}, [contentType]);
|
|
||||||
|
|
||||||
// Load sources on mount
|
|
||||||
useEffect(() => {
|
|
||||||
loadMetadataSources();
|
|
||||||
}, [loadMetadataSources]);
|
|
||||||
|
|
||||||
// Update selected source when currentSource prop changes
|
|
||||||
useEffect(() => {
|
|
||||||
if (currentSource) {
|
|
||||||
setSelectedSource(currentSource);
|
|
||||||
}
|
|
||||||
}, [currentSource]);
|
|
||||||
|
|
||||||
const handleSourceSelect = useCallback((source: MetadataSource) => {
|
|
||||||
Haptics.impactAsync(Haptics.ImpactFeedbackStyle.Light);
|
|
||||||
setSelectedSource(source.id);
|
|
||||||
setIsVisible(false);
|
|
||||||
onSourceChange(source.id, source.type);
|
|
||||||
}, [onSourceChange]);
|
|
||||||
|
|
||||||
const handleOpenModal = useCallback(() => {
|
|
||||||
Haptics.impactAsync(Haptics.ImpactFeedbackStyle.Light);
|
|
||||||
// Reset animation values
|
|
||||||
scaleAnim.setValue(0.8);
|
|
||||||
opacityAnim.setValue(0);
|
|
||||||
setIsVisible(true);
|
|
||||||
Animated.parallel([
|
|
||||||
Animated.spring(scaleAnim, {
|
|
||||||
toValue: 1,
|
|
||||||
useNativeDriver: true,
|
|
||||||
tension: 100,
|
|
||||||
friction: 8,
|
|
||||||
}),
|
|
||||||
Animated.timing(opacityAnim, {
|
|
||||||
toValue: 1,
|
|
||||||
duration: 200,
|
|
||||||
useNativeDriver: true,
|
|
||||||
}),
|
|
||||||
]).start();
|
|
||||||
}, [scaleAnim, opacityAnim]);
|
|
||||||
|
|
||||||
const handleCloseModal = useCallback(() => {
|
|
||||||
Animated.parallel([
|
|
||||||
Animated.spring(scaleAnim, {
|
|
||||||
toValue: 0.8,
|
|
||||||
useNativeDriver: true,
|
|
||||||
tension: 100,
|
|
||||||
friction: 8,
|
|
||||||
}),
|
|
||||||
Animated.timing(opacityAnim, {
|
|
||||||
toValue: 0,
|
|
||||||
duration: 150,
|
|
||||||
useNativeDriver: true,
|
|
||||||
}),
|
|
||||||
]).start(() => {
|
|
||||||
setIsVisible(false);
|
|
||||||
});
|
|
||||||
}, [scaleAnim, opacityAnim]);
|
|
||||||
|
|
||||||
const currentSourceName = useMemo(() => {
|
|
||||||
const source = sources.find(s => s.id === selectedSource);
|
|
||||||
return source?.name || 'Auto (Best Available)';
|
|
||||||
}, [sources, selectedSource]);
|
|
||||||
|
|
||||||
const getSourceIcon = useCallback((source: MetadataSource) => {
|
|
||||||
switch (source.icon) {
|
|
||||||
case 'auto-fix-high':
|
|
||||||
return 'auto-fix-high';
|
|
||||||
case 'movie':
|
|
||||||
return 'movie';
|
|
||||||
case 'extension':
|
|
||||||
return 'extension';
|
|
||||||
default:
|
|
||||||
return 'info';
|
|
||||||
}
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Always show if we have sources and onSourceChange callback
|
|
||||||
if (sources.length === 0 || !onSourceChange) {
|
|
||||||
console.log('[MetadataSourceSelector] Not showing selector:', { sourcesLength: sources.length, hasCallback: !!onSourceChange });
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('[MetadataSourceSelector] Showing selector with sources:', sources.map(s => s.name));
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<View style={styles.container}>
|
|
||||||
<Text style={[styles.label, { color: currentTheme.colors.textMuted }]}>
|
|
||||||
Metadata Source
|
|
||||||
</Text>
|
|
||||||
<TouchableOpacity
|
|
||||||
style={[
|
|
||||||
styles.selector,
|
|
||||||
{
|
|
||||||
backgroundColor: currentTheme.colors.elevation1,
|
|
||||||
borderColor: currentTheme.colors.border,
|
|
||||||
},
|
|
||||||
disabled && styles.disabled
|
|
||||||
]}
|
|
||||||
onPress={() => !disabled && handleOpenModal()}
|
|
||||||
disabled={disabled}
|
|
||||||
activeOpacity={0.7}
|
|
||||||
>
|
|
||||||
<View style={styles.selectorContent}>
|
|
||||||
<MaterialIcons
|
|
||||||
name={getSourceIcon(sources.find(s => s.id === selectedSource) || sources[0])}
|
|
||||||
size={20}
|
|
||||||
color={currentTheme.colors.text}
|
|
||||||
/>
|
|
||||||
<Text
|
|
||||||
style={[styles.selectorText, { color: currentTheme.colors.text }]}
|
|
||||||
numberOfLines={1}
|
|
||||||
>
|
|
||||||
{currentSourceName}
|
|
||||||
</Text>
|
|
||||||
</View>
|
|
||||||
<MaterialIcons
|
|
||||||
name="expand-more"
|
|
||||||
size={20}
|
|
||||||
color={currentTheme.colors.textMuted}
|
|
||||||
/>
|
|
||||||
</TouchableOpacity>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
<Modal
|
|
||||||
visible={isVisible}
|
|
||||||
transparent
|
|
||||||
animationType="fade"
|
|
||||||
onRequestClose={handleCloseModal}
|
|
||||||
statusBarTranslucent
|
|
||||||
>
|
|
||||||
<View style={styles.modalOverlay}>
|
|
||||||
<Animated.View style={[
|
|
||||||
styles.modalContent,
|
|
||||||
{
|
|
||||||
backgroundColor: 'rgba(30, 30, 30, 0.98)',
|
|
||||||
borderWidth: 1,
|
|
||||||
borderColor: 'rgba(255, 255, 255, 0.12)',
|
|
||||||
transform: [{ scale: scaleAnim }],
|
|
||||||
opacity: opacityAnim,
|
|
||||||
}
|
|
||||||
]}>
|
|
||||||
<View style={styles.modalHeader}>
|
|
||||||
<Text style={[styles.modalTitle, { color: currentTheme.colors.text }]}>
|
|
||||||
Select Metadata Source
|
|
||||||
</Text>
|
|
||||||
<TouchableOpacity
|
|
||||||
onPress={handleCloseModal}
|
|
||||||
style={styles.closeButton}
|
|
||||||
>
|
|
||||||
<MaterialIcons name="close" size={24} color={currentTheme.colors.textMuted} />
|
|
||||||
</TouchableOpacity>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
{loading ? (
|
|
||||||
<View style={styles.loadingContainer}>
|
|
||||||
<ActivityIndicator size="small" color={currentTheme.colors.primary} />
|
|
||||||
<Text style={[styles.loadingText, { color: currentTheme.colors.textMuted }]}>
|
|
||||||
Loading sources...
|
|
||||||
</Text>
|
|
||||||
</View>
|
|
||||||
) : (
|
|
||||||
<ScrollView style={styles.sourcesList}>
|
|
||||||
{sources.map((source) => (
|
|
||||||
<TouchableOpacity
|
|
||||||
key={source.id}
|
|
||||||
style={[
|
|
||||||
styles.sourceItem,
|
|
||||||
{
|
|
||||||
backgroundColor: selectedSource === source.id
|
|
||||||
? currentTheme.colors.primary + '25'
|
|
||||||
: 'rgba(45, 45, 45, 0.95)',
|
|
||||||
borderColor: selectedSource === source.id
|
|
||||||
? currentTheme.colors.primary + '50'
|
|
||||||
: 'rgba(255, 255, 255, 0.15)',
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
onPress={() => handleSourceSelect(source)}
|
|
||||||
activeOpacity={0.7}
|
|
||||||
>
|
|
||||||
<View style={styles.sourceContent}>
|
|
||||||
<View style={[
|
|
||||||
styles.iconContainer,
|
|
||||||
{
|
|
||||||
backgroundColor: selectedSource === source.id
|
|
||||||
? currentTheme.colors.primary + '30'
|
|
||||||
: 'rgba(60, 60, 60, 0.95)'
|
|
||||||
}
|
|
||||||
]}>
|
|
||||||
<MaterialIcons
|
|
||||||
name={getSourceIcon(source)}
|
|
||||||
size={20}
|
|
||||||
color={selectedSource === source.id ? currentTheme.colors.primary : currentTheme.colors.text}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
<View style={styles.sourceInfo}>
|
|
||||||
<Text
|
|
||||||
style={[
|
|
||||||
styles.sourceName,
|
|
||||||
{
|
|
||||||
color: selectedSource === source.id ? currentTheme.colors.primary : currentTheme.colors.text
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
>
|
|
||||||
{source.name}
|
|
||||||
</Text>
|
|
||||||
{source.type === 'addon' && source.hasMetaSupport && (
|
|
||||||
<Text style={[styles.sourceDescription, { color: currentTheme.colors.textMuted }]}>
|
|
||||||
Stremio Addon with metadata support
|
|
||||||
</Text>
|
|
||||||
)}
|
|
||||||
{source.type === 'tmdb' && (
|
|
||||||
<Text style={[styles.sourceDescription, { color: currentTheme.colors.textMuted }]}>
|
|
||||||
Comprehensive movie and TV database
|
|
||||||
</Text>
|
|
||||||
)}
|
|
||||||
{source.id === 'auto' && (
|
|
||||||
<Text style={[styles.sourceDescription, { color: currentTheme.colors.textMuted }]}>
|
|
||||||
Automatically selects the best available source
|
|
||||||
</Text>
|
|
||||||
)}
|
|
||||||
</View>
|
|
||||||
</View>
|
|
||||||
{selectedSource === source.id && (
|
|
||||||
<View style={styles.checkContainer}>
|
|
||||||
<MaterialIcons
|
|
||||||
name="check-circle"
|
|
||||||
size={24}
|
|
||||||
color={currentTheme.colors.primary}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
)}
|
|
||||||
</TouchableOpacity>
|
|
||||||
))}
|
|
||||||
</ScrollView>
|
|
||||||
)}
|
|
||||||
</Animated.View>
|
|
||||||
</View>
|
|
||||||
</Modal>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
|
||||||
container: {
|
|
||||||
marginBottom: 16,
|
|
||||||
},
|
|
||||||
label: {
|
|
||||||
fontSize: 14,
|
|
||||||
fontWeight: '600',
|
|
||||||
marginBottom: 8,
|
|
||||||
},
|
|
||||||
selector: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'space-between',
|
|
||||||
paddingHorizontal: 16,
|
|
||||||
paddingVertical: 12,
|
|
||||||
borderRadius: 12,
|
|
||||||
borderWidth: 1,
|
|
||||||
shadowColor: '#000',
|
|
||||||
shadowOffset: { width: 0, height: 2 },
|
|
||||||
shadowOpacity: 0.1,
|
|
||||||
shadowRadius: 4,
|
|
||||||
elevation: 2,
|
|
||||||
},
|
|
||||||
disabled: {
|
|
||||||
opacity: 0.5,
|
|
||||||
},
|
|
||||||
selectorContent: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
flex: 1,
|
|
||||||
},
|
|
||||||
selectorText: {
|
|
||||||
fontSize: 14,
|
|
||||||
fontWeight: '500',
|
|
||||||
marginLeft: 8,
|
|
||||||
flex: 1,
|
|
||||||
},
|
|
||||||
modalOverlay: {
|
|
||||||
flex: 1,
|
|
||||||
backgroundColor: 'rgba(0, 0, 0, 0.8)',
|
|
||||||
justifyContent: 'center',
|
|
||||||
alignItems: 'center',
|
|
||||||
padding: 20,
|
|
||||||
},
|
|
||||||
modalContent: {
|
|
||||||
width: '90%',
|
|
||||||
maxWidth: 380,
|
|
||||||
maxHeight: '75%',
|
|
||||||
borderRadius: 16,
|
|
||||||
overflow: 'hidden',
|
|
||||||
shadowColor: '#000',
|
|
||||||
shadowOffset: { width: 0, height: 10 },
|
|
||||||
shadowOpacity: 0.4,
|
|
||||||
shadowRadius: 20,
|
|
||||||
elevation: 10,
|
|
||||||
},
|
|
||||||
modalHeader: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'space-between',
|
|
||||||
padding: 20,
|
|
||||||
borderBottomWidth: 1,
|
|
||||||
borderBottomColor: 'rgba(255, 255, 255, 0.15)',
|
|
||||||
},
|
|
||||||
modalTitle: {
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: '700',
|
|
||||||
letterSpacing: 0.3,
|
|
||||||
},
|
|
||||||
closeButton: {
|
|
||||||
padding: 8,
|
|
||||||
borderRadius: 20,
|
|
||||||
backgroundColor: 'rgba(255, 255, 255, 0.15)',
|
|
||||||
},
|
|
||||||
loadingContainer: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'center',
|
|
||||||
padding: 40,
|
|
||||||
},
|
|
||||||
loadingText: {
|
|
||||||
marginLeft: 12,
|
|
||||||
fontSize: 14,
|
|
||||||
fontWeight: '500',
|
|
||||||
},
|
|
||||||
sourcesList: {
|
|
||||||
maxHeight: 350,
|
|
||||||
},
|
|
||||||
sourceItem: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'space-between',
|
|
||||||
padding: 16,
|
|
||||||
marginHorizontal: 12,
|
|
||||||
marginVertical: 4,
|
|
||||||
borderRadius: 12,
|
|
||||||
borderWidth: 1,
|
|
||||||
borderColor: 'transparent',
|
|
||||||
shadowColor: '#000',
|
|
||||||
shadowOffset: { width: 0, height: 1 },
|
|
||||||
shadowOpacity: 0.1,
|
|
||||||
shadowRadius: 2,
|
|
||||||
elevation: 1,
|
|
||||||
},
|
|
||||||
sourceContent: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
flex: 1,
|
|
||||||
},
|
|
||||||
sourceInfo: {
|
|
||||||
marginLeft: 16,
|
|
||||||
flex: 1,
|
|
||||||
},
|
|
||||||
sourceName: {
|
|
||||||
fontSize: 16,
|
|
||||||
fontWeight: '600',
|
|
||||||
marginBottom: 4,
|
|
||||||
letterSpacing: 0.2,
|
|
||||||
},
|
|
||||||
sourceDescription: {
|
|
||||||
fontSize: 13,
|
|
||||||
lineHeight: 18,
|
|
||||||
opacity: 0.8,
|
|
||||||
},
|
|
||||||
iconContainer: {
|
|
||||||
width: 44,
|
|
||||||
height: 44,
|
|
||||||
borderRadius: 22,
|
|
||||||
justifyContent: 'center',
|
|
||||||
alignItems: 'center',
|
|
||||||
shadowColor: '#000',
|
|
||||||
shadowOffset: { width: 0, height: 1 },
|
|
||||||
shadowOpacity: 0.1,
|
|
||||||
shadowRadius: 2,
|
|
||||||
elevation: 1,
|
|
||||||
},
|
|
||||||
checkContainer: {
|
|
||||||
padding: 4,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export default MetadataSourceSelector;
|
|
||||||
|
|
@ -46,6 +46,8 @@ export const SeriesContent: React.FC<SeriesContentProps> = ({
|
||||||
const [episodeProgress, setEpisodeProgress] = useState<{ [key: string]: { currentTime: number; duration: number; lastUpdated: number } }>({});
|
const [episodeProgress, setEpisodeProgress] = useState<{ [key: string]: { currentTime: number; duration: number; lastUpdated: number } }>({});
|
||||||
// Delay item entering animations to avoid FlashList initial layout glitches
|
// Delay item entering animations to avoid FlashList initial layout glitches
|
||||||
const [enableItemAnimations, setEnableItemAnimations] = useState(false);
|
const [enableItemAnimations, setEnableItemAnimations] = useState(false);
|
||||||
|
// Local TMDB hydration for rating/runtime when addon (Cinemeta) lacks these
|
||||||
|
const [tmdbEpisodeOverrides, setTmdbEpisodeOverrides] = useState<{ [epKey: string]: { vote_average?: number; runtime?: number; still_path?: string } }>({});
|
||||||
|
|
||||||
// Add refs for the scroll views
|
// Add refs for the scroll views
|
||||||
const seasonScrollViewRef = useRef<ScrollView | null>(null);
|
const seasonScrollViewRef = useRef<ScrollView | null>(null);
|
||||||
|
|
@ -162,6 +164,50 @@ export const SeriesContent: React.FC<SeriesContentProps> = ({
|
||||||
loadEpisodesProgress();
|
loadEpisodesProgress();
|
||||||
}, [episodes, metadata?.id]);
|
}, [episodes, metadata?.id]);
|
||||||
|
|
||||||
|
// Hydrate TMDB rating/runtime for current season episodes if missing
|
||||||
|
useEffect(() => {
|
||||||
|
const hydrateFromTmdb = async () => {
|
||||||
|
try {
|
||||||
|
if (!metadata?.id || !selectedSeason) return;
|
||||||
|
const currentSeasonEpisodes = groupedEpisodes[selectedSeason] || [];
|
||||||
|
if (currentSeasonEpisodes.length === 0) return;
|
||||||
|
|
||||||
|
// Check if hydration is needed
|
||||||
|
const needsHydration = currentSeasonEpisodes.some(ep => !(ep as any).runtime || !(ep as any).vote_average);
|
||||||
|
if (!needsHydration) return;
|
||||||
|
|
||||||
|
// Resolve TMDB show id
|
||||||
|
let tmdbShowId: number | null = null;
|
||||||
|
if (metadata.id.startsWith('tmdb:')) {
|
||||||
|
tmdbShowId = parseInt(metadata.id.split(':')[1], 10);
|
||||||
|
} else if (metadata.id.startsWith('tt')) {
|
||||||
|
tmdbShowId = await tmdbService.findTMDBIdByIMDB(metadata.id);
|
||||||
|
}
|
||||||
|
if (!tmdbShowId) return;
|
||||||
|
|
||||||
|
// Fetch all episodes from TMDB and build override map for the current season
|
||||||
|
const all = await tmdbService.getAllEpisodes(tmdbShowId);
|
||||||
|
const overrides: { [k: string]: { vote_average?: number; runtime?: number; still_path?: string } } = {};
|
||||||
|
const seasonEpisodes = all?.[String(selectedSeason)] || [];
|
||||||
|
seasonEpisodes.forEach((tmdbEp: any) => {
|
||||||
|
const key = `${metadata.id}:${tmdbEp.season_number}:${tmdbEp.episode_number}`;
|
||||||
|
overrides[key] = {
|
||||||
|
vote_average: tmdbEp.vote_average,
|
||||||
|
runtime: tmdbEp.runtime,
|
||||||
|
still_path: tmdbEp.still_path,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
if (Object.keys(overrides).length > 0) {
|
||||||
|
setTmdbEpisodeOverrides(prev => ({ ...prev, ...overrides }));
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('[SeriesContent] TMDB hydration failed:', err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
hydrateFromTmdb();
|
||||||
|
}, [metadata?.id, selectedSeason, groupedEpisodes]);
|
||||||
|
|
||||||
// Enable item animations shortly after mount to avoid initial overlap/glitch
|
// Enable item animations shortly after mount to avoid initial overlap/glitch
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const timer = setTimeout(() => setEnableItemAnimations(true), 200);
|
const timer = setTimeout(() => setEnableItemAnimations(true), 200);
|
||||||
|
|
@ -351,6 +397,13 @@ export const SeriesContent: React.FC<SeriesContentProps> = ({
|
||||||
|
|
||||||
// Get episode progress
|
// Get episode progress
|
||||||
const episodeId = episode.stremioId || `${metadata?.id}:${episode.season_number}:${episode.episode_number}`;
|
const episodeId = episode.stremioId || `${metadata?.id}:${episode.season_number}:${episode.episode_number}`;
|
||||||
|
const tmdbOverride = tmdbEpisodeOverrides[`${metadata?.id}:${episode.season_number}:${episode.episode_number}`];
|
||||||
|
const effectiveVote = (tmdbOverride?.vote_average ?? episode.vote_average) || 0;
|
||||||
|
const effectiveRuntime = tmdbOverride?.runtime ?? (episode as any).runtime;
|
||||||
|
if (!episode.still_path && tmdbOverride?.still_path) {
|
||||||
|
const tmdbUrl = tmdbService.getImageUrl(tmdbOverride.still_path, 'w500');
|
||||||
|
if (tmdbUrl) episodeImage = tmdbUrl;
|
||||||
|
}
|
||||||
const progress = episodeProgress[episodeId];
|
const progress = episodeProgress[episodeId];
|
||||||
const progressPercent = progress ? (progress.currentTime / progress.duration) * 100 : 0;
|
const progressPercent = progress ? (progress.currentTime / progress.duration) * 100 : 0;
|
||||||
|
|
||||||
|
|
@ -415,7 +468,7 @@ export const SeriesContent: React.FC<SeriesContentProps> = ({
|
||||||
styles.episodeMetadata,
|
styles.episodeMetadata,
|
||||||
isTablet && styles.episodeMetadataTablet
|
isTablet && styles.episodeMetadataTablet
|
||||||
]}>
|
]}>
|
||||||
{episode.vote_average > 0 && (
|
{effectiveVote > 0 && (
|
||||||
<View style={styles.ratingContainer}>
|
<View style={styles.ratingContainer}>
|
||||||
<Image
|
<Image
|
||||||
source={{ uri: TMDB_LOGO }}
|
source={{ uri: TMDB_LOGO }}
|
||||||
|
|
@ -423,15 +476,15 @@ export const SeriesContent: React.FC<SeriesContentProps> = ({
|
||||||
contentFit="contain"
|
contentFit="contain"
|
||||||
/>
|
/>
|
||||||
<Text style={[styles.ratingText, { color: currentTheme.colors.textMuted }]}>
|
<Text style={[styles.ratingText, { color: currentTheme.colors.textMuted }]}>
|
||||||
{episode.vote_average.toFixed(1)}
|
{effectiveVote.toFixed(1)}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
{episode.runtime && (
|
{effectiveRuntime && (
|
||||||
<View style={styles.runtimeContainer}>
|
<View style={styles.runtimeContainer}>
|
||||||
<MaterialIcons name="schedule" size={14} color={currentTheme.colors.textMuted} />
|
<MaterialIcons name="schedule" size={14} color={currentTheme.colors.textMuted} />
|
||||||
<Text style={[styles.runtimeText, { color: currentTheme.colors.textMuted }]}>
|
<Text style={[styles.runtimeText, { color: currentTheme.colors.textMuted }]}>
|
||||||
{formatRuntime(episode.runtime)}
|
{formatRuntime(effectiveRuntime)}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
|
|
@ -853,10 +906,7 @@ const styles = StyleSheet.create({
|
||||||
ratingContainer: {
|
ratingContainer: {
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
backgroundColor: 'rgba(0,0,0,0.7)',
|
// chip background removed
|
||||||
paddingHorizontal: 4,
|
|
||||||
paddingVertical: 2,
|
|
||||||
borderRadius: 4,
|
|
||||||
},
|
},
|
||||||
tmdbLogo: {
|
tmdbLogo: {
|
||||||
width: 20,
|
width: 20,
|
||||||
|
|
@ -871,10 +921,7 @@ const styles = StyleSheet.create({
|
||||||
runtimeContainer: {
|
runtimeContainer: {
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
backgroundColor: 'rgba(0,0,0,0.7)',
|
// chip background removed
|
||||||
paddingHorizontal: 4,
|
|
||||||
paddingVertical: 2,
|
|
||||||
borderRadius: 4,
|
|
||||||
},
|
},
|
||||||
runtimeText: {
|
runtimeText: {
|
||||||
fontSize: 13,
|
fontSize: 13,
|
||||||
|
|
@ -1049,10 +1096,7 @@ const styles = StyleSheet.create({
|
||||||
runtimeContainerHorizontal: {
|
runtimeContainerHorizontal: {
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
backgroundColor: 'rgba(0,0,0,0.4)',
|
// chip background removed
|
||||||
paddingHorizontal: 5,
|
|
||||||
paddingVertical: 2,
|
|
||||||
borderRadius: 3,
|
|
||||||
},
|
},
|
||||||
runtimeTextHorizontal: {
|
runtimeTextHorizontal: {
|
||||||
color: 'rgba(255,255,255,0.8)',
|
color: 'rgba(255,255,255,0.8)',
|
||||||
|
|
@ -1062,10 +1106,7 @@ const styles = StyleSheet.create({
|
||||||
ratingContainerHorizontal: {
|
ratingContainerHorizontal: {
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
backgroundColor: 'rgba(0,0,0,0.4)',
|
// chip background removed
|
||||||
paddingHorizontal: 5,
|
|
||||||
paddingVertical: 2,
|
|
||||||
borderRadius: 3,
|
|
||||||
gap: 2,
|
gap: 2,
|
||||||
},
|
},
|
||||||
ratingTextHorizontal: {
|
ratingTextHorizontal: {
|
||||||
|
|
|
||||||
|
|
@ -78,8 +78,7 @@ const MetadataScreen: React.FC = () => {
|
||||||
const [selectedCastMember, setSelectedCastMember] = useState<any>(null);
|
const [selectedCastMember, setSelectedCastMember] = useState<any>(null);
|
||||||
const [shouldLoadSecondaryData, setShouldLoadSecondaryData] = useState(false);
|
const [shouldLoadSecondaryData, setShouldLoadSecondaryData] = useState(false);
|
||||||
const [isScreenFocused, setIsScreenFocused] = useState(true);
|
const [isScreenFocused, setIsScreenFocused] = useState(true);
|
||||||
const [currentMetadataSource, setCurrentMetadataSource] = useState<string>(addonId || 'auto');
|
// Source switching removed
|
||||||
const [loadingMetadataSource, setLoadingMetadataSource] = useState(false);
|
|
||||||
const transitionOpacity = useSharedValue(1);
|
const transitionOpacity = useSharedValue(1);
|
||||||
const interactionComplete = useRef(false);
|
const interactionComplete = useRef(false);
|
||||||
|
|
||||||
|
|
@ -467,70 +466,7 @@ const MetadataScreen: React.FC = () => {
|
||||||
setShowCastModal(true);
|
setShowCastModal(true);
|
||||||
}, [isScreenFocused]);
|
}, [isScreenFocused]);
|
||||||
|
|
||||||
const handleMetadataSourceChange = useCallback(async (sourceId: string, sourceType: 'addon' | 'tmdb') => {
|
// Source switching removed
|
||||||
if (!isScreenFocused) return;
|
|
||||||
|
|
||||||
setCurrentMetadataSource(sourceId);
|
|
||||||
setLoadingMetadataSource(true);
|
|
||||||
|
|
||||||
// Reload metadata with the new source
|
|
||||||
try {
|
|
||||||
let newMetadata = null;
|
|
||||||
|
|
||||||
if (sourceType === 'tmdb') {
|
|
||||||
// Load from TMDB
|
|
||||||
if (id.startsWith('tt')) {
|
|
||||||
// Convert IMDB ID to TMDB ID first
|
|
||||||
const tmdbId = await tmdbService.findTMDBIdByIMDB(id);
|
|
||||||
if (tmdbId) {
|
|
||||||
if (type === 'movie') {
|
|
||||||
const movieDetails = await tmdbService.getMovieDetails(tmdbId.toString());
|
|
||||||
if (movieDetails) {
|
|
||||||
newMetadata = {
|
|
||||||
id: id,
|
|
||||||
type: 'movie',
|
|
||||||
name: movieDetails.title,
|
|
||||||
poster: tmdbService.getImageUrl(movieDetails.poster_path) || '',
|
|
||||||
banner: tmdbService.getImageUrl(movieDetails.backdrop_path) || '',
|
|
||||||
description: movieDetails.overview || '',
|
|
||||||
year: movieDetails.release_date ? parseInt(movieDetails.release_date.substring(0, 4)) : undefined,
|
|
||||||
genres: movieDetails.genres?.map((g: { name: string }) => g.name) || [],
|
|
||||||
inLibrary: metadata?.inLibrary || false,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
} else if (type === 'series') {
|
|
||||||
const showDetails = await tmdbService.getTVShowDetails(tmdbId);
|
|
||||||
if (showDetails) {
|
|
||||||
newMetadata = {
|
|
||||||
id: id,
|
|
||||||
type: 'series',
|
|
||||||
name: showDetails.name,
|
|
||||||
poster: tmdbService.getImageUrl(showDetails.poster_path) || '',
|
|
||||||
banner: tmdbService.getImageUrl(showDetails.backdrop_path) || '',
|
|
||||||
description: showDetails.overview || '',
|
|
||||||
year: showDetails.first_air_date ? parseInt(showDetails.first_air_date.substring(0, 4)) : undefined,
|
|
||||||
genres: showDetails.genres?.map((g: { name: string }) => g.name) || [],
|
|
||||||
inLibrary: metadata?.inLibrary || false,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Load from addon or auto
|
|
||||||
const addonIdToUse = sourceId === 'auto' ? undefined : sourceId;
|
|
||||||
newMetadata = await catalogService.getEnhancedContentDetails(type, id, addonIdToUse);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (newMetadata) {
|
|
||||||
setMetadata(newMetadata);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('[MetadataScreen] Failed to reload metadata with new source:', error);
|
|
||||||
} finally {
|
|
||||||
setLoadingMetadataSource(false);
|
|
||||||
}
|
|
||||||
}, [isScreenFocused, id, type, metadata, tmdbService, catalogService, setMetadata]);
|
|
||||||
|
|
||||||
// Ultra-optimized animated styles - minimal calculations with conditional updates
|
// Ultra-optimized animated styles - minimal calculations with conditional updates
|
||||||
const containerStyle = useAnimatedStyle(() => ({
|
const containerStyle = useAnimatedStyle(() => ({
|
||||||
|
|
@ -659,9 +595,7 @@ const MetadataScreen: React.FC = () => {
|
||||||
imdbId={imdbId}
|
imdbId={imdbId}
|
||||||
type={type as 'movie' | 'series'}
|
type={type as 'movie' | 'series'}
|
||||||
contentId={id}
|
contentId={id}
|
||||||
currentMetadataSource={currentMetadataSource}
|
loadingMetadata={false}
|
||||||
onMetadataSourceChange={handleMetadataSourceChange}
|
|
||||||
loadingMetadata={loadingMetadataSource}
|
|
||||||
renderRatings={() => imdbId && shouldLoadSecondaryData ? (
|
renderRatings={() => imdbId && shouldLoadSecondaryData ? (
|
||||||
<MemoizedRatingsSection imdbId={imdbId} type={type === 'series' ? 'show' : 'movie'} />
|
<MemoizedRatingsSection imdbId={imdbId} type={type === 'series' ? 'show' : 'movie'} />
|
||||||
) : null}
|
) : null}
|
||||||
|
|
|
||||||
|
|
@ -846,6 +846,46 @@ export const StreamsScreen = () => {
|
||||||
);
|
);
|
||||||
}, [selectedEpisode, groupedEpisodes, id]);
|
}, [selectedEpisode, groupedEpisodes, id]);
|
||||||
|
|
||||||
|
// TMDB hydration for series hero (rating/runtime/still)
|
||||||
|
const [tmdbEpisodeOverride, setTmdbEpisodeOverride] = useState<{ vote_average?: number; runtime?: number; still_path?: string } | null>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const hydrateEpisodeFromTmdb = async () => {
|
||||||
|
try {
|
||||||
|
setTmdbEpisodeOverride(null);
|
||||||
|
if (type !== 'series' || !currentEpisode || !id) return;
|
||||||
|
// Skip if data already present
|
||||||
|
const needsHydration = !(currentEpisode as any).runtime || !(currentEpisode as any).vote_average || !currentEpisode.still_path;
|
||||||
|
if (!needsHydration) return;
|
||||||
|
|
||||||
|
// Resolve TMDB show id
|
||||||
|
let tmdbShowId: number | null = null;
|
||||||
|
if (id.startsWith('tmdb:')) {
|
||||||
|
tmdbShowId = parseInt(id.split(':')[1], 10);
|
||||||
|
} else if (id.startsWith('tt')) {
|
||||||
|
tmdbShowId = await tmdbService.findTMDBIdByIMDB(id);
|
||||||
|
}
|
||||||
|
if (!tmdbShowId) return;
|
||||||
|
|
||||||
|
const allEpisodes: Record<string, any[]> = await tmdbService.getAllEpisodes(tmdbShowId) as any;
|
||||||
|
const seasonKey = String(currentEpisode.season_number);
|
||||||
|
const seasonList: any[] = (allEpisodes && (allEpisodes as any)[seasonKey]) || [];
|
||||||
|
const ep = seasonList.find((e: any) => e.episode_number === currentEpisode.episode_number);
|
||||||
|
if (ep) {
|
||||||
|
setTmdbEpisodeOverride({
|
||||||
|
vote_average: ep.vote_average,
|
||||||
|
runtime: ep.runtime,
|
||||||
|
still_path: ep.still_path,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
logger.warn('[StreamsScreen] TMDB hydration failed:', e);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
hydrateEpisodeFromTmdb();
|
||||||
|
}, [type, id, currentEpisode?.season_number, currentEpisode?.episode_number]);
|
||||||
|
|
||||||
const navigateToPlayer = useCallback(async (stream: Stream, options?: { forceVlc?: boolean; headers?: Record<string, string> }) => {
|
const navigateToPlayer = useCallback(async (stream: Stream, options?: { forceVlc?: boolean; headers?: Record<string, string> }) => {
|
||||||
// Prepare available streams for the change source feature
|
// Prepare available streams for the change source feature
|
||||||
const streamsToPass = type === 'series' ? episodeStreams : groupedStreams;
|
const streamsToPass = type === 'series' ? episodeStreams : groupedStreams;
|
||||||
|
|
@ -1336,14 +1376,29 @@ export const StreamsScreen = () => {
|
||||||
return tmdbService.getImageUrl(episodeThumbnail, 'original');
|
return tmdbService.getImageUrl(episodeThumbnail, 'original');
|
||||||
}
|
}
|
||||||
if (!currentEpisode) return null;
|
if (!currentEpisode) return null;
|
||||||
if (currentEpisode.still_path) {
|
const hydratedStill = tmdbEpisodeOverride?.still_path;
|
||||||
|
if (currentEpisode.still_path || hydratedStill) {
|
||||||
if (currentEpisode.still_path.startsWith('http')) {
|
if (currentEpisode.still_path.startsWith('http')) {
|
||||||
return currentEpisode.still_path;
|
return currentEpisode.still_path;
|
||||||
}
|
}
|
||||||
return tmdbService.getImageUrl(currentEpisode.still_path, 'original');
|
const path = currentEpisode.still_path || hydratedStill || '';
|
||||||
|
return tmdbService.getImageUrl(path, 'original');
|
||||||
}
|
}
|
||||||
return metadata?.poster || null;
|
return metadata?.poster || null;
|
||||||
}, [currentEpisode, metadata, episodeThumbnail]);
|
}, [currentEpisode, metadata, episodeThumbnail, tmdbEpisodeOverride?.still_path]);
|
||||||
|
|
||||||
|
// Effective TMDB fields for hero (series)
|
||||||
|
const effectiveEpisodeVote = useMemo(() => {
|
||||||
|
if (!currentEpisode) return 0;
|
||||||
|
const v = (tmdbEpisodeOverride?.vote_average ?? currentEpisode.vote_average) || 0;
|
||||||
|
return typeof v === 'number' ? v : Number(v) || 0;
|
||||||
|
}, [currentEpisode, tmdbEpisodeOverride?.vote_average]);
|
||||||
|
|
||||||
|
const effectiveEpisodeRuntime = useMemo(() => {
|
||||||
|
if (!currentEpisode) return undefined as number | undefined;
|
||||||
|
const r = (tmdbEpisodeOverride?.runtime ?? (currentEpisode as any).runtime) as number | undefined;
|
||||||
|
return r;
|
||||||
|
}, [currentEpisode, tmdbEpisodeOverride?.runtime]);
|
||||||
|
|
||||||
// Prefetch hero/backdrop and title logo when StreamsScreen opens
|
// Prefetch hero/backdrop and title logo when StreamsScreen opens
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
|
@ -1508,33 +1563,35 @@ export const StreamsScreen = () => {
|
||||||
{currentEpisode.name}
|
{currentEpisode.name}
|
||||||
</Text>
|
</Text>
|
||||||
{!!currentEpisode.overview && (
|
{!!currentEpisode.overview && (
|
||||||
<Text style={styles.streamsHeroOverview} numberOfLines={2}>
|
<Animated.View entering={FadeIn.duration(400).delay(320)}>
|
||||||
{currentEpisode.overview}
|
<Text style={styles.streamsHeroOverview} numberOfLines={2}>
|
||||||
</Text>
|
{currentEpisode.overview}
|
||||||
|
</Text>
|
||||||
|
</Animated.View>
|
||||||
)}
|
)}
|
||||||
<View style={styles.streamsHeroMeta}>
|
<Animated.View entering={FadeIn.duration(400).delay(360)} style={styles.streamsHeroMeta}>
|
||||||
<Text style={styles.streamsHeroReleased}>
|
<Text style={styles.streamsHeroReleased}>
|
||||||
{tmdbService.formatAirDate(currentEpisode.air_date)}
|
{tmdbService.formatAirDate(currentEpisode.air_date)}
|
||||||
</Text>
|
</Text>
|
||||||
{currentEpisode.vote_average > 0 && (
|
{effectiveEpisodeVote > 0 && (
|
||||||
<View style={styles.streamsHeroRating}>
|
<Animated.View entering={FadeIn.duration(400).delay(380)} style={styles.streamsHeroRating}>
|
||||||
<Image source={{ uri: TMDB_LOGO }} style={styles.tmdbLogo} contentFit="contain" />
|
<Image source={{ uri: TMDB_LOGO }} style={styles.tmdbLogo} contentFit="contain" />
|
||||||
<Text style={styles.streamsHeroRatingText}>
|
<Text style={styles.streamsHeroRatingText}>
|
||||||
{currentEpisode.vote_average.toFixed(1)}
|
{effectiveEpisodeVote.toFixed(1)}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</Animated.View>
|
||||||
)}
|
)}
|
||||||
{!!currentEpisode.runtime && (
|
{!!effectiveEpisodeRuntime && (
|
||||||
<View style={styles.streamsHeroRuntime}>
|
<Animated.View entering={FadeIn.duration(400).delay(400)} style={styles.streamsHeroRuntime}>
|
||||||
<MaterialIcons name="schedule" size={16} color={colors.mediumEmphasis} />
|
<MaterialIcons name="schedule" size={16} color={colors.mediumEmphasis} />
|
||||||
<Text style={styles.streamsHeroRuntimeText}>
|
<Text style={styles.streamsHeroRuntimeText}>
|
||||||
{currentEpisode.runtime >= 60
|
{effectiveEpisodeRuntime >= 60
|
||||||
? `${Math.floor(currentEpisode.runtime / 60)}h ${currentEpisode.runtime % 60}m`
|
? `${Math.floor(effectiveEpisodeRuntime / 60)}h ${effectiveEpisodeRuntime % 60}m`
|
||||||
: `${currentEpisode.runtime}m`}
|
: `${effectiveEpisodeRuntime}m`}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</Animated.View>
|
||||||
)}
|
)}
|
||||||
</View>
|
</Animated.View>
|
||||||
</Animated.View>
|
</Animated.View>
|
||||||
) : (
|
) : (
|
||||||
// Placeholder to reserve space and avoid layout shift while loading
|
// Placeholder to reserve space and avoid layout shift while loading
|
||||||
|
|
@ -1978,10 +2035,7 @@ const createStyles = (colors: any) => StyleSheet.create({
|
||||||
streamsHeroRating: {
|
streamsHeroRating: {
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
backgroundColor: 'rgba(0,0,0,0.7)',
|
// chip background removed
|
||||||
paddingHorizontal: 6,
|
|
||||||
paddingVertical: 3,
|
|
||||||
borderRadius: 4,
|
|
||||||
marginTop: 0,
|
marginTop: 0,
|
||||||
},
|
},
|
||||||
tmdbLogo: {
|
tmdbLogo: {
|
||||||
|
|
@ -1989,7 +2043,7 @@ const createStyles = (colors: any) => StyleSheet.create({
|
||||||
height: 14,
|
height: 14,
|
||||||
},
|
},
|
||||||
streamsHeroRatingText: {
|
streamsHeroRatingText: {
|
||||||
color: colors.accent,
|
color: colors.highEmphasis,
|
||||||
fontSize: 13,
|
fontSize: 13,
|
||||||
fontWeight: '700',
|
fontWeight: '700',
|
||||||
marginLeft: 4,
|
marginLeft: 4,
|
||||||
|
|
@ -2070,10 +2124,7 @@ const createStyles = (colors: any) => StyleSheet.create({
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
gap: 4,
|
gap: 4,
|
||||||
backgroundColor: 'rgba(0,0,0,0.5)',
|
// chip background removed
|
||||||
paddingHorizontal: 8,
|
|
||||||
paddingVertical: 4,
|
|
||||||
borderRadius: 6,
|
|
||||||
},
|
},
|
||||||
streamsHeroRuntimeText: {
|
streamsHeroRuntimeText: {
|
||||||
color: colors.mediumEmphasis,
|
color: colors.mediumEmphasis,
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue