mirror of
https://github.com/ManInDark/HabitTrove.git
synced 2026-01-21 06:34:30 +01:00
fix modal and invalid frequency
This commit is contained in:
@@ -1,5 +1,13 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## Version 0.2.19
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
* settings button not working
|
||||||
|
* fixed delete dialog modal blocks page interaction (#149)
|
||||||
|
* disable submit button when frequency is invaid
|
||||||
|
|
||||||
## Version 0.2.18
|
## Version 0.2.18
|
||||||
|
|
||||||
### Improved
|
### Improved
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ export default function AddEditHabitModal({ onClose, onSave, habit, isTask }: Ad
|
|||||||
const [ruleText, setRuleText] = useState<string>(initialRuleText)
|
const [ruleText, setRuleText] = useState<string>(initialRuleText)
|
||||||
const { currentUser } = useHelpers()
|
const { currentUser } = useHelpers()
|
||||||
const [isQuickDatesOpen, setIsQuickDatesOpen] = useState(false)
|
const [isQuickDatesOpen, setIsQuickDatesOpen] = useState(false)
|
||||||
const [ruleError, setRuleError] = useState<string | null>(null); // State for validation message
|
const [errorMessage, setErrorMessage] = useState<string | null>(null); // State for validation message
|
||||||
const [selectedUserIds, setSelectedUserIds] = useState<string[]>((habit?.userIds || []).filter(id => id !== currentUser?.id))
|
const [selectedUserIds, setSelectedUserIds] = useState<string[]>((habit?.userIds || []).filter(id => id !== currentUser?.id))
|
||||||
const [usersData] = useAtom(usersAtom)
|
const [usersData] = useAtom(usersAtom)
|
||||||
const users = usersData.users
|
const users = usersData.users
|
||||||
@@ -183,9 +183,10 @@ export default function AddEditHabitModal({ onClose, onSave, habit, isTask }: Ad
|
|||||||
<div className="col-start-2 col-span-3 text-sm">
|
<div className="col-start-2 col-span-3 text-sm">
|
||||||
{(() => {
|
{(() => {
|
||||||
let displayText = '';
|
let displayText = '';
|
||||||
let errorMessage: string | null = null;
|
|
||||||
const { result, message } = convertHumanReadableFrequencyToMachineReadable({ text: ruleText, timezone: settings.system.timezone, isRecurring: isRecurRule });
|
const { result, message } = convertHumanReadableFrequencyToMachineReadable({ text: ruleText, timezone: settings.system.timezone, isRecurring: isRecurRule });
|
||||||
errorMessage = message;
|
if (message !== errorMessage) { // Only update if it changed to avoid re-renders
|
||||||
|
setErrorMessage(message);
|
||||||
|
}
|
||||||
displayText = convertMachineReadableFrequencyToHumanReadable({ frequency: result, isRecurRule, timezone: settings.system.timezone })
|
displayText = convertMachineReadableFrequencyToHumanReadable({ frequency: result, isRecurRule, timezone: settings.system.timezone })
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -320,7 +321,7 @@ export default function AddEditHabitModal({ onClose, onSave, habit, isTask }: Ad
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
<DialogFooter>
|
<DialogFooter>
|
||||||
<Button type="submit">
|
<Button type="submit" disabled={!!errorMessage}>
|
||||||
{habit
|
{habit
|
||||||
? t('saveChangesButton')
|
? t('saveChangesButton')
|
||||||
: t(isTask ? 'addTaskButton' : 'addHabitButton')}
|
: t(isTask ? 'addTaskButton' : 'addHabitButton')}
|
||||||
|
|||||||
@@ -113,11 +113,13 @@ export default function HabitItem({ habit, onEdit, onDelete }: HabitItemProps) {
|
|||||||
</CardHeader>
|
</CardHeader>
|
||||||
<CardContent className="flex-1">
|
<CardContent className="flex-1">
|
||||||
<p className={`text-sm ${habit.archived ? 'text-gray-400 dark:text-gray-500' : 'text-gray-500'}`}>
|
<p className={`text-sm ${habit.archived ? 'text-gray-400 dark:text-gray-500' : 'text-gray-500'}`}>
|
||||||
{t('whenLabel', { frequency: convertMachineReadableFrequencyToHumanReadable({
|
{t('whenLabel', {
|
||||||
|
frequency: convertMachineReadableFrequencyToHumanReadable({
|
||||||
frequency: habit.frequency,
|
frequency: habit.frequency,
|
||||||
isRecurRule,
|
isRecurRule,
|
||||||
timezone: settings.system.timezone
|
timezone: settings.system.timezone
|
||||||
})})}
|
})
|
||||||
|
})}
|
||||||
</p>
|
</p>
|
||||||
<div className="flex items-center mt-2">
|
<div className="flex items-center mt-2">
|
||||||
<Coins className={`h-4 w-4 mr-1 ${habit.archived ? 'text-gray-400 dark:text-gray-500' : 'text-yellow-400'}`} />
|
<Coins className={`h-4 w-4 mr-1 ${habit.archived ? 'text-gray-400 dark:text-gray-500' : 'text-yellow-400'}`} />
|
||||||
@@ -190,7 +192,7 @@ export default function HabitItem({ habit, onEdit, onDelete }: HabitItemProps) {
|
|||||||
<span className="ml-2">{t('editButton')}</span>
|
<span className="ml-2">{t('editButton')}</span>
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
<DropdownMenu>
|
<DropdownMenu modal={false}>
|
||||||
<DropdownMenuTrigger asChild>
|
<DropdownMenuTrigger asChild>
|
||||||
<Button variant="ghost" size="sm" className="h-8 w-8 p-0">
|
<Button variant="ghost" size="sm" className="h-8 w-8 p-0">
|
||||||
<MoreVertical className="h-4 w-4" />
|
<MoreVertical className="h-4 w-4" />
|
||||||
|
|||||||
@@ -110,17 +110,18 @@ export function Profile() {
|
|||||||
</div>
|
</div>
|
||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
<DropdownMenuItem className="cursor-pointer px-2 py-1.5" asChild>
|
<DropdownMenuItem className="cursor-pointer px-2 py-1.5" asChild>
|
||||||
<div className="flex items-center justify-between w-full">
|
{/* need the Link element to be the direct child of the DropdownMenuItem, since we are using asChild here */}
|
||||||
<div className="flex items-center gap-2">
|
|
||||||
<Settings className="h-4 w-4" />
|
|
||||||
<Link
|
<Link
|
||||||
href="/settings"
|
href="/settings"
|
||||||
aria-label={t('settingsLink')}
|
aria-label={t('settingsLink')}
|
||||||
|
className="flex items-center justify-between w-full"
|
||||||
|
onClick={() => setOpen(false)} // Ensure dropdown closes on click
|
||||||
>
|
>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Settings className="h-4 w-4" />
|
||||||
<span>{t('settingsLink')}</span>
|
<span>{t('settingsLink')}</span>
|
||||||
|
</div>
|
||||||
</Link>
|
</Link>
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
<DropdownMenuItem className="cursor-pointer px-2 py-1.5" onClick={() => {
|
<DropdownMenuItem className="cursor-pointer px-2 py-1.5" onClick={() => {
|
||||||
setOpen(false); // Close the dropdown
|
setOpen(false); // Close the dropdown
|
||||||
|
|||||||
@@ -141,7 +141,7 @@ export default function WishlistItem({
|
|||||||
<span className="ml-2">{t('editButton')}</span>
|
<span className="ml-2">{t('editButton')}</span>
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
<DropdownMenu>
|
<DropdownMenu modal={false}>
|
||||||
<DropdownMenuTrigger asChild>
|
<DropdownMenuTrigger asChild>
|
||||||
<Button variant="ghost" size="sm" className="h-8 w-8 p-0">
|
<Button variant="ghost" size="sm" className="h-8 w-8 p-0">
|
||||||
<MoreVertical className="h-4 w-4" />
|
<MoreVertical className="h-4 w-4" />
|
||||||
@@ -166,7 +166,7 @@ export default function WishlistItem({
|
|||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
<DropdownMenuSeparator className="sm:hidden" />
|
<DropdownMenuSeparator className="sm:hidden" />
|
||||||
<DropdownMenuItem
|
<DropdownMenuItem
|
||||||
className="text-red-600 focus:text-red-600 dark:text-red-400 dark:focus:text-red-400 cursor-pointer"
|
className="text-red-600 focus:text-red-600 dark:text-red-400 dark:focus:text-red-400"
|
||||||
onClick={onDelete}
|
onClick={onDelete}
|
||||||
disabled={!canWrite}
|
disabled={!canWrite}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ const DialogOverlay = React.forwardRef<
|
|||||||
<DialogPrimitive.Overlay
|
<DialogPrimitive.Overlay
|
||||||
ref={ref}
|
ref={ref}
|
||||||
className={cn(
|
className={cn(
|
||||||
"fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=open]:fade-in-0 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:pointer-events-none",
|
"fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0",
|
||||||
className
|
className
|
||||||
)}
|
)}
|
||||||
{...props}
|
{...props}
|
||||||
|
|||||||
@@ -423,6 +423,7 @@
|
|||||||
"cancel": "Abbrechen"
|
"cancel": "Abbrechen"
|
||||||
},
|
},
|
||||||
"useCoins": {
|
"useCoins": {
|
||||||
|
"addedCoinsDescription": "{amount} Münzen hinzugefügt",
|
||||||
"invalidAmountTitle": "Ungültiger Betrag",
|
"invalidAmountTitle": "Ungültiger Betrag",
|
||||||
"invalidAmountDescription": "Bitte geben Sie eine gültige positive Zahl ein",
|
"invalidAmountDescription": "Bitte geben Sie eine gültige positive Zahl ein",
|
||||||
"successTitle": "Erfolg",
|
"successTitle": "Erfolg",
|
||||||
|
|||||||
@@ -408,6 +408,7 @@
|
|||||||
"notEnoughCoinsDescription": "You need {coinsNeeded} more coins to redeem this reward."
|
"notEnoughCoinsDescription": "You need {coinsNeeded} more coins to redeem this reward."
|
||||||
},
|
},
|
||||||
"useCoins": {
|
"useCoins": {
|
||||||
|
"addedCoinsDescription": "Added {amount} coins",
|
||||||
"invalidAmountTitle": "Invalid amount",
|
"invalidAmountTitle": "Invalid amount",
|
||||||
"invalidAmountDescription": "Please enter a valid positive number",
|
"invalidAmountDescription": "Please enter a valid positive number",
|
||||||
"successTitle": "Success",
|
"successTitle": "Success",
|
||||||
|
|||||||
@@ -423,6 +423,7 @@
|
|||||||
"cancel": "Cancelar"
|
"cancel": "Cancelar"
|
||||||
},
|
},
|
||||||
"useCoins": {
|
"useCoins": {
|
||||||
|
"addedCoinsDescription": "Se añadieron {amount} monedas",
|
||||||
"invalidAmountTitle": "Cantidad inválida",
|
"invalidAmountTitle": "Cantidad inválida",
|
||||||
"invalidAmountDescription": "Por favor ingresa un número positivo válido",
|
"invalidAmountDescription": "Por favor ingresa un número positivo válido",
|
||||||
"successTitle": "Éxito",
|
"successTitle": "Éxito",
|
||||||
|
|||||||
@@ -423,6 +423,7 @@
|
|||||||
"cancel": "Annuler"
|
"cancel": "Annuler"
|
||||||
},
|
},
|
||||||
"useCoins": {
|
"useCoins": {
|
||||||
|
"addedCoinsDescription": "{amount} pièces ajoutées",
|
||||||
"invalidAmountTitle": "Montant invalide",
|
"invalidAmountTitle": "Montant invalide",
|
||||||
"invalidAmountDescription": "Veuillez entrer un nombre positif valide",
|
"invalidAmountDescription": "Veuillez entrer un nombre positif valide",
|
||||||
"successTitle": "Succès",
|
"successTitle": "Succès",
|
||||||
|
|||||||
@@ -423,6 +423,7 @@
|
|||||||
"cancel": "キャンセル"
|
"cancel": "キャンセル"
|
||||||
},
|
},
|
||||||
"useCoins": {
|
"useCoins": {
|
||||||
|
"addedCoinsDescription": "{amount}コインを追加しました",
|
||||||
"invalidAmountTitle": "無効な値です",
|
"invalidAmountTitle": "無効な値です",
|
||||||
"invalidAmountDescription": "有効な正の数を入力してください",
|
"invalidAmountDescription": "有効な正の数を入力してください",
|
||||||
"successTitle": "成功しました",
|
"successTitle": "成功しました",
|
||||||
|
|||||||
@@ -423,6 +423,7 @@
|
|||||||
"cancel": "Отмена"
|
"cancel": "Отмена"
|
||||||
},
|
},
|
||||||
"useCoins": {
|
"useCoins": {
|
||||||
|
"addedCoinsDescription": "Добавлено {amount} монет",
|
||||||
"invalidAmountTitle": "Неверная сумма",
|
"invalidAmountTitle": "Неверная сумма",
|
||||||
"invalidAmountDescription": "Пожалуйста, введите положительное число",
|
"invalidAmountDescription": "Пожалуйста, введите положительное число",
|
||||||
"successTitle": "Успех",
|
"successTitle": "Успех",
|
||||||
|
|||||||
@@ -423,6 +423,7 @@
|
|||||||
"cancel": "取消"
|
"cancel": "取消"
|
||||||
},
|
},
|
||||||
"useCoins": {
|
"useCoins": {
|
||||||
|
"addedCoinsDescription": "已添加 {amount} 个金币",
|
||||||
"invalidAmountTitle": "无效金额",
|
"invalidAmountTitle": "无效金额",
|
||||||
"invalidAmountDescription": "请输入有效的正数",
|
"invalidAmountDescription": "请输入有效的正数",
|
||||||
"successTitle": "成功",
|
"successTitle": "成功",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "habittrove",
|
"name": "habittrove",
|
||||||
"version": "0.2.18",
|
"version": "0.2.19",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev --turbopack",
|
"dev": "next dev --turbopack",
|
||||||
|
|||||||
Reference in New Issue
Block a user