webamp/js/selectors.js
2018-01-17 19:00:36 -08:00

210 lines
5.9 KiB
JavaScript

import { createSelector } from "reselect";
import { denormalize, getTimeStr, clamp, percentToIndex } from "./utils";
import {
BANDS,
PLAYLIST_RESIZE_SEGMENT_HEIGHT,
TRACK_HEIGHT
} from "./constants";
import { createPlaylistURL } from "./playlistHtml";
import * as fromPlaylist from "./reducers/playlist";
export const getEqfData = state => {
const { sliders } = state.equalizer;
const preset = {
name: "Entry1",
preamp: denormalize(sliders.preamp)
};
BANDS.forEach(band => {
preset[`hz${band}`] = denormalize(sliders[band]);
});
const eqfData = {
presets: [preset],
type: "Winamp EQ library file v1.1"
};
return eqfData;
};
const getTracks = state => state.playlist.tracks;
const getTrackOrder = state => state.playlist.trackOrder;
export const getOrderedTracks = createSelector(
getTracks,
getTrackOrder,
(tracks, trackOrder) => trackOrder.filter(id => tracks[id])
);
const getOrderedTrackObjects = createSelector(
getTracks,
getOrderedTracks,
(tracks, trackOrder) => trackOrder.map(id => tracks[id])
);
export const getSelectedTrackObjects = createSelector(
getOrderedTrackObjects,
tracks => tracks.filter(track => track.selected)
);
const runningTimeFromTracks = tracks =>
tracks.reduce((time, track) => time + Number(track.duration), 0);
const getTotalRunningTime = createSelector(
getOrderedTrackObjects,
runningTimeFromTracks
);
const getSelectedRunningTime = createSelector(
getSelectedTrackObjects,
runningTimeFromTracks
);
// Note: We should append "+" to these values if some of the tracks are of unknown time.
export const getRunningTimeMessage = createSelector(
getTotalRunningTime,
getSelectedRunningTime,
(totalRunningTime, selectedRunningTime) =>
`${getTimeStr(selectedRunningTime)}/${getTimeStr(totalRunningTime)}`
);
// TODO: use slectors to get memoization
export const getCurrentTrackIndex = state =>
state.playlist.trackOrder.indexOf(state.playlist.currentTrack);
export const getCurrentTrackNumber = createSelector(
getCurrentTrackIndex,
currentTrackIndex => currentTrackIndex + 1
);
export const getCurrentTrackId = state => state.playlist.currentTrack;
export const nextTrack = (state, n = 1) => {
const { playlist: { trackOrder }, media: { repeat } } = state;
if (trackOrder.length === 0) {
return null;
}
const currentIndex = getCurrentTrackIndex(state);
let nextIndex = currentIndex + n;
if (repeat) {
nextIndex = nextIndex % trackOrder.length;
if (nextIndex < 0) {
// Handle wrapping around backwards
nextIndex += trackOrder.length;
}
return trackOrder[nextIndex];
}
if (currentIndex === trackOrder.length - 1 && n > 0) {
return null;
} else if (currentIndex === 0 && n < 0) {
return null;
}
nextIndex = clamp(nextIndex, 0, trackOrder.length - 1);
return trackOrder[nextIndex];
};
const BASE_WINDOW_HEIGHT = 58;
export const getNumberOfVisibleTracks = state => {
const { playlistSize } = state.display;
return Math.floor(
(BASE_WINDOW_HEIGHT + PLAYLIST_RESIZE_SEGMENT_HEIGHT * playlistSize[1]) /
TRACK_HEIGHT
);
};
export const getOverflowTrackCount = createSelector(
getTrackOrder,
getNumberOfVisibleTracks,
(trackOrder, numberOfVisibleTracks) =>
Math.max(0, trackOrder.length - numberOfVisibleTracks)
);
const _getPlaylistScrollPosition = state =>
state.display.playlistScrollPosition;
export const getPlaylistScrollPosition = createSelector(
getOverflowTrackCount,
_getPlaylistScrollPosition,
(overflowTrackCount, playlistScrollPosition) => {
if (overflowTrackCount === 0) {
return 0;
}
return Math.round(
Math.round(overflowTrackCount * playlistScrollPosition / 100) /
overflowTrackCount *
100
);
}
);
export const getScrollOffset = createSelector(
_getPlaylistScrollPosition,
getTrackOrder,
getNumberOfVisibleTracks,
(playlistScrollPosition, trackOrder, numberOfVisibleTracks) => {
const overflow = Math.max(0, trackOrder.length - numberOfVisibleTracks);
return percentToIndex(playlistScrollPosition / 100, overflow + 1);
}
);
export const getVisibleTrackIds = createSelector(
getScrollOffset,
getTrackOrder,
getNumberOfVisibleTracks,
(offset, trackOrder, numberOfVisibleTracks) =>
trackOrder.slice(offset, offset + numberOfVisibleTracks)
);
export const getPlaylist = state => state.playlist;
export const getDuration = state => state.media.length;
export const getTrackDisplayName = (state, trackId) =>
fromPlaylist.getTrackDisplayName(getPlaylist(state), trackId);
export const getCurrentTrackDisplayName = state => {
const id = getCurrentTrackId(state);
return getTrackDisplayName(state, id);
};
export const getMinimalMediaText = createSelector(
getCurrentTrackNumber,
getCurrentTrackDisplayName,
(trackNumber, name) => (name == null ? null : `${trackNumber}. ${name}`)
);
export const getMediaText = createSelector(
getMinimalMediaText,
getDuration,
(minimalMediaText, duration) =>
minimalMediaText == null
? null
: `${minimalMediaText} (${getTimeStr(duration)}) *** `
);
const getNumberOfTracks = state => getTrackOrder(state).length;
const getPlaylistDuration = createSelector(getTracks, tracks =>
Object.values(tracks).reduce((total, track) => total + track.duration, 0)
);
// TODO: Move to action creator
export const getPlaylistURL = createSelector(
state => state,
getNumberOfTracks,
getPlaylistDuration,
getTrackOrder,
getTracks,
(state, numberOfTracks, playlistDuration, trackOrder, tracks) =>
createPlaylistURL({
numberOfTracks,
averageTrackLength: getTimeStr(playlistDuration / numberOfTracks),
// TODO: Handle hours
playlistLengthMinutes: Math.floor(playlistDuration / 60),
playlistLengthSeconds: Math.floor(playlistDuration % 60),
tracks: trackOrder.map(
(id, i) =>
`${i + 1}. ${getTrackDisplayName(state, id)} (${getTimeStr(
tracks[id].duration
)})`
)
})
);