Skip to content

Commit

Permalink
Merge pull request #1915 from coopcycle/fix/1914-restaurant-opening-h…
Browse files Browse the repository at this point in the history
…ours-warning

display the next shipping time range instead of the opening hours
  • Loading branch information
vladimir-8 authored Nov 26, 2024
2 parents a314705 + 10c59f5 commit e8224a9
Show file tree
Hide file tree
Showing 9 changed files with 79 additions and 70 deletions.
4 changes: 2 additions & 2 deletions src/components/RestaurantCard.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { Text } from 'native-base';
import { Image, StyleSheet, View } from 'react-native';
import { TimingBadge } from '../navigation/checkout/components/RestaurantBadges';
import { useBackgroundContainerColor } from '../styles/theme';
import { isRestaurantAvailable } from '../utils/checkout';
import { isRestaurantOrderingAvailable } from '../utils/checkout';
import { RestaurantBadge } from './RestaurantBadge';
import { RestaurantTag } from './RestaurantTag';
import { RestaurantBanner } from './RestaurantBanner';
Expand Down Expand Up @@ -92,7 +92,7 @@ const styles = StyleSheet.create({

export const RestaurantCard = ({ restaurant }) => {
const backgroundColor = useBackgroundContainerColor();
const isAvailable = isRestaurantAvailable(restaurant);
const isAvailable = isRestaurantOrderingAvailable(restaurant);

return (
<View style={[styles.item, { backgroundColor }]}>
Expand Down
1 change: 1 addition & 0 deletions src/i18n/locales/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -427,6 +427,7 @@
"RESTAURANT_PRODUCT_OPTIONS": "Product options",
"RESTAURANT_PRE_ORDER": "Pre-order",
"RESTAURANT_CLOSED_BUT_OPENS": "This restaurant is closed now but opens {{ datetime }}. You can still order for later on though!",
"RESTAURANT_CLOSED_PRE_ORDER": "This restaurant is closed now. You can still order for {{ datetime }} though!",
"RESTAURANT_CLOSED_AND_NOT_AVAILABLE": "This restaurant is closed and not available until {{ datetime }}.",
"CHECKOUT_AS_GUEST": "Checkout as guest",
"GUEST_CHECKOUT_ORDER_EMAIL_HELP": "To send you updates about your order",
Expand Down
1 change: 1 addition & 0 deletions src/i18n/locales/es.json
Original file line number Diff line number Diff line change
Expand Up @@ -407,6 +407,7 @@
"PRIVACY": "Privacidad",
"RESTAURANT_PRE_ORDER": "Reservar",
"RESTAURANT_CLOSED_BUT_OPENS": "Este comercio se encuentra cerrado pero vuelve a abrir {{ datetime }}. Puedes programar tu pedido!",
"RESTAURANT_CLOSED_PRE_ORDER": "Este restaurante está cerrado ahora. ¡Aún puedes pedir para {{ datetime }}!",
"RESTAURANT_CLOSED_AND_NOT_AVAILABLE": "Este comercio se encuentra cerrado y no disponible hasta el {{ datetime }}.",
"MY_ORDER": "Mi pedido",
"CHECKOUT_AS_GUEST": "Comprar como inivitado",
Expand Down
1 change: 1 addition & 0 deletions src/i18n/locales/fr.json
Original file line number Diff line number Diff line change
Expand Up @@ -403,6 +403,7 @@
"CONNECT_WITH_FACEBOOK": "Se connecter avec Facebook",
"RESTAURANT_PRE_ORDER": "Pré-commander",
"RESTAURANT_CLOSED_BUT_OPENS": "Ce restaurant est fermé maintenant mais ouvre {{ datetime }}. Vous pouvez quand même commander pour plus tard !",
"RESTAURANT_CLOSED_PRE_ORDER": "Ce restaurant est fermé actuellement, vous pouvez tout même commander pour une livraison à {{ datetime }} !",
"RESTAURANT_CLOSED_AND_NOT_AVAILABLE": "Ce restaurant est fermé et non disponible jusqu'à {{ datetime }}.",
"CHECKOUT_AS_GUEST": "Commander en tant qu'invité",
"GUEST_CHECKOUT_ORDER_EMAIL_HELP": "Pour vous envoyer des mises à jour à propos de votre commande",
Expand Down
2 changes: 1 addition & 1 deletion src/navigation/checkout/Restaurant.js
Original file line number Diff line number Diff line change
Expand Up @@ -221,7 +221,7 @@ function Restaurant(props) {
{i18n.t('RESTAURANT_OPENING_HOURS')}
</Text>
<OpeningHours
openingHoursSpecification={props.openingHoursSpecification}
openingHoursSpecification={openingHoursSpecification}
/>
</Box>
<Pressable
Expand Down
4 changes: 2 additions & 2 deletions src/navigation/checkout/components/CartFooterButton.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import { Animated, View } from 'react-native';
import { formatPrice } from '../../../utils/formatting';
import FontAwesome from 'react-native-vector-icons/FontAwesome';
import {
isRestaurantAvailable,
isRestaurantOrderingAvailable,
shouldShowPreOrder,
} from '../../../utils/checkout';
import { useSolidButtonTextColor } from '../../../styles/theme';
Expand Down Expand Up @@ -65,7 +65,7 @@ class CartFooterButton extends Component {
render() {
const { cart, restaurant } = this.props;

const isAvailable = isRestaurantAvailable(restaurant);
const isAvailable = isRestaurantOrderingAvailable(restaurant);
const showPreOrder = shouldShowPreOrder(restaurant);

if (!cart || cart.items.length === 0 || !isAvailable) {
Expand Down
6 changes: 3 additions & 3 deletions src/navigation/checkout/components/RestaurantBadges.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import Svg, { Path } from 'react-native-svg';
import { useBaseTextColor } from '../../../styles/theme';
import {
getNextShippingTimeAsText,
isRestaurantAvailable,
isRestaurantOrderingAvailable,
shouldShowPreOrder,
} from '../../../utils/checkout';

Expand Down Expand Up @@ -49,9 +49,9 @@ export const CategoryBadge = ({ label }) => {
export const TimingBadge = ({ restaurant }) => {
const color = useBaseTextColor();

const isAvailable = isRestaurantAvailable(restaurant);
const shippingTime = getNextShippingTimeAsText(restaurant);
const isAvailable = isRestaurantOrderingAvailable(restaurant);
const showPreOrder = shouldShowPreOrder(restaurant);
const shippingTime = getNextShippingTimeAsText(restaurant);

return (
<HStack style={[styles.badge]}>
Expand Down
69 changes: 38 additions & 31 deletions src/navigation/checkout/components/RestaurantProfile.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,8 +15,8 @@ import DangerAlert from '../../../components/DangerAlert';
import i18n from '../../../i18n';
import moment from 'moment/moment';
import {
isRestaurantAvailable,
isRestaurantOpeningSoon,
getNextShippingTime,
isRestaurantOrderingAvailable,
shouldShowPreOrder,
} from '../../../utils/checkout';
import { RestaurantBanner } from '../../../components/RestaurantBanner';
Expand Down Expand Up @@ -96,17 +96,26 @@ const styles = StyleSheet.create({
});

function OpeningHoursWarning({
currentTimeSlot,
isAvailable,
isOpeningSoon,
openingHoursSpecification,
isOrderingAvailable,
showPreOrder,
nextShippingTime,
}) {
if (showPreOrder) {
if (!isOrderingAvailable) {
//FIXME: this is based on the regular opening hours
// and does not take closing rules ("Holidays") into account
const nextOpeningHours = openingHoursSpecification.currentTimeSlot.timeSlot;

if (!nextOpeningHours) {
// when restaurant is disabled it will be shown on the banner
// and in the 'timing' section
return null;
}

return (
<DangerAlert
adjustsFontSizeToFit={true}
text={`${i18n.t('RESTAURANT_CLOSED_BUT_OPENS', {
datetime: moment(currentTimeSlot.timeSlot[0])
text={`${i18n.t('RESTAURANT_CLOSED_AND_NOT_AVAILABLE', {
datetime: moment(nextOpeningHours[0])
.calendar(moment(), {
sameElse: 'llll',
})
Expand All @@ -116,11 +125,12 @@ function OpeningHoursWarning({
);
}

if (!isAvailable && isOpeningSoon) {
if (showPreOrder) {
return (
<DangerAlert
text={`${i18n.t('RESTAURANT_CLOSED_AND_NOT_AVAILABLE', {
datetime: moment(currentTimeSlot.timeSlot[0])
adjustsFontSizeToFit={true}
text={`${i18n.t('RESTAURANT_CLOSED_PRE_ORDER', {
datetime: moment(nextShippingTime.range[0])
.calendar(moment(), {
sameElse: 'llll',
})
Expand All @@ -130,13 +140,12 @@ function OpeningHoursWarning({
);
}

// when restaurant is not available
// it will be shown on the banner and in the 'timing' section
// no warnings
return null;
}

function BannerOverlay({ isAvailable, showPreOrder }) {
if (!isAvailable) {
function BannerOverlay({ isOrderingAvailable, showPreOrder }) {
if (!isOrderingAvailable) {
return <RestaurantNotAvailableBannerOverlay />;
}

Expand All @@ -152,28 +161,26 @@ function RestaurantProfile({ restaurant, openingHoursSpecification, onInfo }) {
const stroke = useBaseTextColor();
const textSecondary = useSecondaryTextColor();

const currentTimeSlot = useMemo(
() => openingHoursSpecification.currentTimeSlot,
[openingHoursSpecification],
);

const isAvailable = useMemo(
() => isRestaurantAvailable(restaurant),
[restaurant],
);
const isOpeningSoon = useMemo(
() => isRestaurantOpeningSoon(restaurant),
const isOrderingAvailable = useMemo(
() => isRestaurantOrderingAvailable(restaurant),
[restaurant],
);
const showPreOrder = useMemo(
() => shouldShowPreOrder(restaurant),
[restaurant],
);
const nextShippingTime = useMemo(
() => getNextShippingTime(restaurant),
[restaurant],
);

return (
<View style={([styles.profile], { backgroundColor })}>
<RestaurantBanner src={restaurant.bannerImage ?? restaurant.image} />
<BannerOverlay isAvailable={isAvailable} showPreOrder={showPreOrder} />
<BannerOverlay
isOrderingAvailable={isOrderingAvailable}
showPreOrder={showPreOrder}
/>
<View style={styles.detailsWrapper}>
<View style={[styles.logoWrapper, { backgroundColor }]}>
<View style={styles.logoWrapperShadow}>
Expand All @@ -200,10 +207,10 @@ function RestaurantProfile({ restaurant, openingHoursSpecification, onInfo }) {
</View>
<View style={styles.content}>
<OpeningHoursWarning
currentTimeSlot={currentTimeSlot}
isAvailable={isAvailable}
isOpeningSoon={isOpeningSoon}
openingHoursSpecification={openingHoursSpecification}
isOrderingAvailable={isOrderingAvailable}
showPreOrder={showPreOrder}
nextShippingTime={nextShippingTime}
/>
<View style={{ display: 'flex', flexDirection: 'row', gap: 4 }}>
<TimingBadge restaurant={restaurant} />
Expand Down
61 changes: 30 additions & 31 deletions src/utils/checkout.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,13 +11,6 @@ function round5(x) {
}

function timingAsText(timing, now) {
// FIXME
// This hotfixes a bug on the API
// https://github.com/coopcycle/coopcycle-web/issues/2213
if (timing.range[0] === timing.range[1]) {
return i18n.t('NOT_AVAILABLE_ATM');
}

const lower = moment.parseZone(timing.range[0]);

if (timing.fast) {
Expand All @@ -43,18 +36,38 @@ function timingAsText(timing, now) {
return lower.calendar(now);
}

export function getNextShippingTime(restaurant) {

const timing = restaurant.timing.delivery || restaurant.timing.collection;

if (!timing) {
return null;
}

// FIXME
// This hotfixes a bug on the API
// https://github.com/coopcycle/coopcycle-web/issues/2213
if (!Array.isArray(timing.range)) {
return null;
}
if (timing.range[0] === timing.range[1]) {
return null;
}

return timing;
}


export function getNextShippingTimeAsText(restaurant, now) {
now = now || moment();

if (restaurant.timing.delivery) {
return timingAsText(restaurant.timing.delivery, now);
}
const timing = getNextShippingTime(restaurant);

if (restaurant.timing.collection) {
return timingAsText(restaurant.timing.collection, now);
if (!timing) {
return i18n.t('NOT_AVAILABLE_ATM');
}

return i18n.t('NOT_AVAILABLE_ATM');
return timingAsText(timing, now);
}

export function getRestaurantCaption(restaurant) {
Expand All @@ -65,23 +78,9 @@ export function getRestaurantCaption(restaurant) {
* While the restaurant might be available (for ordering)
* it might be either opened or closed at the moment
*/
export function isRestaurantAvailable(restaurant) {
if (!restaurant.timing.delivery && !restaurant.timing.collection) {
return false;
}

// FIXME
// This hotfixes a bug on the API
// https://github.com/coopcycle/coopcycle-web/issues/2213
if (
restaurant.timing.delivery &&
Array.isArray(restaurant.timing.delivery.range) &&
restaurant.timing.delivery.range[0] === restaurant.timing.delivery.range[1]
) {
return false;
}

return true;
export function isRestaurantOrderingAvailable(restaurant) {
const timing = getNextShippingTime(restaurant);
return Boolean(timing);
}

/**
Expand All @@ -105,7 +104,7 @@ export function isRestaurantOpeningSoon(restaurant) {
* If the pre-order is soon, we show a regular order button
*/
export function shouldShowPreOrder(restaurant) {
if (!isRestaurantAvailable(restaurant)) {
if (!isRestaurantOrderingAvailable(restaurant)) {
return false;
}

Expand Down

0 comments on commit e8224a9

Please sign in to comment.