mirror of
https://github.com/LaCasemate/fab-manager.git
synced 2024-12-10 21:24:20 +01:00
Merge branch 'admin-access-user-reservations' into dev
This commit is contained in:
commit
2e327b5486
@ -6,6 +6,7 @@
|
|||||||
- Add extra_authorize_params to OpenIdConnect config
|
- Add extra_authorize_params to OpenIdConnect config
|
||||||
- Improvement : add a notification to remind users to upload their supporting documents
|
- Improvement : add a notification to remind users to upload their supporting documents
|
||||||
- Cancel payment schedule subscription after update the payment mean
|
- Cancel payment schedule subscription after update the payment mean
|
||||||
|
- admin can see reservations of a member
|
||||||
|
|
||||||
## v6.0.14 2023 September 6
|
## v6.0.14 2023 September 6
|
||||||
|
|
||||||
|
@ -6,9 +6,11 @@ import { useTranslation } from 'react-i18next';
|
|||||||
import { Credit, CreditableType } from '../../../models/credit';
|
import { Credit, CreditableType } from '../../../models/credit';
|
||||||
import CreditAPI from '../../../api/credit';
|
import CreditAPI from '../../../api/credit';
|
||||||
import { HtmlTranslate } from '../../base/html-translate';
|
import { HtmlTranslate } from '../../base/html-translate';
|
||||||
|
import { User } from '../../../models/user';
|
||||||
|
|
||||||
interface CreditsPanelProps {
|
interface CreditsPanelProps {
|
||||||
userId: number,
|
userId: number,
|
||||||
|
currentUser?: User,
|
||||||
onError: (message: string) => void,
|
onError: (message: string) => void,
|
||||||
reservableType: CreditableType
|
reservableType: CreditableType
|
||||||
}
|
}
|
||||||
@ -16,7 +18,7 @@ interface CreditsPanelProps {
|
|||||||
/**
|
/**
|
||||||
* List all available credits for the given user and the given resource
|
* List all available credits for the given user and the given resource
|
||||||
*/
|
*/
|
||||||
const CreditsPanel: React.FC<CreditsPanelProps> = ({ userId, onError, reservableType }) => {
|
const CreditsPanel: React.FC<CreditsPanelProps> = ({ userId, currentUser = null, onError, reservableType }) => {
|
||||||
const { t } = useTranslation('logged');
|
const { t } = useTranslation('logged');
|
||||||
|
|
||||||
const [credits, setCredits] = useState<Array<Credit>>([]);
|
const [credits, setCredits] = useState<Array<Credit>>([]);
|
||||||
@ -37,16 +39,30 @@ const CreditsPanel: React.FC<CreditsPanelProps> = ({ userId, onError, reservable
|
|||||||
/**
|
/**
|
||||||
* Display a placeholder when there's no credits to display
|
* Display a placeholder when there's no credits to display
|
||||||
*/
|
*/
|
||||||
const noCredits = (): ReactNode => {
|
const noCredits = (currentUser: User): ReactNode => {
|
||||||
return (
|
return (
|
||||||
<div className="fab-alert fab-alert--warning">{t('app.logged.dashboard.reservations_dashboard.credits_panel.no_credits')}</div>
|
<div className="fab-alert fab-alert--warning">{t(`app.logged.dashboard.reservations_dashboard.${translationKeyPrefix(currentUser)}.no_credits`) /* eslint-disable-line fabmanager/scoped-translation */ }</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns true if there is a currentUser and current user is manager or admin
|
||||||
|
*/
|
||||||
|
const currentUserIsAdminOrManager = (currentUser: User): boolean => {
|
||||||
|
return currentUser && (currentUser.role === 'admin' || currentUser.role === 'manager');
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns translation key prefix
|
||||||
|
*/
|
||||||
|
const translationKeyPrefix = (currentUser: User): string => {
|
||||||
|
return currentUserIsAdminOrManager(currentUser) ? 'credits_panel_as_admin' : 'credits_panel';
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<FabPanel className="credits-panel">
|
<FabPanel className="credits-panel">
|
||||||
<p className="title">{t('app.logged.dashboard.reservations_dashboard.credits_panel.title')}</p>
|
<p className="title">{t(`app.logged.dashboard.reservations_dashboard.${translationKeyPrefix(currentUser)}.title`) /* eslint-disable-line fabmanager/scoped-translation */}</p>
|
||||||
{credits.length !== 0 &&
|
{credits.length !== 0 && !currentUserIsAdminOrManager(currentUser) &&
|
||||||
<div className="fab-alert fab-alert--warning">
|
<div className="fab-alert fab-alert--warning">
|
||||||
{t('app.logged.dashboard.reservations_dashboard.credits_panel.info')}
|
{t('app.logged.dashboard.reservations_dashboard.credits_panel.info')}
|
||||||
</div>
|
</div>
|
||||||
@ -56,14 +72,14 @@ const CreditsPanel: React.FC<CreditsPanelProps> = ({ userId, onError, reservable
|
|||||||
{credits.map(c => <div key={c.id} className="credits-list-item">
|
{credits.map(c => <div key={c.id} className="credits-list-item">
|
||||||
<p className="title">{c.creditable.name}</p>
|
<p className="title">{c.creditable.name}</p>
|
||||||
<p>
|
<p>
|
||||||
<HtmlTranslate trKey="app.logged.dashboard.reservations_dashboard.credits_panel.remaining_credits_html" options={{ REMAINING: remainingHours(c) }} /><br />
|
<HtmlTranslate trKey={`app.logged.dashboard.reservations_dashboard.${translationKeyPrefix(currentUser)}.remaining_credits_html` /* eslint-disable-line fabmanager/scoped-translation */} options={{ REMAINING: remainingHours(c) }} /><br />
|
||||||
{(c.hours_used && c.hours_used > 0) &&
|
{(c.hours_used && c.hours_used > 0) &&
|
||||||
<HtmlTranslate trKey="app.logged.dashboard.reservations_dashboard.credits_panel.used_credits_html" options={{ USED: c.hours_used }} />
|
<HtmlTranslate trKey={`app.logged.dashboard.reservations_dashboard.${translationKeyPrefix(currentUser)}.used_credits_html` /* eslint-disable-line fabmanager/scoped-translation */} options={{ USED: c.hours_used }} />
|
||||||
}
|
}
|
||||||
</p>
|
</p>
|
||||||
</div>)}
|
</div>)}
|
||||||
</div>
|
</div>
|
||||||
{credits.length === 0 && noCredits()}
|
{credits.length === 0 && noCredits(currentUser)}
|
||||||
</FabPanel>
|
</FabPanel>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -21,13 +21,14 @@ import { HtmlTranslate } from '../../base/html-translate';
|
|||||||
|
|
||||||
interface PrepaidPacksPanelProps {
|
interface PrepaidPacksPanelProps {
|
||||||
user: User,
|
user: User,
|
||||||
|
currentUser?: User,
|
||||||
onError: (message: string) => void
|
onError: (message: string) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* List all available prepaid packs for the given user
|
* List all available prepaid packs for the given user
|
||||||
*/
|
*/
|
||||||
const PrepaidPacksPanel: React.FC<PrepaidPacksPanelProps> = ({ user, onError }) => {
|
const PrepaidPacksPanel: React.FC<PrepaidPacksPanelProps> = ({ user, currentUser = null, onError }) => {
|
||||||
const { t } = useTranslation('logged');
|
const { t } = useTranslation('logged');
|
||||||
|
|
||||||
const [machines, setMachines] = useState<Array<Machine>>([]);
|
const [machines, setMachines] = useState<Array<Machine>>([]);
|
||||||
@ -101,6 +102,20 @@ const PrepaidPacksPanel: React.FC<PrepaidPacksPanelProps> = ({ user, onError })
|
|||||||
return (packs.length > 0 && (!packsForSubscribers || (packsForSubscribers && user.subscribed_plan != null)));
|
return (packs.length > 0 && (!packsForSubscribers || (packsForSubscribers && user.subscribed_plan != null)));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns true if there is a currentUser and current user is manager or admin
|
||||||
|
*/
|
||||||
|
const currentUserIsAdminOrManager = (currentUser: User): boolean => {
|
||||||
|
return currentUser && (currentUser.role === 'admin' || currentUser.role === 'manager');
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns translation key prefix
|
||||||
|
*/
|
||||||
|
const translationKeyPrefix = (currentUser: User): string => {
|
||||||
|
return currentUserIsAdminOrManager(currentUser) ? 'prepaid_packs_panel_as_admin' : 'prepaid_packs_panel';
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Callback triggered when a prepaid pack was successfully bought: refresh the list of packs for the user
|
* Callback triggered when a prepaid pack was successfully bought: refresh the list of packs for the user
|
||||||
*/
|
*/
|
||||||
@ -113,7 +128,7 @@ const PrepaidPacksPanel: React.FC<PrepaidPacksPanelProps> = ({ user, onError })
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<FabPanel className='prepaid-packs-panel'>
|
<FabPanel className='prepaid-packs-panel'>
|
||||||
<p className="title">{t('app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.title')}</p>
|
<p className="title">{t(`app.logged.dashboard.reservations_dashboard.${translationKeyPrefix(currentUser)}.title`) /* eslint-disable-line fabmanager/scoped-translation */}</p>
|
||||||
|
|
||||||
{userPacks.map(pack => (
|
{userPacks.map(pack => (
|
||||||
<div className={`prepaid-packs ${isLow(pack) ? 'is-low' : ''}`} key={pack.id}>
|
<div className={`prepaid-packs ${isLow(pack) ? 'is-low' : ''}`} key={pack.id}>
|
||||||
@ -124,7 +139,7 @@ const PrepaidPacksPanel: React.FC<PrepaidPacksPanelProps> = ({ user, onError })
|
|||||||
|
|
||||||
<div className='prepaid-packs-list-item'>
|
<div className='prepaid-packs-list-item'>
|
||||||
<p className='name'>{pack.prepaid_pack.priceable.name}</p>
|
<p className='name'>{pack.prepaid_pack.priceable.name}</p>
|
||||||
{FormatLib.date(pack.expires_at) && <p className="end">{FormatLib.date(pack.expires_at)}</p>}
|
{pack.expires_at && FormatLib.date(pack.expires_at) && <p className="end">{FormatLib.date(pack.expires_at)}</p>}
|
||||||
<p className="countdown"><span>{(pack.prepaid_pack.minutes - pack.minutes_used) / 60}H</span> / {pack.prepaid_pack.minutes / 60}H</p>
|
<p className="countdown"><span>{(pack.prepaid_pack.minutes - pack.minutes_used) / 60}H</span> / {pack.prepaid_pack.minutes / 60}H</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -143,7 +158,7 @@ const PrepaidPacksPanel: React.FC<PrepaidPacksPanelProps> = ({ user, onError })
|
|||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
|
|
||||||
{canBuyPacks() && <div className='prepaid-packs-cta'>
|
{canBuyPacks() && !currentUserIsAdminOrManager(currentUser) && <div className='prepaid-packs-cta'>
|
||||||
<p>{t('app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.cta_info')}</p>
|
<p>{t('app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.cta_info')}</p>
|
||||||
<form onSubmit={handleSubmit(onBuyPack)}>
|
<form onSubmit={handleSubmit(onBuyPack)}>
|
||||||
<FormSelect options={buildMachinesOptions(machines)} control={control} id="machine_id" rules={{ required: true }} formState={formState} label={t('app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.select_machine')} />
|
<FormSelect options={buildMachinesOptions(machines)} control={control} id="machine_id" rules={{ required: true }} formState={formState} label={t('app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.select_machine')} />
|
||||||
@ -163,7 +178,7 @@ const PrepaidPacksPanel: React.FC<PrepaidPacksPanelProps> = ({ user, onError })
|
|||||||
onSuccess={onPackBoughtSuccess} />}
|
onSuccess={onPackBoughtSuccess} />}
|
||||||
</div>}
|
</div>}
|
||||||
{packs.length === 0 && <p>{t('app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.no_packs')}</p>}
|
{packs.length === 0 && <p>{t('app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.no_packs')}</p>}
|
||||||
{(packsForSubscribers && user.subscribed_plan == null && packs.length > 0) &&
|
{(packsForSubscribers && user.subscribed_plan == null && packs.length > 0 && !currentUserIsAdminOrManager(currentUser)) &&
|
||||||
<HtmlTranslate trKey={'app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.reserved_for_subscribers_html'} options={{ LINK: '#!/plans' }} />
|
<HtmlTranslate trKey={'app.logged.dashboard.reservations_dashboard.prepaid_packs_panel.reserved_for_subscribers_html'} options={{ LINK: '#!/plans' }} />
|
||||||
}
|
}
|
||||||
</FabPanel>
|
</FabPanel>
|
||||||
|
@ -14,13 +14,14 @@ declare const Application: IApplication;
|
|||||||
|
|
||||||
interface ReservationsDashboardProps {
|
interface ReservationsDashboardProps {
|
||||||
onError: (message: string) => void,
|
onError: (message: string) => void,
|
||||||
user: User
|
user: User,
|
||||||
|
currentUser?: User
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* User dashboard showing everything about his spaces/machine reservations and also remaining credits
|
* User dashboard showing everything about his spaces/machine reservations and also remaining credits
|
||||||
*/
|
*/
|
||||||
const ReservationsDashboard: React.FC<ReservationsDashboardProps> = ({ onError, user }) => {
|
const ReservationsDashboard: React.FC<ReservationsDashboardProps> = ({ onError, user, currentUser = null }) => {
|
||||||
const { t } = useTranslation('logged');
|
const { t } = useTranslation('logged');
|
||||||
const [modules, setModules] = useState<Map<SettingName, string>>();
|
const [modules, setModules] = useState<Map<SettingName, string>>();
|
||||||
|
|
||||||
@ -34,17 +35,17 @@ const ReservationsDashboard: React.FC<ReservationsDashboardProps> = ({ onError,
|
|||||||
<div className="reservations-dashboard">
|
<div className="reservations-dashboard">
|
||||||
{modules?.get('machines_module') !== 'false' && <div className="section">
|
{modules?.get('machines_module') !== 'false' && <div className="section">
|
||||||
<p className="section-title">{t('app.logged.dashboard.reservations_dashboard.machine_section_title')}</p>
|
<p className="section-title">{t('app.logged.dashboard.reservations_dashboard.machine_section_title')}</p>
|
||||||
<CreditsPanel userId={user.id} onError={onError} reservableType="Machine" />
|
<CreditsPanel userId={user.id} currentUser={currentUser} onError={onError} reservableType="Machine" />
|
||||||
<PrepaidPacksPanel user={user} onError={onError} />
|
<PrepaidPacksPanel user={user} currentUser={currentUser} onError={onError} />
|
||||||
<ReservationsPanel userId={user.id} onError={onError} reservableType="Machine" />
|
<ReservationsPanel userId={user.id} currentUser={currentUser} onError={onError} reservableType="Machine" />
|
||||||
</div>}
|
</div>}
|
||||||
{modules?.get('spaces_module') !== 'false' && <div className="section">
|
{modules?.get('spaces_module') !== 'false' && <div className="section">
|
||||||
<p className="section-title">{t('app.logged.dashboard.reservations_dashboard.space_section_title')}</p>
|
<p className="section-title">{t('app.logged.dashboard.reservations_dashboard.space_section_title')}</p>
|
||||||
<CreditsPanel userId={user.id} onError={onError} reservableType="Space" />
|
<CreditsPanel userId={user.id} currentUser={currentUser} onError={onError} reservableType="Space" />
|
||||||
<ReservationsPanel userId={user.id} onError={onError} reservableType="Space" />
|
<ReservationsPanel userId={user.id} currentUser={currentUser} onError={onError} reservableType="Space" />
|
||||||
</div>}
|
</div>}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
Application.Components.component('reservationsDashboard', react2angular(ReservationsDashboard, ['onError', 'user']));
|
Application.Components.component('reservationsDashboard', react2angular(ReservationsDashboard, ['onError', 'user', 'currentUser']));
|
||||||
|
@ -9,9 +9,11 @@ import { Loader } from '../../base/loader';
|
|||||||
import FormatLib from '../../../lib/format';
|
import FormatLib from '../../../lib/format';
|
||||||
import _ from 'lodash';
|
import _ from 'lodash';
|
||||||
import { FabButton } from '../../base/fab-button';
|
import { FabButton } from '../../base/fab-button';
|
||||||
|
import { User } from '../../../models/user';
|
||||||
|
|
||||||
interface SpaceReservationsProps {
|
interface SpaceReservationsProps {
|
||||||
userId: number,
|
userId: number,
|
||||||
|
currentUser?: User,
|
||||||
onError: (message: string) => void,
|
onError: (message: string) => void,
|
||||||
reservableType: 'Machine' | 'Space'
|
reservableType: 'Machine' | 'Space'
|
||||||
}
|
}
|
||||||
@ -19,7 +21,7 @@ interface SpaceReservationsProps {
|
|||||||
/**
|
/**
|
||||||
* List all reservations for the given user and the given type
|
* List all reservations for the given user and the given type
|
||||||
*/
|
*/
|
||||||
const ReservationsPanel: React.FC<SpaceReservationsProps> = ({ userId, onError, reservableType }) => {
|
const ReservationsPanel: React.FC<SpaceReservationsProps> = ({ userId, currentUser = null, onError, reservableType }) => {
|
||||||
const { t } = useTranslation('logged');
|
const { t } = useTranslation('logged');
|
||||||
|
|
||||||
const [reservations, setReservations] = useState<Array<Reservation>>([]);
|
const [reservations, setReservations] = useState<Array<Reservation>>([]);
|
||||||
@ -71,6 +73,20 @@ const ReservationsPanel: React.FC<SpaceReservationsProps> = ({ userId, onError,
|
|||||||
return reservation.slots_reservations_attributes.map(sr => sr.canceled_at).every(ca => ca != null);
|
return reservation.slots_reservations_attributes.map(sr => sr.canceled_at).every(ca => ca != null);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns true if there is a currentUser and current user is manager or admin
|
||||||
|
*/
|
||||||
|
const currentUserIsAdminOrManager = (currentUser: User): boolean => {
|
||||||
|
return currentUser && (currentUser.role === 'admin' || currentUser.role === 'manager');
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns translation key prefix
|
||||||
|
*/
|
||||||
|
const translationKeyPrefix = (currentUser: User): string => {
|
||||||
|
return currentUserIsAdminOrManager(currentUser) ? 'reservations_panel_as_admin' : 'reservations_panel';
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Render the reservation in a user-friendly way
|
* Render the reservation in a user-friendly way
|
||||||
*/
|
*/
|
||||||
@ -95,7 +111,7 @@ const ReservationsPanel: React.FC<SpaceReservationsProps> = ({ userId, onError,
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<FabPanel className="reservations-panel">
|
<FabPanel className="reservations-panel">
|
||||||
<p className="title">{t('app.logged.dashboard.reservations_dashboard.reservations_panel.title')}</p>
|
<p className="title">{t(`app.logged.dashboard.reservations_dashboard.${translationKeyPrefix(currentUser)}.title`) /* eslint-disable-line fabmanager/scoped-translation */}</p>
|
||||||
<div className="reservations">
|
<div className="reservations">
|
||||||
{futur.length === 0
|
{futur.length === 0
|
||||||
? noReservations()
|
? noReservations()
|
||||||
|
@ -62,6 +62,10 @@
|
|||||||
|
|
||||||
</uib-tab>
|
</uib-tab>
|
||||||
|
|
||||||
|
<uib-tab heading="{{ 'app.admin.members_edit.reservations' | translate }}">
|
||||||
|
<reservations-dashboard user="user" current-user="currentUser" on-error="onError" />
|
||||||
|
</uib-tab>
|
||||||
|
|
||||||
<uib-tab heading="{{ 'app.admin.members_edit.supporting_documents' | translate }}" ng-show="hasProofOfIdentityTypes">
|
<uib-tab heading="{{ 'app.admin.members_edit.supporting_documents' | translate }}" ng-show="hasProofOfIdentityTypes">
|
||||||
<supporting-documents-validation
|
<supporting-documents-validation
|
||||||
operator="currentUser"
|
operator="currentUser"
|
||||||
|
@ -19,6 +19,6 @@ class CreditPolicy < ApplicationPolicy
|
|||||||
end
|
end
|
||||||
|
|
||||||
def user_resource?
|
def user_resource?
|
||||||
record.id == user.id
|
record.id == user.id || user.admin?
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -80,6 +80,7 @@ Rails.application.configure do
|
|||||||
config.file_watcher = ActiveSupport::EventedFileUpdateChecker
|
config.file_watcher = ActiveSupport::EventedFileUpdateChecker
|
||||||
|
|
||||||
config.log_level = Rails.application.secrets.log_level || :debug
|
config.log_level = Rails.application.secrets.log_level || :debug
|
||||||
|
config.log_tags = [:request_id]
|
||||||
|
|
||||||
config.action_controller.default_url_options = {
|
config.action_controller.default_url_options = {
|
||||||
host: Rails.application.secrets.default_host,
|
host: Rails.application.secrets.default_host,
|
||||||
|
@ -1273,6 +1273,7 @@ en:
|
|||||||
#edit a member
|
#edit a member
|
||||||
members_edit:
|
members_edit:
|
||||||
subscription: "Subscription"
|
subscription: "Subscription"
|
||||||
|
reservations: "Reservations"
|
||||||
duration: "Duration:"
|
duration: "Duration:"
|
||||||
expires_at: "Expires at:"
|
expires_at: "Expires at:"
|
||||||
price_: "Price:"
|
price_: "Price:"
|
||||||
|
@ -1273,6 +1273,7 @@ fr:
|
|||||||
#edit a member
|
#edit a member
|
||||||
members_edit:
|
members_edit:
|
||||||
subscription: "Abonnement"
|
subscription: "Abonnement"
|
||||||
|
reservations: "Réservations"
|
||||||
duration: "Durée :"
|
duration: "Durée :"
|
||||||
expires_at: "Expire le :"
|
expires_at: "Expire le :"
|
||||||
price_: "Prix :"
|
price_: "Prix :"
|
||||||
@ -1799,6 +1800,7 @@ fr:
|
|||||||
reservation_context_feature: "Activer la fonctionnalité \"Nature de réservation\""
|
reservation_context_feature: "Activer la fonctionnalité \"Nature de réservation\""
|
||||||
reservation_context_options: Options de nature de réservation
|
reservation_context_options: Options de nature de réservation
|
||||||
add_a_reservation_context: Ajouter une nouvelle nature
|
add_a_reservation_context: Ajouter une nouvelle nature
|
||||||
|
confirmation_required: Confirmation requise
|
||||||
do_you_really_want_to_delete_this_reservation_context: "Êtes-vous sûr de vouloir supprimer cette nature ?"
|
do_you_really_want_to_delete_this_reservation_context: "Êtes-vous sûr de vouloir supprimer cette nature ?"
|
||||||
unable_to_delete_reservation_context_already_related_to_reservations: "Impossible de supprimer ce contexte car il est déjà associé à une réservation"
|
unable_to_delete_reservation_context_already_related_to_reservations: "Impossible de supprimer ce contexte car il est déjà associé à une réservation"
|
||||||
unable_to_delete_reservation_context_an_error_occured: "Impossible de supprimer : une erreur est survenue"
|
unable_to_delete_reservation_context_an_error_occured: "Impossible de supprimer : une erreur est survenue"
|
||||||
|
@ -154,12 +154,19 @@ en:
|
|||||||
no_reservation: "No reservation"
|
no_reservation: "No reservation"
|
||||||
show_more: "Show more"
|
show_more: "Show more"
|
||||||
cancelled_slot: "Cancelled"
|
cancelled_slot: "Cancelled"
|
||||||
|
reservations_panel_as_admin:
|
||||||
|
title: "Reservations"
|
||||||
credits_panel:
|
credits_panel:
|
||||||
title: "My credits"
|
title: "My credits"
|
||||||
info: "Your subscription comes with free credits you can use when reserving"
|
info: "Your subscription comes with free credits you can use when reserving"
|
||||||
remaining_credits_html: "You can book {REMAINING} {REMAINING, plural, one{slot} other{slots}} for free."
|
remaining_credits_html: "You can book {REMAINING} {REMAINING, plural, one{slot} other{slots}} for free."
|
||||||
used_credits_html: "You have already used <strong> {USED} {USED, plural, =0{credit} one{credit} other{credits}}</strong>."
|
used_credits_html: "You have already used <strong> {USED} {USED, plural, =0{credit} one{credit} other{credits}}</strong>."
|
||||||
no_credits: "You don't have any credits yet. Some subscriptions may allow you to book some slots for free."
|
no_credits: "You don't have any credits yet. Some subscriptions may allow you to book some slots for free."
|
||||||
|
credits_panel_as_admin:
|
||||||
|
title: "Credits"
|
||||||
|
remaining_credits_html: "{REMAINING} {REMAINING, plural, one{slot} other{slots}} can be booked for free."
|
||||||
|
used_credits_html: "<strong> {USED} {USED, plural, =0{credit} one{credit} other{credits}}</strong> already used."
|
||||||
|
no_credits: "No credits yet."
|
||||||
prepaid_packs_panel:
|
prepaid_packs_panel:
|
||||||
title: "My prepaid packs"
|
title: "My prepaid packs"
|
||||||
name: "Prepaid pack name"
|
name: "Prepaid pack name"
|
||||||
@ -172,6 +179,8 @@ en:
|
|||||||
cta_button: "Buy a pack"
|
cta_button: "Buy a pack"
|
||||||
no_packs: "No prepaid packs available for sale"
|
no_packs: "No prepaid packs available for sale"
|
||||||
reserved_for_subscribers_html: 'The purchase of prepaid packs is reserved for subscribers. <a href="{LINK}">Subscribe now</a> to benefit.'
|
reserved_for_subscribers_html: 'The purchase of prepaid packs is reserved for subscribers. <a href="{LINK}">Subscribe now</a> to benefit.'
|
||||||
|
prepaid_packs_panel_as_admin:
|
||||||
|
title: "Prepaid packs"
|
||||||
#public profil of a member
|
#public profil of a member
|
||||||
members_show:
|
members_show:
|
||||||
members_list: "Members list"
|
members_list: "Members list"
|
||||||
|
@ -154,12 +154,19 @@ fr:
|
|||||||
no_reservation: "Aucune réservation"
|
no_reservation: "Aucune réservation"
|
||||||
show_more: "Afficher plus"
|
show_more: "Afficher plus"
|
||||||
cancelled_slot: "Annulé"
|
cancelled_slot: "Annulé"
|
||||||
|
reservations_panel_as_admin:
|
||||||
|
title: "Réservations"
|
||||||
credits_panel:
|
credits_panel:
|
||||||
title: "Mes crédits"
|
title: "Mes crédits"
|
||||||
info: "Avec votre abonnement, vous bénéficiez de crédits gratuits que vous pouvez utiliser lors de vos réservations"
|
info: "Avec votre abonnement, vous bénéficiez de crédits gratuits que vous pouvez utiliser lors de vos réservations"
|
||||||
remaining_credits_html: "Vous pouvez réserver {REMAINING} {REMAINING, plural, one{créneau} other{créneaux}} gratuitement."
|
remaining_credits_html: "Vous pouvez réserver {REMAINING} {REMAINING, plural, one{créneau} other{créneaux}} gratuitement."
|
||||||
used_credits_html: "Vous avez déjà utilisé <strong> {USED} {USED, plural, =0{crédit} one{crédit} other{crédits}}</strong>."
|
used_credits_html: "Vous avez déjà utilisé <strong> {USED} {USED, plural, =0{crédit} one{crédit} other{crédits}}</strong>."
|
||||||
no_credits: "Vous n'avez pas encore de crédits. Certains abonnements peuvent vous permettre de réserver des créneaux gratuitement."
|
no_credits: "Vous n'avez pas encore de crédits. Certains abonnements peuvent vous permettre de réserver des créneaux gratuitement."
|
||||||
|
credits_panel_as_admin:
|
||||||
|
title: "Crédits"
|
||||||
|
remaining_credits_html: "{REMAINING} {REMAINING, plural, one{créneau} other{créneaux}} peuvent être réservés gratuitement."
|
||||||
|
used_credits_html: "<strong> {USED} {USED, plural, =0{crédit} one{crédit} other{crédits}}</strong> déjà utilisés."
|
||||||
|
no_credits: "Pas de crédits."
|
||||||
prepaid_packs_panel:
|
prepaid_packs_panel:
|
||||||
title: "Mes packs d'heures prépayées"
|
title: "Mes packs d'heures prépayées"
|
||||||
name: "Nom du pack d'heures prépayées"
|
name: "Nom du pack d'heures prépayées"
|
||||||
@ -172,6 +179,8 @@ fr:
|
|||||||
cta_button: "Acheter un pack"
|
cta_button: "Acheter un pack"
|
||||||
no_packs: "Aucun pack prépayé disponible à la vente"
|
no_packs: "Aucun pack prépayé disponible à la vente"
|
||||||
reserved_for_subscribers_html: 'L''achat de packs prépayés est réservé aux abonnés. <a href="{LINK}">Abonnez-vous maintenant</a> pour en bénéficier.'
|
reserved_for_subscribers_html: 'L''achat de packs prépayés est réservé aux abonnés. <a href="{LINK}">Abonnez-vous maintenant</a> pour en bénéficier.'
|
||||||
|
prepaid_packs_panel_as_admin:
|
||||||
|
title: "Packs d'heures prépayées"
|
||||||
#public profil of a member
|
#public profil of a member
|
||||||
members_show:
|
members_show:
|
||||||
members_list: "Liste des membres"
|
members_list: "Liste des membres"
|
||||||
|
@ -3390,9 +3390,7 @@ CREATE TABLE public.spaces (
|
|||||||
updated_at timestamp without time zone NOT NULL,
|
updated_at timestamp without time zone NOT NULL,
|
||||||
characteristics text,
|
characteristics text,
|
||||||
disabled boolean,
|
disabled boolean,
|
||||||
deleted_at timestamp without time zone,
|
deleted_at timestamp without time zone
|
||||||
ancestry character varying NOT NULL COLLATE pg_catalog."C",
|
|
||||||
ancestry_depth integer DEFAULT 0
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
@ -7470,13 +7468,6 @@ CREATE INDEX index_spaces_availabilities_on_availability_id ON public.spaces_ava
|
|||||||
CREATE INDEX index_spaces_availabilities_on_space_id ON public.spaces_availabilities USING btree (space_id);
|
CREATE INDEX index_spaces_availabilities_on_space_id ON public.spaces_availabilities USING btree (space_id);
|
||||||
|
|
||||||
|
|
||||||
--
|
|
||||||
-- Name: index_spaces_on_ancestry; Type: INDEX; Schema: public; Owner: -
|
|
||||||
--
|
|
||||||
|
|
||||||
CREATE INDEX index_spaces_on_ancestry ON public.spaces USING btree (ancestry);
|
|
||||||
|
|
||||||
|
|
||||||
--
|
--
|
||||||
-- Name: index_spaces_on_deleted_at; Type: INDEX; Schema: public; Owner: -
|
-- Name: index_spaces_on_deleted_at; Type: INDEX; Schema: public; Owner: -
|
||||||
--
|
--
|
||||||
@ -9266,5 +9257,3 @@ INSERT INTO "schema_migrations" (version) VALUES
|
|||||||
('20230831103208'),
|
('20230831103208'),
|
||||||
('20230901090637'),
|
('20230901090637'),
|
||||||
('20230907124230');
|
('20230907124230');
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user