diff --git a/src/components/PortfolioTable.tsx b/src/components/PortfolioTable.tsx
index d805359..b559607 100644
--- a/src/components/PortfolioTable.tsx
+++ b/src/components/PortfolioTable.tsx
@@ -82,9 +82,9 @@ export default function PortfolioTable() {
The performance of your portfolio is {performance.summary.performancePercentage.toFixed(2)}%
The average (acc.) performance of all positions is {averagePerformance}%
-
The average (p.a.) performance of every year is {performance.summary.performancePerAnnoPerformance.toFixed(2)}%
-
Best p.a.: {performance.summary.bestPerformancePerAnno?.[0]?.percentage?.toFixed(2) || "0.00"}% ({performance.summary.bestPerformancePerAnno?.[0]?.year || "N/A"})
-
Worst p.a.: {performance.summary.worstPerformancePerAnno?.[0]?.percentage?.toFixed(2) || "0.00"}% ({performance.summary.worstPerformancePerAnno?.[0]?.year || "N/A"})
+
The average (p.a.) performance of every year is {(performance.summary.performancePerAnnoPerformance || 0)?.toFixed(2)}%
+
Best p.a.: {(performance.summary.bestPerformancePerAnno?.[0]?.percentage || 0)?.toFixed(2)}% ({performance.summary.bestPerformancePerAnno?.[0]?.year || "N/A"})
+
Worst p.a.: {(performance.summary.worstPerformancePerAnno?.[0]?.percentage || 0)?.toFixed(2)}% ({performance.summary.worstPerformancePerAnno?.[0]?.year || "N/A"})
Note: An average performance of positions doesn't always match your entire portfolio's average,
especially with single investments or investments on different time ranges.
@@ -307,8 +307,7 @@ export default function PortfolioTable() {
groupId,
amount: firstInvestment.amount,
dayOfMonth: firstInvestment.date?.getDate() || 0,
- interval: 30, // You might want to store this in the investment object
- // Add dynamic settings if available
+ interval: 1,
})}
className="p-1 hover:bg-gray-100 dark:hover:bg-slate-700 rounded transition-colors"
>
@@ -349,7 +348,7 @@ export default function PortfolioTable() {
investedAmount: performance.summary.totalInvested.toFixed(2),
investedAtPrice: "",
currentValue: performance.summary.currentValue.toFixed(2),
- performancePercentage: `${performance.summary.performancePercentage.toFixed(2)}% (avg. acc. ${averagePerformance}%) (avg. p.a. ${performance.summary.performancePerAnnoPerformance.toFixed(2)}%)`,
+ performancePercentage: `${performance.summary.performancePercentage.toFixed(2)}% (avg. acc. ${averagePerformance}%) (avg. p.a. ${(performance.summary.performancePerAnnoPerformance || 0).toFixed(2)}%)`,
periodicGroupId: "",
},
{
@@ -411,7 +410,7 @@ export default function PortfolioTable() {
{performance.summary.performancePercentage.toFixed(2)}%
- (avg. acc. {averagePerformance}%)
- - (avg. p.a. {performance.summary.performancePerAnnoPerformance.toFixed(2)}%)
+ - (avg. p.a. {(performance.summary.performancePerAnnoPerformance || 0).toFixed(2)}%)
- (best p.a. {performance.summary.bestPerformancePerAnno?.[0]?.percentage?.toFixed(2) || "0.00"}% {performance.summary.bestPerformancePerAnno?.[0]?.year || "N/A"})
- (worst p.a. {performance.summary.worstPerformancePerAnno?.[0]?.percentage?.toFixed(2) || "0.00"}% {performance.summary.worstPerformancePerAnno?.[0]?.year || "N/A"})
diff --git a/src/components/utils/DateRangePicker.tsx b/src/components/utils/DateRangePicker.tsx
index 98e5aaa..2125107 100644
--- a/src/components/utils/DateRangePicker.tsx
+++ b/src/components/utils/DateRangePicker.tsx
@@ -1,7 +1,9 @@
-import { format, isValid, parseISO } from "date-fns";
import { useRef } from "react";
import { useDebouncedCallback } from "use-debounce";
+import { useLocaleDateFormat } from "../../hooks/useLocalDateFormat";
+import { formatDateToISO, isValidDate } from "../../utils/formatters";
+
interface DateRangePickerProps {
startDate: Date;
endDate: Date;
@@ -17,24 +19,12 @@ export const DateRangePicker = ({
}: DateRangePickerProps) => {
const startDateRef = useRef
(null);
const endDateRef = useRef(null);
-
- const formatDateToISO = (date: Date) => {
- return format(date, 'yyyy-MM-dd');
- };
-
- const isValidDate = (dateString: string) => {
- const parsed = parseISO(dateString);
- return isValid(parsed);
- };
+ const localeDateFormat = useLocaleDateFormat();
const debouncedStartDateChange = useDebouncedCallback(
(dateString: string) => {
if (isValidDate(dateString)) {
- const newDate = new Date(Date.UTC(
- parseISO(dateString).getUTCFullYear(),
- parseISO(dateString).getUTCMonth(),
- parseISO(dateString).getUTCDate()
- ));
+ const newDate = new Date(dateString);
if (newDate.getTime() !== startDate.getTime()) {
onStartDateChange(newDate);
@@ -47,11 +37,7 @@ export const DateRangePicker = ({
const debouncedEndDateChange = useDebouncedCallback(
(dateString: string) => {
if (isValidDate(dateString)) {
- const newDate = new Date(Date.UTC(
- parseISO(dateString).getUTCFullYear(),
- parseISO(dateString).getUTCMonth(),
- parseISO(dateString).getUTCDate()
- ));
+ const newDate = new Date(dateString);
if (newDate.getTime() !== endDate.getTime()) {
onEndDateChange(newDate);
@@ -76,7 +62,9 @@ export const DateRangePicker = ({
return (
diff --git a/src/hooks/useLocalDateFormat.tsx b/src/hooks/useLocalDateFormat.tsx
new file mode 100644
index 0000000..b5951a4
--- /dev/null
+++ b/src/hooks/useLocalDateFormat.tsx
@@ -0,0 +1,20 @@
+import { useMemo } from "react";
+
+export const useLocaleDateFormat = () => {
+ return useMemo(() => {
+ const formatter = new Intl.DateTimeFormat(undefined, {
+ year: 'numeric',
+ month: '2-digit',
+ day: '2-digit',
+ });
+
+ const testDate = new Date(2024, 0, 1);
+ const formattedParts = formatter.formatToParts(testDate);
+
+ const order = formattedParts
+ .filter(part => part.type !== 'literal') // Entferne Trennzeichen
+ .map(part => part.type);
+
+ return order.join('/').toUpperCase().replace(/DAY/g, 'DD').replace(/MONTH/g, 'MM').replace(/YEAR/g, 'YYYY');
+ }, []);
+};
diff --git a/src/utils/calculations/futureProjection.ts b/src/utils/calculations/futureProjection.ts
index 7609c1a..25b465c 100644
--- a/src/utils/calculations/futureProjection.ts
+++ b/src/utils/calculations/futureProjection.ts
@@ -52,6 +52,7 @@ export const calculateFutureProjection = async (
yearsToProject: number,
annualReturnRate: number,
withdrawalPlan?: WithdrawalPlan,
+ startFromZero: boolean = false
): Promise<{
projection: ProjectionData[];
sustainability: SustainabilityAnalysis;
@@ -67,6 +68,7 @@ export const calculateFutureProjection = async (
const periodicInvestments = currentAssets.flatMap(asset => {
const patterns = new Map();
+ // When startFromZero is true, only include periodic investments
asset.investments.forEach(inv => {
if (inv.type === 'periodic' && inv.periodicGroupId) {
if (!patterns.has(inv.periodicGroupId)) {
@@ -115,17 +117,27 @@ export const calculateFutureProjection = async (
// Calculate monthly values
let currentDate = new Date();
- let totalInvested = currentAssets.reduce(
- (sum, asset) => sum + asset.investments.reduce(
- (assetSum, inv) => assetSum + inv.amount, 0
- ), 0
- );
+
+ // Initialize totalInvested based on startFromZero flag
+ let totalInvested = 0;
+ if (!startFromZero) {
+ // Include all investments if not starting from zero
+ totalInvested = currentAssets.reduce(
+ (sum, asset) => sum + asset.investments.reduce(
+ (assetSum, inv) => assetSum + inv.amount, 0
+ ), 0
+ );
+ } else {
+ // Start from zero when startFromZero is true
+ totalInvested = 0;
+ // Don't initialize with any periodic investments - they'll accumulate over time
+ }
let totalWithdrawn = 0;
let yearsToReachTarget = 0;
let targetValue = 0;
let sustainableYears: number | 'infinite' = 'infinite';
- let portfolioValue = totalInvested; // Initialize portfolio value with current investments
+ let portfolioValue = startFromZero ? 0 : totalInvested; // Start from 0 if startFromZero is true
let withdrawalsStarted = false;
let withdrawalStartDate: Date | null = null;
let portfolioDepletionDate: Date | null = null;
@@ -172,11 +184,12 @@ export const calculateFutureProjection = async (
const monthlyInvestment = monthInvestments.reduce(
(sum, inv) => sum + inv.amount, 0
);
+
+ // Always add periodic investments to both totalInvested and portfolioValue
totalInvested += monthlyInvestment;
portfolioValue += monthlyInvestment;
}
-
// Handle withdrawals
let monthlyWithdrawal = 0;
if (withdrawalsStarted && portfolioValue > 0) {
diff --git a/src/utils/calculations/portfolioValue.ts b/src/utils/calculations/portfolioValue.ts
index d7e4482..4a2f5bf 100644
--- a/src/utils/calculations/portfolioValue.ts
+++ b/src/utils/calculations/portfolioValue.ts
@@ -31,7 +31,7 @@ export const calculatePortfolioValue = (assets: Asset[], dateRange: DateRange) =
for (const asset of assets) {
// calculate the invested kapital
for (const investment of asset.investments) {
- if (!isAfter(new Date(investment.date!), currentDate)) {
+ if (!isAfter(new Date(investment.date!), currentDate) && !isSameDay(new Date(investment.date!), currentDate)) {
dayData.invested += investment.amount;
}
}
@@ -75,7 +75,7 @@ export const calculatePortfolioValue = (assets: Asset[], dateRange: DateRange) =
const totalInvested = dayData.invested; // Total invested amount for the day
const totalCurrentValue = dayData.total; // Total current value for the day
- dayData.percentageChange = totalInvested > 0
+ dayData.percentageChange = totalInvested > 0 && totalCurrentValue > 0
? ((totalCurrentValue - totalInvested) / totalInvested) * 100
: 0;
diff --git a/src/utils/formatters.ts b/src/utils/formatters.ts
index 3b0188a..01ed4b1 100644
--- a/src/utils/formatters.ts
+++ b/src/utils/formatters.ts
@@ -1,3 +1,5 @@
+import { formatDate, isValid, parseISO } from "date-fns";
+
export const formatCurrency = (value: number): string => {
return `€${value.toLocaleString('de-DE', {
minimumFractionDigits: 2,
@@ -34,3 +36,6 @@ export const getHexColor = (usedColors: Set, isDarkMode: boolean): strin
// Fallback to random color if all predefined colors are used
return `#${Math.floor(Math.random() * 16777215).toString(16)}`;
};
+
+export const formatDateToISO = (date: Date) => formatDate(date, 'yyyy-MM-dd');
+export const isValidDate = (dateString: string) => isValid(parseISO(dateString));