diff --git a/CHANGELOG.md b/CHANGELOG.md
index a6e3b0ca5..c60c62904 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -7,10 +7,52 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
## Unreleased
+### Added
+
+- Set up a notification service for alert and confirmation dialogs
+
+### Changed
+
+- Refactored the dark theme CSS selector
+- Improved the language localization for German (`de`)
+- Upgraded `zone.js` from version `0.14.7` to `0.14.10`
+
+### Fixed
+
+- Removed `read_only: true` from the `docker-compose.yml` file to allow `prisma` to run migrations
+
+## 2.103.0 - 2024-08-10
+
+### Changed
+
+- Improved the color assignment in the chart of the holdings tab on the home page (experimental)
+- Enabled Catalan (`ca`) as an option in the user settings (experimental)
+- Enabled Polish (`pl`) as an option in the user settings (experimental)
+- Improved the language localization for Portuguese (`pt`)
+- Optimized the docker image layers to reduce the image size
+- Updated the binary targets of `debian-openssl` for `prisma`
+- Upgraded `prisma` from version `5.17.0` to `5.18.0`
+
+## 2.102.0 - 2024-08-07
+
+### Added
+
+- Added support to clone an activity from the account detail dialog (experimental)
+- Added support to edit an activity from the account detail dialog (experimental)
+- Added support to clone an activity from the holding detail dialog (experimental)
+- Added support to edit an activity from the holding detail dialog (experimental)
+
### Changed
+- Improved the caching of the benchmarks in the markets overview by returning cached data and recalculating in the background when it expires
+- Improved the language localization for German (`de`)
+- Improved the language localization for Polish (`pl`)
- Upgraded `Nx` from version `19.5.1` to `19.5.6`
+### Fixed
+
+- Fixed the cache flush endpoint response
+
## 2.101.0 - 2024-08-03
### Changed
diff --git a/Dockerfile b/Dockerfile
index 8ca86a308..e6c38f273 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -3,6 +3,14 @@ FROM --platform=$BUILDPLATFORM node:20-slim AS builder
# Build application and add additional files
WORKDIR /ghostfolio
+RUN apt-get update && apt-get install -y --no-install-suggests \
+ g++ \
+ git \
+ make \
+ openssl \
+ python3 \
+ && rm -rf /var/lib/apt/lists/*
+
# Only add basic files without the application itself to avoid rebuilding
# layers when files (package.json etc.) have not changed
COPY ./CHANGELOG.md CHANGELOG.md
@@ -11,13 +19,6 @@ COPY ./package.json package.json
COPY ./package-lock.json package-lock.json
COPY ./prisma/schema.prisma prisma/schema.prisma
-RUN apt-get update && apt-get install -y --no-install-suggests \
- g++ \
- git \
- make \
- openssl \
- python3 \
- && rm -rf /var/lib/apt/lists/*
RUN npm install
# See https://github.com/nrwl/nx/issues/6586 for further details
@@ -58,9 +59,8 @@ RUN apt-get update && apt-get install -y --no-install-suggests \
openssl \
&& rm -rf /var/lib/apt/lists/*
-COPY --from=builder /ghostfolio/dist/apps /ghostfolio/apps
-COPY ./docker/entrypoint.sh /ghostfolio/entrypoint.sh
-RUN chown -R node:node /ghostfolio
+COPY --chown=node:node --from=builder /ghostfolio/dist/apps /ghostfolio/apps
+COPY --chown=node:node ./docker/entrypoint.sh /ghostfolio/entrypoint.sh
WORKDIR /ghostfolio/apps/api
EXPOSE ${PORT:-3333}
USER node
diff --git a/README.md b/README.md
index 6d31b96ba..ecb5276ab 100644
--- a/README.md
+++ b/README.md
@@ -71,7 +71,7 @@ The backend is based on [NestJS](https://nestjs.com) using [PostgreSQL](https://
### Frontend
-The frontend is built with [Angular](https://angular.io) and uses [Angular Material](https://material.angular.io) with utility classes from [Bootstrap](https://getbootstrap.com).
+The frontend is built with [Angular](https://angular.dev) and uses [Angular Material](https://material.angular.io) with utility classes from [Bootstrap](https://getbootstrap.com).
## Self-hosting
diff --git a/apps/api/src/app/benchmark/benchmark.controller.ts b/apps/api/src/app/benchmark/benchmark.controller.ts
index ea9ba8025..66c268b9b 100644
--- a/apps/api/src/app/benchmark/benchmark.controller.ts
+++ b/apps/api/src/app/benchmark/benchmark.controller.ts
@@ -1,8 +1,8 @@
import { HasPermission } from '@ghostfolio/api/decorators/has-permission.decorator';
import { HasPermissionGuard } from '@ghostfolio/api/guards/has-permission.guard';
-import { getInterval } from '@ghostfolio/api/helper/portfolio.helper';
import { TransformDataSourceInRequestInterceptor } from '@ghostfolio/api/interceptors/transform-data-source-in-request/transform-data-source-in-request.interceptor';
import { TransformDataSourceInResponseInterceptor } from '@ghostfolio/api/interceptors/transform-data-source-in-response/transform-data-source-in-response.interceptor';
+import { getIntervalFromDateRange } from '@ghostfolio/common/calculation-helper';
import type {
AssetProfileIdentifier,
BenchmarkMarketDataDetails,
@@ -113,7 +113,7 @@ export class BenchmarkController {
@Param('symbol') symbol: string,
@Query('range') dateRange: DateRange = 'max'
): Promise {
- const { endDate, startDate } = getInterval(
+ const { endDate, startDate } = getIntervalFromDateRange(
dateRange,
new Date(startDateString)
);
diff --git a/apps/api/src/app/benchmark/benchmark.service.ts b/apps/api/src/app/benchmark/benchmark.service.ts
index e9495b44b..710eb0212 100644
--- a/apps/api/src/app/benchmark/benchmark.service.ts
+++ b/apps/api/src/app/benchmark/benchmark.service.ts
@@ -29,15 +29,19 @@ import { Injectable, Logger } from '@nestjs/common';
import { SymbolProfile } from '@prisma/client';
import { Big } from 'big.js';
import {
+ addHours,
differenceInDays,
eachDayOfInterval,
format,
+ isAfter,
isSameDay,
subDays
} from 'date-fns';
import { isNumber, last, uniqBy } from 'lodash';
import ms from 'ms';
+import { BenchmarkValue } from './interfaces/benchmark-value.interface';
+
@Injectable()
export class BenchmarkService {
private readonly CACHE_KEY_BENCHMARKS = 'BENCHMARKS';
@@ -92,99 +96,28 @@ export class BenchmarkService {
enableSharing = false,
useCache = true
} = {}): Promise {
- let benchmarks: BenchmarkResponse['benchmarks'];
-
if (useCache) {
try {
- benchmarks = JSON.parse(
- await this.redisCacheService.get(this.CACHE_KEY_BENCHMARKS)
+ const cachedBenchmarkValue = await this.redisCacheService.get(
+ this.CACHE_KEY_BENCHMARKS
);
- if (benchmarks) {
- return benchmarks;
- }
- } catch {}
- }
-
- const benchmarkAssetProfiles = await this.getBenchmarkAssetProfiles({
- enableSharing
- });
-
- const promisesAllTimeHighs: Promise<{ date: Date; marketPrice: number }>[] =
- [];
- const promisesBenchmarkTrends: Promise<{
- trend50d: BenchmarkTrend;
- trend200d: BenchmarkTrend;
- }>[] = [];
-
- const quotes = await this.dataProviderService.getQuotes({
- items: benchmarkAssetProfiles.map(({ dataSource, symbol }) => {
- return { dataSource, symbol };
- }),
- requestTimeout: ms('30 seconds'),
- useCache: false
- });
-
- for (const { dataSource, symbol } of benchmarkAssetProfiles) {
- promisesAllTimeHighs.push(
- this.marketDataService.getMax({ dataSource, symbol })
- );
- promisesBenchmarkTrends.push(
- this.getBenchmarkTrends({ dataSource, symbol })
- );
- }
-
- const [allTimeHighs, benchmarkTrends] = await Promise.all([
- Promise.all(promisesAllTimeHighs),
- Promise.all(promisesBenchmarkTrends)
- ]);
- let storeInCache = useCache;
+ const { benchmarks, expiration }: BenchmarkValue =
+ JSON.parse(cachedBenchmarkValue);
- benchmarks = allTimeHighs.map((allTimeHigh, index) => {
- const { marketPrice } =
- quotes[benchmarkAssetProfiles[index].symbol] ?? {};
+ Logger.debug('Fetched benchmarks from cache', 'BenchmarkService');
- let performancePercentFromAllTimeHigh = 0;
-
- if (allTimeHigh?.marketPrice && marketPrice) {
- performancePercentFromAllTimeHigh = this.calculateChangeInPercentage(
- allTimeHigh.marketPrice,
- marketPrice
- );
- } else {
- storeInCache = false;
- }
-
- return {
- dataSource: benchmarkAssetProfiles[index].dataSource,
- marketCondition: this.getMarketCondition(
- performancePercentFromAllTimeHigh
- ),
- name: benchmarkAssetProfiles[index].name,
- performances: {
- allTimeHigh: {
- date: allTimeHigh?.date,
- performancePercent:
- performancePercentFromAllTimeHigh >= 0
- ? 0
- : performancePercentFromAllTimeHigh
- }
- },
- symbol: benchmarkAssetProfiles[index].symbol,
- trend50d: benchmarkTrends[index].trend50d,
- trend200d: benchmarkTrends[index].trend200d
- };
- });
+ if (isAfter(new Date(), new Date(expiration))) {
+ this.calculateAndCacheBenchmarks({
+ enableSharing
+ });
+ }
- if (storeInCache) {
- await this.redisCacheService.set(
- this.CACHE_KEY_BENCHMARKS,
- JSON.stringify(benchmarks),
- ms('2 hours') / 1000
- );
+ return benchmarks;
+ } catch {}
}
- return benchmarks;
+ return this.calculateAndCacheBenchmarks({ enableSharing });
}
public async getBenchmarkAssetProfiles({
@@ -422,6 +355,97 @@ export class BenchmarkService {
};
}
+ private async calculateAndCacheBenchmarks({
+ enableSharing = false
+ }): Promise {
+ Logger.debug('Calculate benchmarks', 'BenchmarkService');
+
+ const benchmarkAssetProfiles = await this.getBenchmarkAssetProfiles({
+ enableSharing
+ });
+
+ const promisesAllTimeHighs: Promise<{ date: Date; marketPrice: number }>[] =
+ [];
+ const promisesBenchmarkTrends: Promise<{
+ trend50d: BenchmarkTrend;
+ trend200d: BenchmarkTrend;
+ }>[] = [];
+
+ const quotes = await this.dataProviderService.getQuotes({
+ items: benchmarkAssetProfiles.map(({ dataSource, symbol }) => {
+ return { dataSource, symbol };
+ }),
+ requestTimeout: ms('30 seconds'),
+ useCache: false
+ });
+
+ for (const { dataSource, symbol } of benchmarkAssetProfiles) {
+ promisesAllTimeHighs.push(
+ this.marketDataService.getMax({ dataSource, symbol })
+ );
+ promisesBenchmarkTrends.push(
+ this.getBenchmarkTrends({ dataSource, symbol })
+ );
+ }
+
+ const [allTimeHighs, benchmarkTrends] = await Promise.all([
+ Promise.all(promisesAllTimeHighs),
+ Promise.all(promisesBenchmarkTrends)
+ ]);
+ let storeInCache = true;
+
+ const benchmarks = allTimeHighs.map((allTimeHigh, index) => {
+ const { marketPrice } =
+ quotes[benchmarkAssetProfiles[index].symbol] ?? {};
+
+ let performancePercentFromAllTimeHigh = 0;
+
+ if (allTimeHigh?.marketPrice && marketPrice) {
+ performancePercentFromAllTimeHigh = this.calculateChangeInPercentage(
+ allTimeHigh.marketPrice,
+ marketPrice
+ );
+ } else {
+ storeInCache = false;
+ }
+
+ return {
+ dataSource: benchmarkAssetProfiles[index].dataSource,
+ marketCondition: this.getMarketCondition(
+ performancePercentFromAllTimeHigh
+ ),
+ name: benchmarkAssetProfiles[index].name,
+ performances: {
+ allTimeHigh: {
+ date: allTimeHigh?.date,
+ performancePercent:
+ performancePercentFromAllTimeHigh >= 0
+ ? 0
+ : performancePercentFromAllTimeHigh
+ }
+ },
+ symbol: benchmarkAssetProfiles[index].symbol,
+ trend50d: benchmarkTrends[index].trend50d,
+ trend200d: benchmarkTrends[index].trend200d
+ };
+ });
+
+ if (storeInCache) {
+ const expiration = addHours(new Date(), 2);
+
+ await this.redisCacheService.set(
+ this.CACHE_KEY_BENCHMARKS,
+ JSON.stringify({
+ benchmarks,
+ expiration: expiration.getTime()
+ }),
+ ms('12 hours') / 1000
+ );
+ }
+
+ return benchmarks;
+ }
+
private getMarketCondition(
aPerformanceInPercent: number
): Benchmark['marketCondition'] {
diff --git a/apps/api/src/app/benchmark/interfaces/benchmark-value.interface.ts b/apps/api/src/app/benchmark/interfaces/benchmark-value.interface.ts
new file mode 100644
index 000000000..eda302f90
--- /dev/null
+++ b/apps/api/src/app/benchmark/interfaces/benchmark-value.interface.ts
@@ -0,0 +1,6 @@
+import { BenchmarkResponse } from '@ghostfolio/common/interfaces';
+
+export interface BenchmarkValue {
+ benchmarks: BenchmarkResponse['benchmarks'];
+ expiration: number;
+}
diff --git a/apps/api/src/app/cache/cache.controller.ts b/apps/api/src/app/cache/cache.controller.ts
index edfd16c49..4d34a2eff 100644
--- a/apps/api/src/app/cache/cache.controller.ts
+++ b/apps/api/src/app/cache/cache.controller.ts
@@ -14,6 +14,6 @@ export class CacheController {
@Post('flush')
@UseGuards(AuthGuard('jwt'), HasPermissionGuard)
public async flushCache(): Promise {
- return this.redisCacheService.reset();
+ await this.redisCacheService.reset();
}
}
diff --git a/apps/api/src/app/order/order.controller.ts b/apps/api/src/app/order/order.controller.ts
index f9190d1eb..7a9cf3d17 100644
--- a/apps/api/src/app/order/order.controller.ts
+++ b/apps/api/src/app/order/order.controller.ts
@@ -1,12 +1,12 @@
import { HasPermission } from '@ghostfolio/api/decorators/has-permission.decorator';
import { HasPermissionGuard } from '@ghostfolio/api/guards/has-permission.guard';
-import { getInterval } from '@ghostfolio/api/helper/portfolio.helper';
import { RedactValuesInResponseInterceptor } from '@ghostfolio/api/interceptors/redact-values-in-response/redact-values-in-response.interceptor';
import { TransformDataSourceInRequestInterceptor } from '@ghostfolio/api/interceptors/transform-data-source-in-request/transform-data-source-in-request.interceptor';
import { TransformDataSourceInResponseInterceptor } from '@ghostfolio/api/interceptors/transform-data-source-in-response/transform-data-source-in-response.interceptor';
import { ApiService } from '@ghostfolio/api/services/api/api.service';
import { DataGatheringService } from '@ghostfolio/api/services/data-gathering/data-gathering.service';
import { ImpersonationService } from '@ghostfolio/api/services/impersonation/impersonation.service';
+import { getIntervalFromDateRange } from '@ghostfolio/common/calculation-helper';
import {
DATA_GATHERING_QUEUE_PRIORITY_HIGH,
HEADER_KEY_IMPERSONATION
@@ -36,7 +36,7 @@ import { parseISO } from 'date-fns';
import { StatusCodes, getReasonPhrase } from 'http-status-codes';
import { CreateOrderDto } from './create-order.dto';
-import { Activities } from './interfaces/activities.interface';
+import { Activities, Activity } from './interfaces/activities.interface';
import { OrderService } from './order.service';
import { UpdateOrderDto } from './update-order.dto';
@@ -110,7 +110,7 @@ export class OrderController {
let startDate: Date;
if (dateRange) {
- ({ endDate, startDate } = getInterval(dateRange));
+ ({ endDate, startDate } = getIntervalFromDateRange(dateRange));
}
const filters = this.apiService.buildFiltersFromQueryParams({
@@ -140,6 +140,38 @@ export class OrderController {
return { activities, count };
}
+ @Get(':id')
+ @UseGuards(AuthGuard('jwt'), HasPermissionGuard)
+ @UseInterceptors(RedactValuesInResponseInterceptor)
+ @UseInterceptors(TransformDataSourceInResponseInterceptor)
+ public async getOrderById(
+ @Headers(HEADER_KEY_IMPERSONATION.toLowerCase()) impersonationId,
+ @Param('id') id: string
+ ): Promise {
+ const impersonationUserId =
+ await this.impersonationService.validateImpersonationId(impersonationId);
+ const userCurrency = this.request.user.Settings.settings.baseCurrency;
+
+ const { activities } = await this.orderService.getOrders({
+ userCurrency,
+ userId: impersonationUserId || this.request.user.id,
+ withExcludedAccounts: true
+ });
+
+ const activity = activities.find((activity) => {
+ return activity.id === id;
+ });
+
+ if (!activity) {
+ throw new HttpException(
+ getReasonPhrase(StatusCodes.NOT_FOUND),
+ StatusCodes.NOT_FOUND
+ );
+ }
+
+ return activity;
+ }
+
@HasPermission(permissions.createOrder)
@Post()
@UseGuards(AuthGuard('jwt'), HasPermissionGuard)
diff --git a/apps/api/src/app/portfolio/calculator/portfolio-calculator.ts b/apps/api/src/app/portfolio/calculator/portfolio-calculator.ts
index d6e5d8bd9..4b0e752c0 100644
--- a/apps/api/src/app/portfolio/calculator/portfolio-calculator.ts
+++ b/apps/api/src/app/portfolio/calculator/portfolio-calculator.ts
@@ -4,13 +4,11 @@ import { PortfolioOrder } from '@ghostfolio/api/app/portfolio/interfaces/portfol
import { TransactionPointSymbol } from '@ghostfolio/api/app/portfolio/interfaces/transaction-point-symbol.interface';
import { TransactionPoint } from '@ghostfolio/api/app/portfolio/interfaces/transaction-point.interface';
import { RedisCacheService } from '@ghostfolio/api/app/redis-cache/redis-cache.service';
-import {
- getFactor,
- getInterval
-} from '@ghostfolio/api/helper/portfolio.helper';
+import { getFactor } from '@ghostfolio/api/helper/portfolio.helper';
import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service';
import { ExchangeRateDataService } from '@ghostfolio/api/services/exchange-rate-data/exchange-rate-data.service';
import { IDataGatheringItem } from '@ghostfolio/api/services/interfaces/interfaces';
+import { getIntervalFromDateRange } from '@ghostfolio/common/calculation-helper';
import { MAX_CHART_ITEMS } from '@ghostfolio/common/config';
import {
DATE_FORMAT,
@@ -141,7 +139,7 @@ export abstract class PortfolioCalculator {
this.useCache = false; // TODO: useCache
this.userId = userId;
- const { endDate, startDate } = getInterval(
+ const { endDate, startDate } = getIntervalFromDateRange(
'max',
subDays(dateOfFirstActivity, 1)
);
@@ -371,7 +369,7 @@ export abstract class PortfolioCalculator {
const feeInBaseCurrency = item.fee.mul(
exchangeRatesByCurrency[`${item.currency}${this.currency}`]?.[
lastTransactionPoint.date
- ]
+ ] ?? 1
);
const marketPriceInBaseCurrency = (
@@ -659,7 +657,10 @@ export abstract class PortfolioCalculator {
return [];
}
- const { endDate, startDate } = getInterval(dateRange, this.getStartDate());
+ const { endDate, startDate } = getIntervalFromDateRange(
+ dateRange,
+ this.getStartDate()
+ );
const daysInMarket = differenceInDays(endDate, startDate) + 1;
const step = withDataDecimation
diff --git a/apps/api/src/app/portfolio/portfolio.controller.ts b/apps/api/src/app/portfolio/portfolio.controller.ts
index 819cf13af..2d7c49aff 100644
--- a/apps/api/src/app/portfolio/portfolio.controller.ts
+++ b/apps/api/src/app/portfolio/portfolio.controller.ts
@@ -7,7 +7,6 @@ import {
hasNotDefinedValuesInObject,
nullifyValuesInObject
} from '@ghostfolio/api/helper/object.helper';
-import { getInterval } from '@ghostfolio/api/helper/portfolio.helper';
import { RedactValuesInResponseInterceptor } from '@ghostfolio/api/interceptors/redact-values-in-response/redact-values-in-response.interceptor';
import { TransformDataSourceInRequestInterceptor } from '@ghostfolio/api/interceptors/transform-data-source-in-request/transform-data-source-in-request.interceptor';
import { TransformDataSourceInResponseInterceptor } from '@ghostfolio/api/interceptors/transform-data-source-in-response/transform-data-source-in-response.interceptor';
@@ -15,6 +14,7 @@ import { ApiService } from '@ghostfolio/api/services/api/api.service';
import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service';
import { ExchangeRateDataService } from '@ghostfolio/api/services/exchange-rate-data/exchange-rate-data.service';
import { ImpersonationService } from '@ghostfolio/api/services/impersonation/impersonation.service';
+import { getIntervalFromDateRange } from '@ghostfolio/common/calculation-helper';
import {
DEFAULT_CURRENCY,
HEADER_KEY_IMPERSONATION
@@ -259,7 +259,7 @@ export class PortfolioController {
await this.impersonationService.validateImpersonationId(impersonationId);
const userCurrency = this.request.user.Settings.settings.baseCurrency;
- const { endDate, startDate } = getInterval(dateRange);
+ const { endDate, startDate } = getIntervalFromDateRange(dateRange);
const { activities } = await this.orderService.getOrders({
endDate,
diff --git a/apps/api/src/app/portfolio/portfolio.service.ts b/apps/api/src/app/portfolio/portfolio.service.ts
index 1f21fa728..b0e07d0b2 100644
--- a/apps/api/src/app/portfolio/portfolio.service.ts
+++ b/apps/api/src/app/portfolio/portfolio.service.ts
@@ -4,10 +4,7 @@ import { CashDetails } from '@ghostfolio/api/app/account/interfaces/cash-details
import { Activity } from '@ghostfolio/api/app/order/interfaces/activities.interface';
import { OrderService } from '@ghostfolio/api/app/order/order.service';
import { UserService } from '@ghostfolio/api/app/user/user.service';
-import {
- getFactor,
- getInterval
-} from '@ghostfolio/api/helper/portfolio.helper';
+import { getFactor } from '@ghostfolio/api/helper/portfolio.helper';
import { AccountClusterRiskCurrentInvestment } from '@ghostfolio/api/models/rules/account-cluster-risk/current-investment';
import { AccountClusterRiskSingleAccount } from '@ghostfolio/api/models/rules/account-cluster-risk/single-account';
import { CurrencyClusterRiskBaseCurrencyCurrentInvestment } from '@ghostfolio/api/models/rules/currency-cluster-risk/base-currency-current-investment';
@@ -18,7 +15,10 @@ import { DataProviderService } from '@ghostfolio/api/services/data-provider/data
import { ExchangeRateDataService } from '@ghostfolio/api/services/exchange-rate-data/exchange-rate-data.service';
import { ImpersonationService } from '@ghostfolio/api/services/impersonation/impersonation.service';
import { SymbolProfileService } from '@ghostfolio/api/services/symbol-profile/symbol-profile.service';
-import { getAnnualizedPerformancePercent } from '@ghostfolio/common/calculation-helper';
+import {
+ getAnnualizedPerformancePercent,
+ getIntervalFromDateRange
+} from '@ghostfolio/common/calculation-helper';
import {
DEFAULT_CURRENCY,
EMERGENCY_FUND_TAG_ID,
@@ -72,7 +72,7 @@ import {
parseISO,
set
} from 'date-fns';
-import { isEmpty, uniq, uniqBy } from 'lodash';
+import { isEmpty, last, uniq, uniqBy } from 'lodash';
import { PortfolioCalculator } from './calculator/portfolio-calculator';
import {
@@ -933,7 +933,7 @@ export class PortfolioService {
const userId = await this.getUserId(impersonationId, this.request.user.id);
const user = await this.userService.user({ id: userId });
- const { endDate } = getInterval(dateRange);
+ const { endDate } = getIntervalFromDateRange(dateRange);
const { activities } = await this.orderService.getOrders({
endDate,
@@ -1115,7 +1115,7 @@ export class PortfolioService {
)
);
- const { endDate, startDate } = getInterval(dateRange);
+ const { endDate, startDate } = getIntervalFromDateRange(dateRange);
console.time('------- PortfolioService.getPerformance - 2');
diff --git a/apps/api/src/helper/portfolio.helper.ts b/apps/api/src/helper/portfolio.helper.ts
index 21b111395..6ebe48d3c 100644
--- a/apps/api/src/helper/portfolio.helper.ts
+++ b/apps/api/src/helper/portfolio.helper.ts
@@ -1,17 +1,4 @@
-import { resetHours } from '@ghostfolio/common/helper';
-import { DateRange } from '@ghostfolio/common/types';
-
import { Type as ActivityType } from '@prisma/client';
-import {
- endOfDay,
- max,
- subDays,
- startOfMonth,
- startOfWeek,
- startOfYear,
- subYears,
- endOfYear
-} from 'date-fns';
export function getFactor(activityType: ActivityType) {
let factor: number;
@@ -30,61 +17,3 @@ export function getFactor(activityType: ActivityType) {
return factor;
}
-
-export function getInterval(
- aDateRange: DateRange,
- portfolioStart = new Date(0)
-) {
- let endDate = endOfDay(new Date(Date.now()));
- let startDate = portfolioStart;
-
- switch (aDateRange) {
- case '1d':
- startDate = max([
- startDate,
- subDays(resetHours(new Date(Date.now())), 1)
- ]);
- break;
- case 'mtd':
- startDate = max([
- startDate,
- subDays(startOfMonth(resetHours(new Date(Date.now()))), 1)
- ]);
- break;
- case 'wtd':
- startDate = max([
- startDate,
- subDays(
- startOfWeek(resetHours(new Date(Date.now())), { weekStartsOn: 1 }),
- 1
- )
- ]);
- break;
- case 'ytd':
- startDate = max([
- startDate,
- subDays(startOfYear(resetHours(new Date(Date.now()))), 1)
- ]);
- break;
- case '1y':
- startDate = max([
- startDate,
- subYears(resetHours(new Date(Date.now())), 1)
- ]);
- break;
- case '5y':
- startDate = max([
- startDate,
- subYears(resetHours(new Date(Date.now())), 5)
- ]);
- break;
- case 'max':
- break;
- default:
- // '2024', '2023', '2022', etc.
- endDate = endOfYear(new Date(aDateRange));
- startDate = max([startDate, new Date(aDateRange)]);
- }
-
- return { endDate, startDate };
-}
diff --git a/apps/client/src/app/app.component.scss b/apps/client/src/app/app.component.scss
index a23e94fbb..6037e9639 100644
--- a/apps/client/src/app/app.component.scss
+++ b/apps/client/src/app/app.component.scss
@@ -46,7 +46,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
footer {
background-color: rgba(var(--palette-foreground-text-dark), 0.05);
}
diff --git a/apps/client/src/app/app.component.ts b/apps/client/src/app/app.component.ts
index 4f1464408..ad6e6e808 100644
--- a/apps/client/src/app/app.component.ts
+++ b/apps/client/src/app/app.component.ts
@@ -255,6 +255,10 @@ export class AppComponent implements OnDestroy, OnInit {
colorScheme: this.user?.settings?.colorScheme,
deviceType: this.deviceType,
hasImpersonationId: this.hasImpersonationId,
+ hasPermissionToCreateOrder:
+ !this.hasImpersonationId &&
+ hasPermission(this.user?.permissions, permissions.createOrder) &&
+ !this.user?.settings?.isRestrictedView,
hasPermissionToReportDataGlitch: hasPermission(
this.user?.permissions,
permissions.reportDataGlitch
@@ -262,10 +266,11 @@ export class AppComponent implements OnDestroy, OnInit {
hasPermissionToUpdateOrder:
!this.hasImpersonationId &&
hasPermission(this.user?.permissions, permissions.updateOrder) &&
- !user?.settings?.isRestrictedView,
+ !this.user?.settings?.isRestrictedView,
locale: this.user?.settings?.locale
},
height: this.deviceType === 'mobile' ? '97.5vh' : '80vh',
+ maxWidth: this.deviceType === 'mobile' ? '95vw' : '50rem',
width: this.deviceType === 'mobile' ? '100vw' : '50rem'
});
@@ -292,9 +297,9 @@ export class AppComponent implements OnDestroy, OnInit {
);
if (isDarkTheme) {
- this.document.body.classList.add('is-dark-theme');
+ this.document.body.classList.add('theme-dark');
} else {
- this.document.body.classList.remove('is-dark-theme');
+ this.document.body.classList.remove('theme-dark');
}
this.document
diff --git a/apps/client/src/app/app.module.ts b/apps/client/src/app/app.module.ts
index 9a311ac69..04602dd2e 100644
--- a/apps/client/src/app/app.module.ts
+++ b/apps/client/src/app/app.module.ts
@@ -33,6 +33,7 @@ import { GfSubscriptionInterstitialDialogModule } from './components/subscriptio
import { authInterceptorProviders } from './core/auth.interceptor';
import { httpResponseInterceptorProviders } from './core/http-response.interceptor';
import { LanguageService } from './core/language.service';
+import { GfNotificationModule } from './core/notification/notification.module';
export function NgxStripeFactory(): string {
return environment.stripePublicKey;
@@ -47,6 +48,7 @@ export function NgxStripeFactory(): string {
BrowserModule,
GfHeaderModule,
GfLogoComponent,
+ GfNotificationModule,
GfSubscriptionInterstitialDialogModule,
MarkdownModule.forRoot(),
MatAutocompleteModule,
diff --git a/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.component.ts b/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.component.ts
index 919b9549d..1cec23aba 100644
--- a/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.component.ts
+++ b/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.component.ts
@@ -23,6 +23,7 @@ import {
import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
import { Sort, SortDirection } from '@angular/material/sort';
import { MatTableDataSource } from '@angular/material/table';
+import { Router } from '@angular/router';
import { Big } from 'big.js';
import { format, parseISO } from 'date-fns';
import { isNumber } from 'lodash';
@@ -66,6 +67,7 @@ export class AccountDetailDialog implements OnDestroy, OnInit {
@Inject(MAT_DIALOG_DATA) public data: AccountDetailDialogParams,
private dataService: DataService,
public dialogRef: MatDialogRef,
+ private router: Router,
private userService: UserService
) {
this.userService.stateChanged
@@ -92,6 +94,14 @@ export class AccountDetailDialog implements OnDestroy, OnInit {
this.fetchPortfolioPerformance();
}
+ public onCloneActivity(aActivity: Activity) {
+ this.router.navigate(['/portfolio', 'activities'], {
+ queryParams: { activityId: aActivity.id, createDialog: true }
+ });
+
+ this.dialogRef.close();
+ }
+
public onClose() {
this.dialogRef.close();
}
@@ -147,6 +157,14 @@ export class AccountDetailDialog implements OnDestroy, OnInit {
this.fetchActivities();
}
+ public onUpdateActivity(aActivity: Activity) {
+ this.router.navigate(['/portfolio', 'activities'], {
+ queryParams: { activityId: aActivity.id, editDialog: true }
+ });
+
+ this.dialogRef.close();
+ }
+
private fetchAccount() {
this.dataService
.fetchAccount(this.data.accountId)
diff --git a/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.html b/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.html
index 0f0091ce5..a814a19a8 100644
--- a/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.html
+++ b/apps/client/src/app/components/account-detail-dialog/account-detail-dialog.html
@@ -101,10 +101,17 @@
[hasPermissionToFilter]="false"
[hasPermissionToOpenDetails]="false"
[locale]="user?.settings?.locale"
- [showActions]="false"
+ [showActions]="
+ !data.hasImpersonationId &&
+ data.hasPermissionToCreateOrder &&
+ user?.settings?.isExperimentalFeatures &&
+ !user?.settings?.isRestrictedView
+ "
[sortColumn]="sortColumn"
[sortDirection]="sortDirection"
[totalItems]="totalItems"
+ (activityToClone)="onCloneActivity($event)"
+ (activityToUpdate)="onUpdateActivity($event)"
(export)="onExport()"
(sortChanged)="onSortChanged($event)"
/>
diff --git a/apps/client/src/app/components/account-detail-dialog/interfaces/interfaces.ts b/apps/client/src/app/components/account-detail-dialog/interfaces/interfaces.ts
index 016fc3b7d..9ad6a5ba4 100644
--- a/apps/client/src/app/components/account-detail-dialog/interfaces/interfaces.ts
+++ b/apps/client/src/app/components/account-detail-dialog/interfaces/interfaces.ts
@@ -2,4 +2,5 @@ export interface AccountDetailDialogParams {
accountId: string;
deviceType: string;
hasImpersonationId: boolean;
+ hasPermissionToCreateOrder: boolean;
}
diff --git a/apps/client/src/app/components/accounts-table/accounts-table.component.ts b/apps/client/src/app/components/accounts-table/accounts-table.component.ts
index 702803aa0..d19cd748f 100644
--- a/apps/client/src/app/components/accounts-table/accounts-table.component.ts
+++ b/apps/client/src/app/components/accounts-table/accounts-table.component.ts
@@ -1,3 +1,5 @@
+import { ConfirmationDialogType } from '@ghostfolio/client/core/notification/confirmation-dialog/confirmation-dialog.type';
+import { NotificationService } from '@ghostfolio/client/core/notification/notification.service';
import { getLocale } from '@ghostfolio/common/helper';
import {
@@ -54,7 +56,10 @@ export class AccountsTableComponent implements OnChanges, OnDestroy, OnInit {
private unsubscribeSubject = new Subject();
- public constructor(private router: Router) {}
+ public constructor(
+ private notificationService: NotificationService,
+ private router: Router
+ ) {}
public ngOnInit() {}
@@ -97,13 +102,13 @@ export class AccountsTableComponent implements OnChanges, OnDestroy, OnInit {
}
public onDeleteAccount(aId: string) {
- const confirmation = confirm(
- $localize`Do you really want to delete this account?`
- );
-
- if (confirmation) {
- this.accountDeleted.emit(aId);
- }
+ this.notificationService.confirm({
+ confirmFn: () => {
+ this.accountDeleted.emit(aId);
+ },
+ confirmType: ConfirmationDialogType.Warn,
+ title: $localize`Do you really want to delete this account?`
+ });
}
public onOpenAccountDetailDialog(accountId: string) {
diff --git a/apps/client/src/app/components/header/header.component.scss b/apps/client/src/app/components/header/header.component.scss
index 6a1521795..d73bf1a8a 100644
--- a/apps/client/src/app/components/header/header.component.scss
+++ b/apps/client/src/app/components/header/header.component.scss
@@ -50,7 +50,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.mat-toolbar {
background-color: var(--dark-background);
diff --git a/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.component.ts b/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.component.ts
index 5673cd0c0..64c062c7e 100644
--- a/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.component.ts
+++ b/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.component.ts
@@ -48,6 +48,7 @@ import { MatFormFieldModule } from '@angular/material/form-field';
import { SortDirection } from '@angular/material/sort';
import { MatTableDataSource } from '@angular/material/table';
import { MatTabsModule } from '@angular/material/tabs';
+import { Router } from '@angular/router';
import { Account, Tag } from '@prisma/client';
import { format, isSameMonth, isToday, parseISO } from 'date-fns';
import { NgxSkeletonLoaderModule } from 'ngx-skeleton-loader';
@@ -141,6 +142,7 @@ export class GfHoldingDetailDialogComponent implements OnDestroy, OnInit {
public dialogRef: MatDialogRef,
@Inject(MAT_DIALOG_DATA) public data: HoldingDetailDialogParams,
private formBuilder: FormBuilder,
+ private router: Router,
private userService: UserService
) {}
@@ -424,6 +426,14 @@ export class GfHoldingDetailDialogComponent implements OnDestroy, OnInit {
this.tagInput.nativeElement.value = '';
}
+ public onCloneActivity(aActivity: Activity) {
+ this.router.navigate(['/portfolio', 'activities'], {
+ queryParams: { activityId: aActivity.id, createDialog: true }
+ });
+
+ this.dialogRef.close();
+ }
+
public onClose() {
this.dialogRef.close();
}
@@ -456,6 +466,14 @@ export class GfHoldingDetailDialogComponent implements OnDestroy, OnInit {
);
}
+ public onUpdateActivity(aActivity: Activity) {
+ this.router.navigate(['/portfolio', 'activities'], {
+ queryParams: { activityId: aActivity.id, editDialog: true }
+ });
+
+ this.dialogRef.close();
+ }
+
public ngOnDestroy() {
this.unsubscribeSubject.next();
this.unsubscribeSubject.complete();
diff --git a/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html b/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
index b7474a7a3..04770837a 100644
--- a/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
+++ b/apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
@@ -346,12 +346,19 @@
[hasPermissionToFilter]="false"
[hasPermissionToOpenDetails]="false"
[locale]="data.locale"
- [showActions]="false"
+ [showActions]="
+ !data.hasImpersonationId &&
+ data.hasPermissionToCreateOrder &&
+ user?.settings?.isExperimentalFeatures &&
+ !user?.settings?.isRestrictedView
+ "
[showNameColumn]="false"
[sortColumn]="sortColumn"
[sortDirection]="sortDirection"
[sortDisabled]="true"
[totalItems]="totalItems"
+ (activityToClone)="onCloneActivity($event)"
+ (activityToUpdate)="onUpdateActivity($event)"
(export)="onExport()"
/>
diff --git a/apps/client/src/app/components/holding-detail-dialog/interfaces/interfaces.ts b/apps/client/src/app/components/holding-detail-dialog/interfaces/interfaces.ts
index 8178838ab..cb98ab3a7 100644
--- a/apps/client/src/app/components/holding-detail-dialog/interfaces/interfaces.ts
+++ b/apps/client/src/app/components/holding-detail-dialog/interfaces/interfaces.ts
@@ -8,6 +8,7 @@ export interface HoldingDetailDialogParams {
dataSource: DataSource;
deviceType: string;
hasImpersonationId: boolean;
+ hasPermissionToCreateOrder: boolean;
hasPermissionToReportDataGlitch: boolean;
hasPermissionToUpdateOrder: boolean;
locale: string;
diff --git a/apps/client/src/app/components/home-holdings/home-holdings.html b/apps/client/src/app/components/home-holdings/home-holdings.html
index b3ebe941c..bd9e57bb2 100644
--- a/apps/client/src/app/components/home-holdings/home-holdings.html
+++ b/apps/client/src/app/components/home-holdings/home-holdings.html
@@ -38,6 +38,7 @@
diff --git a/apps/client/src/app/components/investment-chart/investment-chart.component.ts b/apps/client/src/app/components/investment-chart/investment-chart.component.ts
index 429eaae6f..15a4a6f9a 100644
--- a/apps/client/src/app/components/investment-chart/investment-chart.component.ts
+++ b/apps/client/src/app/components/investment-chart/investment-chart.component.ts
@@ -14,7 +14,7 @@ import {
} from '@ghostfolio/common/helper';
import { LineChartItem } from '@ghostfolio/common/interfaces';
import { InvestmentItem } from '@ghostfolio/common/interfaces/investment-item.interface';
-import { ColorScheme, DateRange, GroupBy } from '@ghostfolio/common/types';
+import { ColorScheme, GroupBy } from '@ghostfolio/common/types';
import {
ChangeDetectionStrategy,
@@ -58,7 +58,6 @@ export class InvestmentChartComponent implements OnChanges, OnDestroy {
@Input() isInPercent = false;
@Input() isLoading = false;
@Input() locale = getLocale();
- @Input() range: DateRange = 'max';
@Input() savingsRate = 0;
@ViewChild('chartCanvas') chartCanvas;
diff --git a/apps/client/src/app/components/portfolio-performance/portfolio-performance.component.ts b/apps/client/src/app/components/portfolio-performance/portfolio-performance.component.ts
index 3083184bb..7ca4677b0 100644
--- a/apps/client/src/app/components/portfolio-performance/portfolio-performance.component.ts
+++ b/apps/client/src/app/components/portfolio-performance/portfolio-performance.component.ts
@@ -1,3 +1,4 @@
+import { NotificationService } from '@ghostfolio/client/core/notification/notification.service';
import {
getLocale,
getNumberFormatDecimal,
@@ -39,7 +40,7 @@ export class PortfolioPerformanceComponent implements OnChanges {
@ViewChild('value') value: ElementRef;
- public constructor() {}
+ public constructor(private notificationService: NotificationService) {}
public ngOnChanges() {
this.precision = this.precision >= 0 ? this.precision : 2;
@@ -74,12 +75,15 @@ export class PortfolioPerformanceComponent implements OnChanges {
}
public onShowErrors() {
- const errorMessageParts = [$localize`Market data is delayed for`];
+ const errorMessageParts = [];
for (const error of this.errors) {
errorMessageParts.push(`${error.symbol} (${error.dataSource})`);
}
- alert(errorMessageParts.join('\n'));
+ this.notificationService.alert({
+ message: errorMessageParts.join('
'),
+ title: $localize`Market data is delayed for`
+ });
}
}
diff --git a/apps/client/src/app/components/rule/rule.component.scss b/apps/client/src/app/components/rule/rule.component.scss
index 7246f41de..54ddce823 100644
--- a/apps/client/src/app/components/rule/rule.component.scss
+++ b/apps/client/src/app/components/rule/rule.component.scss
@@ -20,7 +20,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.icon-container {
background-color: rgba(var(--light-primary-text), 0.05);
}
diff --git a/apps/client/src/app/components/toggle/toggle.component.scss b/apps/client/src/app/components/toggle/toggle.component.scss
index 84ca3fd37..b61a02865 100644
--- a/apps/client/src/app/components/toggle/toggle.component.scss
+++ b/apps/client/src/app/components/toggle/toggle.component.scss
@@ -25,7 +25,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.mat-mdc-radio-button {
&.mat-mdc-radio-checked {
background-color: rgba(var(--light-dividers));
diff --git a/apps/client/src/app/components/user-account-access/user-account-access.scss b/apps/client/src/app/components/user-account-access/user-account-access.scss
index 39eb6792e..c19bfd343 100644
--- a/apps/client/src/app/components/user-account-access/user-account-access.scss
+++ b/apps/client/src/app/components/user-account-access/user-account-access.scss
@@ -3,6 +3,6 @@
display: block;
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/components/user-account-membership/user-account-membership.scss b/apps/client/src/app/components/user-account-membership/user-account-membership.scss
index bb296c89b..0b66f6ee9 100644
--- a/apps/client/src/app/components/user-account-membership/user-account-membership.scss
+++ b/apps/client/src/app/components/user-account-membership/user-account-membership.scss
@@ -4,6 +4,6 @@
height: 100%;
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/components/user-account-settings/user-account-settings.html b/apps/client/src/app/components/user-account-settings/user-account-settings.html
index 8b512ce3f..3896bbb46 100644
--- a/apps/client/src/app/components/user-account-settings/user-account-settings.html
+++ b/apps/client/src/app/components/user-account-settings/user-account-settings.html
@@ -73,12 +73,10 @@
Deutsch
English
@if (user?.settings?.isExperimentalFeatures) {
-
+ Català (Community)
}
@if (user?.settings?.isExperimentalFeatures) {
)
@if (user?.settings?.isExperimentalFeatures) {
-
+ Polski (Community)
}
Português (Community;
private shouldReloadSubject = new Subject();
- public constructor() {
+ public constructor(
+ private deviceService: DeviceDetectorService,
+ private notificationService: NotificationService
+ ) {
this.shouldReloadContent$ = this.shouldReloadSubject.asObservable();
+
+ const deviceType = this.deviceService.getDeviceInfo().deviceType;
+
+ this.notificationService.setDialogWidth(
+ deviceType === 'mobile'
+ ? '95vw'
+ : LayoutService.DEFAULT_NOTIFICATION_WIDTH
+ );
+
+ this.notificationService.setDialogMaxWidth(
+ deviceType === 'mobile'
+ ? '95vw'
+ : LayoutService.DEFAULT_NOTIFICATION_MAX_WIDTH
+ );
}
public getShouldReloadSubject() {
diff --git a/apps/client/src/app/core/notification/alert-dialog/alert-dialog.component.ts b/apps/client/src/app/core/notification/alert-dialog/alert-dialog.component.ts
new file mode 100644
index 000000000..65439ec42
--- /dev/null
+++ b/apps/client/src/app/core/notification/alert-dialog/alert-dialog.component.ts
@@ -0,0 +1,27 @@
+import { CommonModule } from '@angular/common';
+import { Component } from '@angular/core';
+import { MatButtonModule } from '@angular/material/button';
+import { MatDialogModule, MatDialogRef } from '@angular/material/dialog';
+
+import { IAlertDialogParams } from './interfaces/interfaces';
+
+@Component({
+ imports: [CommonModule, MatButtonModule, MatDialogModule],
+ selector: 'gf-alert-dialog',
+ standalone: true,
+ styleUrls: ['./alert-dialog.scss'],
+ templateUrl: './alert-dialog.html'
+})
+export class GfAlertDialogComponent {
+ public discardLabel: string;
+ public message: string;
+ public title: string;
+
+ public constructor(public dialogRef: MatDialogRef) {}
+
+ public initialize(aParams: IAlertDialogParams) {
+ this.discardLabel = aParams.discardLabel;
+ this.message = aParams.message;
+ this.title = aParams.title;
+ }
+}
diff --git a/apps/client/src/app/core/notification/alert-dialog/alert-dialog.html b/apps/client/src/app/core/notification/alert-dialog/alert-dialog.html
new file mode 100644
index 000000000..6602078d3
--- /dev/null
+++ b/apps/client/src/app/core/notification/alert-dialog/alert-dialog.html
@@ -0,0 +1,11 @@
+@if (title) {
+
+}
+
+@if (message) {
+
+}
+
+
+
+
diff --git a/apps/client/src/app/core/notification/alert-dialog/alert-dialog.scss b/apps/client/src/app/core/notification/alert-dialog/alert-dialog.scss
new file mode 100644
index 000000000..dc9093b45
--- /dev/null
+++ b/apps/client/src/app/core/notification/alert-dialog/alert-dialog.scss
@@ -0,0 +1,2 @@
+:host {
+}
diff --git a/apps/client/src/app/core/notification/alert-dialog/interfaces/interfaces.ts b/apps/client/src/app/core/notification/alert-dialog/interfaces/interfaces.ts
new file mode 100644
index 000000000..7cff077a7
--- /dev/null
+++ b/apps/client/src/app/core/notification/alert-dialog/interfaces/interfaces.ts
@@ -0,0 +1,6 @@
+export interface IAlertDialogParams {
+ confirmLabel?: string;
+ discardLabel?: string;
+ message?: string;
+ title: string;
+}
diff --git a/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.component.ts b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.component.ts
new file mode 100644
index 000000000..3545d39b7
--- /dev/null
+++ b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.component.ts
@@ -0,0 +1,41 @@
+import { CommonModule } from '@angular/common';
+import { Component, HostListener } from '@angular/core';
+import { MatButtonModule } from '@angular/material/button';
+import { MatDialogModule, MatDialogRef } from '@angular/material/dialog';
+
+import { ConfirmationDialogType } from './confirmation-dialog.type';
+import { IConfirmDialogParams } from './interfaces/interfaces';
+
+@Component({
+ imports: [CommonModule, MatButtonModule, MatDialogModule],
+ selector: 'gf-confirmation-dialog',
+ standalone: true,
+ styleUrls: ['./confirmation-dialog.scss'],
+ templateUrl: './confirmation-dialog.html'
+})
+export class GfConfirmationDialogComponent {
+ public confirmLabel: string;
+ public confirmType: ConfirmationDialogType;
+ public discardLabel: string;
+ public message: string;
+ public title: string;
+
+ public constructor(
+ public dialogRef: MatDialogRef
+ ) {}
+
+ @HostListener('window:keyup', ['$event'])
+ public keyEvent(event: KeyboardEvent) {
+ if (event.key === 'Enter') {
+ this.dialogRef.close('confirm');
+ }
+ }
+
+ public initialize(aParams: IConfirmDialogParams) {
+ this.confirmLabel = aParams.confirmLabel;
+ this.confirmType = aParams.confirmType;
+ this.discardLabel = aParams.discardLabel;
+ this.message = aParams.message;
+ this.title = aParams.title;
+ }
+}
diff --git a/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.html b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.html
new file mode 100644
index 000000000..e9e2b693c
--- /dev/null
+++ b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.html
@@ -0,0 +1,20 @@
+@if (title) {
+
+}
+
+@if (message) {
+
+}
+
+
+
+
+
diff --git a/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.scss b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.scss
new file mode 100644
index 000000000..dc9093b45
--- /dev/null
+++ b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.scss
@@ -0,0 +1,2 @@
+:host {
+}
diff --git a/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.type.ts b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.type.ts
new file mode 100644
index 000000000..1fe1fc7c9
--- /dev/null
+++ b/apps/client/src/app/core/notification/confirmation-dialog/confirmation-dialog.type.ts
@@ -0,0 +1,5 @@
+export enum ConfirmationDialogType {
+ Accent = 'accent',
+ Primary = 'primary',
+ Warn = 'warn'
+}
diff --git a/apps/client/src/app/core/notification/confirmation-dialog/interfaces/interfaces.ts b/apps/client/src/app/core/notification/confirmation-dialog/interfaces/interfaces.ts
new file mode 100644
index 000000000..834988ceb
--- /dev/null
+++ b/apps/client/src/app/core/notification/confirmation-dialog/interfaces/interfaces.ts
@@ -0,0 +1,9 @@
+import { ConfirmationDialogType } from '../confirmation-dialog.type';
+
+export interface IConfirmDialogParams {
+ confirmLabel?: string;
+ confirmType: ConfirmationDialogType;
+ discardLabel?: string;
+ message?: string;
+ title: string;
+}
diff --git a/apps/client/src/app/core/notification/interfaces/interfaces.ts b/apps/client/src/app/core/notification/interfaces/interfaces.ts
new file mode 100644
index 000000000..f5a526c92
--- /dev/null
+++ b/apps/client/src/app/core/notification/interfaces/interfaces.ts
@@ -0,0 +1,19 @@
+import { ConfirmationDialogType } from '../confirmation-dialog/confirmation-dialog.type';
+
+export interface IAlertParams {
+ discardFn?: () => void;
+ discardLabel?: string;
+ message?: string;
+ title: string;
+}
+
+export interface IConfirmParams {
+ confirmFn: () => void;
+ confirmLabel?: string;
+ confirmType?: ConfirmationDialogType;
+ disableClose?: boolean;
+ discardFn?: () => void;
+ discardLabel?: string;
+ message?: string;
+ title: string;
+}
diff --git a/apps/client/src/app/core/notification/notification.module.ts b/apps/client/src/app/core/notification/notification.module.ts
new file mode 100644
index 000000000..542cae928
--- /dev/null
+++ b/apps/client/src/app/core/notification/notification.module.ts
@@ -0,0 +1,18 @@
+import { CommonModule } from '@angular/common';
+import { NgModule } from '@angular/core';
+import { MatDialogModule } from '@angular/material/dialog';
+
+import { GfAlertDialogComponent } from './alert-dialog/alert-dialog.component';
+import { GfConfirmationDialogComponent } from './confirmation-dialog/confirmation-dialog.component';
+import { NotificationService } from './notification.service';
+
+@NgModule({
+ imports: [
+ CommonModule,
+ GfAlertDialogComponent,
+ GfConfirmationDialogComponent,
+ MatDialogModule
+ ],
+ providers: [NotificationService]
+})
+export class GfNotificationModule {}
diff --git a/apps/client/src/app/core/notification/notification.service.ts b/apps/client/src/app/core/notification/notification.service.ts
new file mode 100644
index 000000000..2e7d9de6c
--- /dev/null
+++ b/apps/client/src/app/core/notification/notification.service.ts
@@ -0,0 +1,83 @@
+import { translate } from '@ghostfolio/ui/i18n';
+
+import { Injectable } from '@angular/core';
+import { MatDialog } from '@angular/material/dialog';
+import { isFunction } from 'lodash';
+
+import { GfAlertDialogComponent } from './alert-dialog/alert-dialog.component';
+import { GfConfirmationDialogComponent } from './confirmation-dialog/confirmation-dialog.component';
+import { ConfirmationDialogType } from './confirmation-dialog/confirmation-dialog.type';
+import { IAlertParams, IConfirmParams } from './interfaces/interfaces';
+
+@Injectable()
+export class NotificationService {
+ private dialogMaxWidth: string;
+ private dialogWidth: string;
+
+ public constructor(private matDialog: MatDialog) {}
+
+ public alert(aParams: IAlertParams) {
+ if (!aParams.discardLabel) {
+ aParams.discardLabel = translate('CLOSE');
+ }
+
+ const dialog = this.matDialog.open(GfAlertDialogComponent, {
+ autoFocus: false,
+ maxWidth: this.dialogMaxWidth,
+ width: this.dialogWidth
+ });
+
+ dialog.componentInstance.initialize({
+ discardLabel: aParams.discardLabel,
+ message: aParams.message,
+ title: aParams.title
+ });
+
+ return dialog.afterClosed().subscribe((result) => {
+ if (isFunction(aParams.discardFn)) {
+ aParams.discardFn();
+ }
+ });
+ }
+
+ public confirm(aParams: IConfirmParams) {
+ if (!aParams.confirmLabel) {
+ aParams.confirmLabel = translate('YES');
+ }
+
+ if (!aParams.discardLabel) {
+ aParams.discardLabel = translate('CANCEL');
+ }
+
+ const dialog = this.matDialog.open(GfConfirmationDialogComponent, {
+ autoFocus: false,
+ disableClose: aParams.disableClose || false,
+ maxWidth: this.dialogMaxWidth,
+ width: this.dialogWidth
+ });
+
+ dialog.componentInstance.initialize({
+ confirmLabel: aParams.confirmLabel,
+ confirmType: aParams.confirmType || ConfirmationDialogType.Primary,
+ discardLabel: aParams.discardLabel,
+ message: aParams.message,
+ title: aParams.title
+ });
+
+ return dialog.afterClosed().subscribe((result) => {
+ if (result === 'confirm' && isFunction(aParams.confirmFn)) {
+ aParams.confirmFn();
+ } else if (result === 'discard' && isFunction(aParams.discardFn)) {
+ aParams.discardFn();
+ }
+ });
+ }
+
+ public setDialogMaxWidth(aDialogMaxWidth: string) {
+ this.dialogMaxWidth = aDialogMaxWidth;
+ }
+
+ public setDialogWidth(aDialogWidth: string) {
+ this.dialogWidth = aDialogWidth;
+ }
+}
diff --git a/apps/client/src/app/pages/about/about-page.scss b/apps/client/src/app/pages/about/about-page.scss
index e87d9a05b..b536ec216 100644
--- a/apps/client/src/app/pages/about/about-page.scss
+++ b/apps/client/src/app/pages/about/about-page.scss
@@ -2,6 +2,6 @@
color: rgb(var(--dark-primary-text));
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/about/changelog/changelog-page.scss b/apps/client/src/app/pages/about/changelog/changelog-page.scss
index aff47b345..6f9de3bcc 100644
--- a/apps/client/src/app/pages/about/changelog/changelog-page.scss
+++ b/apps/client/src/app/pages/about/changelog/changelog-page.scss
@@ -35,6 +35,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/about/license/license-page.scss b/apps/client/src/app/pages/about/license/license-page.scss
index 39eb6792e..c19bfd343 100644
--- a/apps/client/src/app/pages/about/license/license-page.scss
+++ b/apps/client/src/app/pages/about/license/license-page.scss
@@ -3,6 +3,6 @@
display: block;
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/about/overview/about-overview-page.scss b/apps/client/src/app/pages/about/overview/about-overview-page.scss
index df6759442..8b9853f3e 100644
--- a/apps/client/src/app/pages/about/overview/about-overview-page.scss
+++ b/apps/client/src/app/pages/about/overview/about-overview-page.scss
@@ -24,7 +24,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
.about-container {
diff --git a/apps/client/src/app/pages/about/privacy-policy/privacy-policy-page.scss b/apps/client/src/app/pages/about/privacy-policy/privacy-policy-page.scss
index e3c73e9f2..b90d23078 100644
--- a/apps/client/src/app/pages/about/privacy-policy/privacy-policy-page.scss
+++ b/apps/client/src/app/pages/about/privacy-policy/privacy-policy-page.scss
@@ -16,6 +16,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/accounts/accounts-page.component.ts b/apps/client/src/app/pages/accounts/accounts-page.component.ts
index 244333243..9f86bf587 100644
--- a/apps/client/src/app/pages/accounts/accounts-page.component.ts
+++ b/apps/client/src/app/pages/accounts/accounts-page.component.ts
@@ -221,7 +221,11 @@ export class AccountsPageComponent implements OnDestroy, OnInit {
data: {
accountId: aAccountId,
deviceType: this.deviceType,
- hasImpersonationId: this.hasImpersonationId
+ hasImpersonationId: this.hasImpersonationId,
+ hasPermissionToCreateOrder:
+ !this.hasImpersonationId &&
+ hasPermission(this.user?.permissions, permissions.createOrder) &&
+ !this.user?.settings?.isRestrictedView
},
height: this.deviceType === 'mobile' ? '97.5vh' : '80vh',
width: this.deviceType === 'mobile' ? '100vw' : '50rem'
diff --git a/apps/client/src/app/pages/admin/admin-page.scss b/apps/client/src/app/pages/admin/admin-page.scss
index e87d9a05b..b536ec216 100644
--- a/apps/client/src/app/pages/admin/admin-page.scss
+++ b/apps/client/src/app/pages/admin/admin-page.scss
@@ -2,6 +2,6 @@
color: rgb(var(--dark-primary-text));
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/blog/blog-page.scss b/apps/client/src/app/pages/blog/blog-page.scss
index 9235ed3c2..e576551f2 100644
--- a/apps/client/src/app/pages/blog/blog-page.scss
+++ b/apps/client/src/app/pages/blog/blog-page.scss
@@ -9,6 +9,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/faq/faq-page.scss b/apps/client/src/app/pages/faq/faq-page.scss
index e87d9a05b..b536ec216 100644
--- a/apps/client/src/app/pages/faq/faq-page.scss
+++ b/apps/client/src/app/pages/faq/faq-page.scss
@@ -2,6 +2,6 @@
color: rgb(var(--dark-primary-text));
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/features/features-page.scss b/apps/client/src/app/pages/features/features-page.scss
index 4a8680714..c3109fdf7 100644
--- a/apps/client/src/app/pages/features/features-page.scss
+++ b/apps/client/src/app/pages/features/features-page.scss
@@ -12,6 +12,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/home/home-page.scss b/apps/client/src/app/pages/home/home-page.scss
index e87d9a05b..b536ec216 100644
--- a/apps/client/src/app/pages/home/home-page.scss
+++ b/apps/client/src/app/pages/home/home-page.scss
@@ -2,6 +2,6 @@
color: rgb(var(--dark-primary-text));
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/landing/landing-page.scss b/apps/client/src/app/pages/landing/landing-page.scss
index 5a3e129ca..4c0c14efd 100644
--- a/apps/client/src/app/pages/landing/landing-page.scss
+++ b/apps/client/src/app/pages/landing/landing-page.scss
@@ -120,7 +120,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.button-container {
.mat-mdc-outlined-button {
background-color: var(--dark-background);
diff --git a/apps/client/src/app/pages/open/open-page.scss b/apps/client/src/app/pages/open/open-page.scss
index e58d9f237..65c9b4e3c 100644
--- a/apps/client/src/app/pages/open/open-page.scss
+++ b/apps/client/src/app/pages/open/open-page.scss
@@ -14,6 +14,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/portfolio/activities/activities-page.component.ts b/apps/client/src/app/pages/portfolio/activities/activities-page.component.ts
index d6209cdf8..7cd89d62f 100644
--- a/apps/client/src/app/pages/portfolio/activities/activities-page.component.ts
+++ b/apps/client/src/app/pages/portfolio/activities/activities-page.component.ts
@@ -16,7 +16,6 @@ import { PageEvent } from '@angular/material/paginator';
import { Sort, SortDirection } from '@angular/material/sort';
import { MatTableDataSource } from '@angular/material/table';
import { ActivatedRoute, Router } from '@angular/router';
-import { Order as OrderModel } from '@prisma/client';
import { format, parseISO } from 'date-fns';
import { DeviceDetectorService } from 'ngx-device-detector';
import { Subject, Subscription } from 'rxjs';
@@ -63,14 +62,24 @@ export class ActivitiesPageComponent implements OnDestroy, OnInit {
.pipe(takeUntil(this.unsubscribeSubject))
.subscribe((params) => {
if (params['createDialog']) {
- this.openCreateActivityDialog();
+ if (params['activityId']) {
+ this.dataService
+ .fetchActivity(params['activityId'])
+ .pipe(takeUntil(this.unsubscribeSubject))
+ .subscribe((activity) => {
+ this.openCreateActivityDialog(activity);
+ });
+ } else {
+ this.openCreateActivityDialog();
+ }
} else if (params['editDialog']) {
- if (this.dataSource && params['activityId']) {
- const activity = this.dataSource.data.find(({ id }) => {
- return id === params['activityId'];
- });
-
- this.openUpdateActivityDialog(activity);
+ if (params['activityId']) {
+ this.dataService
+ .fetchActivity(params['activityId'])
+ .pipe(takeUntil(this.unsubscribeSubject))
+ .subscribe((activity) => {
+ this.openUpdateActivityDialog(activity);
+ });
} else {
this.router.navigate(['.'], { relativeTo: this.route });
}
@@ -242,7 +251,7 @@ export class ActivitiesPageComponent implements OnDestroy, OnInit {
this.fetchActivities();
}
- public onUpdateActivity(aActivity: OrderModel) {
+ public onUpdateActivity(aActivity: Activity) {
this.router.navigate([], {
queryParams: { activityId: aActivity.id, editDialog: true }
});
diff --git a/apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.scss b/apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.scss
index 6b1415dcb..5af305f9c 100644
--- a/apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.scss
+++ b/apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.scss
@@ -18,7 +18,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.mat-mdc-dialog-content {
.mat-datepicker-input {
&.mat-mdc-input-element:disabled {
diff --git a/apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.scss b/apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.scss
index 2dbbdaefc..54aa8c893 100644
--- a/apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.scss
+++ b/apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.scss
@@ -53,7 +53,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.drop-area {
border-color: rgba(
var(--palette-foreground-divider-dark),
diff --git a/apps/client/src/app/pages/portfolio/allocations/allocations-page.component.ts b/apps/client/src/app/pages/portfolio/allocations/allocations-page.component.ts
index 5af3a3099..89b08ad95 100644
--- a/apps/client/src/app/pages/portfolio/allocations/allocations-page.component.ts
+++ b/apps/client/src/app/pages/portfolio/allocations/allocations-page.component.ts
@@ -12,6 +12,7 @@ import {
PortfolioPosition,
User
} from '@ghostfolio/common/interfaces';
+import { hasPermission, permissions } from '@ghostfolio/common/permissions';
import { Market, MarketAdvanced } from '@ghostfolio/common/types';
import { translate } from '@ghostfolio/ui/i18n';
@@ -584,7 +585,11 @@ export class AllocationsPageComponent implements OnDestroy, OnInit {
data: {
accountId: aAccountId,
deviceType: this.deviceType,
- hasImpersonationId: this.hasImpersonationId
+ hasImpersonationId: this.hasImpersonationId,
+ hasPermissionToCreateOrder:
+ !this.hasImpersonationId &&
+ hasPermission(this.user?.permissions, permissions.createOrder) &&
+ !this.user?.settings?.isRestrictedView
},
height: this.deviceType === 'mobile' ? '97.5vh' : '80vh',
width: this.deviceType === 'mobile' ? '100vw' : '50rem'
diff --git a/apps/client/src/app/pages/portfolio/allocations/allocations-page.scss b/apps/client/src/app/pages/portfolio/allocations/allocations-page.scss
index c73ac7fc3..28071385c 100644
--- a/apps/client/src/app/pages/portfolio/allocations/allocations-page.scss
+++ b/apps/client/src/app/pages/portfolio/allocations/allocations-page.scss
@@ -43,7 +43,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.mat-mdc-progress-bar {
::ng-deep {
.mdc-linear-progress__buffer-bar {
diff --git a/apps/client/src/app/pages/portfolio/analysis/analysis-page.html b/apps/client/src/app/pages/portfolio/analysis/analysis-page.html
index 970a89f75..73817bdcb 100644
--- a/apps/client/src/app/pages/portfolio/analysis/analysis-page.html
+++ b/apps/client/src/app/pages/portfolio/analysis/analysis-page.html
@@ -282,7 +282,6 @@
[isInPercent]="hasImpersonationId || user.settings.isRestrictedView"
[isLoading]="isLoadingInvestmentChart"
[locale]="user?.settings?.locale"
- [range]="user?.settings?.dateRange"
/>
@@ -340,7 +339,6 @@
[isInPercent]="hasImpersonationId || user.settings.isRestrictedView"
[isLoading]="isLoadingInvestmentTimelineChart"
[locale]="user?.settings?.locale"
- [range]="user?.settings?.dateRange"
[savingsRate]="savingsRate"
/>
@@ -377,7 +375,6 @@
[isInPercent]="hasImpersonationId || user.settings.isRestrictedView"
[isLoading]="isLoadingDividendTimelineChart"
[locale]="user?.settings?.locale"
- [range]="user?.settings?.dateRange"
/>
diff --git a/apps/client/src/app/pages/portfolio/portfolio-page.scss b/apps/client/src/app/pages/portfolio/portfolio-page.scss
index e87d9a05b..b536ec216 100644
--- a/apps/client/src/app/pages/portfolio/portfolio-page.scss
+++ b/apps/client/src/app/pages/portfolio/portfolio-page.scss
@@ -2,6 +2,6 @@
color: rgb(var(--dark-primary-text));
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/pricing/pricing-page.scss b/apps/client/src/app/pages/pricing/pricing-page.scss
index 86f4b526f..ad8e97b19 100644
--- a/apps/client/src/app/pages/pricing/pricing-page.scss
+++ b/apps/client/src/app/pages/pricing/pricing-page.scss
@@ -26,6 +26,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/public/public-page.scss b/apps/client/src/app/pages/public/public-page.scss
index 9a6909a98..dbea95e24 100644
--- a/apps/client/src/app/pages/public/public-page.scss
+++ b/apps/client/src/app/pages/public/public-page.scss
@@ -17,6 +17,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/register/register-page.scss b/apps/client/src/app/pages/register/register-page.scss
index 2334b58d4..322a2064f 100644
--- a/apps/client/src/app/pages/register/register-page.scss
+++ b/apps/client/src/app/pages/register/register-page.scss
@@ -24,7 +24,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
.button-container {
.mat-mdc-outlined-button {
background-color: var(--dark-background);
diff --git a/apps/client/src/app/pages/resources/personal-finance-tools/personal-finance-tools-page.scss b/apps/client/src/app/pages/resources/personal-finance-tools/personal-finance-tools-page.scss
index 00667fe84..90a5a41dc 100644
--- a/apps/client/src/app/pages/resources/personal-finance-tools/personal-finance-tools-page.scss
+++ b/apps/client/src/app/pages/resources/personal-finance-tools/personal-finance-tools-page.scss
@@ -20,6 +20,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/resources/personal-finance-tools/product-page.scss b/apps/client/src/app/pages/resources/personal-finance-tools/product-page.scss
index 9726e095f..670cde9a6 100644
--- a/apps/client/src/app/pages/resources/personal-finance-tools/product-page.scss
+++ b/apps/client/src/app/pages/resources/personal-finance-tools/product-page.scss
@@ -16,7 +16,7 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
.call-to-action {
diff --git a/apps/client/src/app/pages/resources/resources-page.scss b/apps/client/src/app/pages/resources/resources-page.scss
index 4a8680714..c3109fdf7 100644
--- a/apps/client/src/app/pages/resources/resources-page.scss
+++ b/apps/client/src/app/pages/resources/resources-page.scss
@@ -12,6 +12,6 @@
}
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/user-account/user-account-page.scss b/apps/client/src/app/pages/user-account/user-account-page.scss
index e87d9a05b..b536ec216 100644
--- a/apps/client/src/app/pages/user-account/user-account-page.scss
+++ b/apps/client/src/app/pages/user-account/user-account-page.scss
@@ -2,6 +2,6 @@
color: rgb(var(--dark-primary-text));
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/pages/zen/zen-page.scss b/apps/client/src/app/pages/zen/zen-page.scss
index e87d9a05b..b536ec216 100644
--- a/apps/client/src/app/pages/zen/zen-page.scss
+++ b/apps/client/src/app/pages/zen/zen-page.scss
@@ -2,6 +2,6 @@
color: rgb(var(--dark-primary-text));
}
-:host-context(.is-dark-theme) {
+:host-context(.theme-dark) {
color: rgb(var(--light-primary-text));
}
diff --git a/apps/client/src/app/services/data.service.ts b/apps/client/src/app/services/data.service.ts
index e74c3f74e..6de3d327d 100644
--- a/apps/client/src/app/services/data.service.ts
+++ b/apps/client/src/app/services/data.service.ts
@@ -4,7 +4,10 @@ import { CreateAccountDto } from '@ghostfolio/api/app/account/create-account.dto
import { TransferBalanceDto } from '@ghostfolio/api/app/account/transfer-balance.dto';
import { UpdateAccountDto } from '@ghostfolio/api/app/account/update-account.dto';
import { CreateOrderDto } from '@ghostfolio/api/app/order/create-order.dto';
-import { Activities } from '@ghostfolio/api/app/order/interfaces/activities.interface';
+import {
+ Activities,
+ Activity
+} from '@ghostfolio/api/app/order/interfaces/activities.interface';
import { UpdateOrderDto } from '@ghostfolio/api/app/order/update-order.dto';
import { PortfolioHoldingDetail } from '@ghostfolio/api/app/portfolio/interfaces/portfolio-holding-detail.interface';
import { LookupItem } from '@ghostfolio/api/app/symbol/interfaces/lookup-item.interface';
@@ -212,6 +215,17 @@ export class DataService {
);
}
+ public fetchActivity(aActivityId: string) {
+ return this.http.get(`/api/v1/order/${aActivityId}`).pipe(
+ map((activity) => {
+ activity.createdAt = parseISO((activity.createdAt));
+ activity.date = parseISO((activity.date));
+
+ return activity;
+ })
+ );
+ }
+
public fetchDividends({
filters,
groupBy = 'month',
diff --git a/apps/client/src/locales/messages.ca.xlf b/apps/client/src/locales/messages.ca.xlf
index 7ff136408..a0c08c862 100644
--- a/apps/client/src/locales/messages.ca.xlf
+++ b/apps/client/src/locales/messages.ca.xlf
@@ -278,7 +278,7 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 186
+ 202
@@ -302,31 +302,39 @@
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 85
+ 77
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 90
+ 83
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 94
+ 88
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 98
+ 92
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 102
+ 96
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 114
+ 100
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 118
+ 105
+
+
+ apps/client/src/app/components/user-account-settings/user-account-settings.html
+ 110
+
+
+ apps/client/src/app/components/user-account-settings/user-account-settings.html
+ 114
apps/client/src/app/pages/features/features-page.html
@@ -335,7 +343,7 @@
The risk of loss in trading can be substantial. It is not advisable to invest money you may need in the short term.
- El risc d'assumir pèrdues en les inversions és substancial. No és recomanable invertir diners que pugui necessitar a curt termini.
+ El risc d’assumir pèrdues en les inversions és substancial. No és recomanable invertir diners que pugui necessitar a curt termini.
apps/client/src/app/app.component.html
199
@@ -422,7 +430,7 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 23
+ 24
@@ -550,7 +558,7 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 24
+ 25
@@ -734,7 +742,7 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 26
+ 27
apps/client/src/app/pages/resources/resources-page.component.ts
@@ -926,7 +934,7 @@
Balanç de Caixa
apps/client/src/app/components/account-detail-dialog/account-detail-dialog.html
- 115
+ 122
@@ -978,7 +986,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 137
+ 138
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -1022,7 +1030,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 143
+ 144
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -1046,31 +1054,31 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 203
+ 204
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 206
+ 207
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 209
+ 210
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 273
+ 274
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 276
+ 277
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 279
+ 280
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 282
+ 283
libs/ui/src/lib/account-balances/account-balances.component.html
@@ -1106,7 +1114,7 @@
apps/client/src/app/components/admin-overview/admin-overview.html
- 83
+ 78
apps/client/src/app/components/admin-platform/admin-platform.component.html
@@ -1138,11 +1146,11 @@
apps/client/src/app/components/admin-overview/admin-overview.html
- 93
+ 88
apps/client/src/app/components/admin-overview/admin-overview.html
- 210
+ 205
apps/client/src/app/components/admin-platform/admin-platform.component.html
@@ -1191,7 +1199,7 @@
Asset Profile
- Perfil d'Actiu
+ Perfil d’Actiu
apps/client/src/app/components/admin-jobs/admin-jobs.html
35
@@ -1246,7 +1254,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 153
+ 154
@@ -1346,7 +1354,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 159
+ 160
libs/ui/src/lib/account-balances/account-balances.component.html
@@ -1406,7 +1414,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 426
+ 427
apps/client/src/app/pages/register/show-access-token-dialog/show-access-token-dialog.html
@@ -1446,7 +1454,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 433
+ 434
@@ -1486,12 +1494,12 @@
Filtra per...
apps/client/src/app/components/admin-market-data/admin-market-data.component.ts
- 322
+ 328
Asset Class
- Classe d'Actiu
+ Classe d’Actiu
apps/client/src/app/components/admin-market-data/admin-market-data.html
86
@@ -1510,12 +1518,12 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 353
+ 354
Asset Sub Class
- Subclasse d'Actiu
+ Subclasse d’Actiu
apps/client/src/app/components/admin-market-data/admin-market-data.html
95
@@ -1534,7 +1542,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 369
+ 370
@@ -1559,7 +1567,7 @@
Activities Count
- Nombre d'Activitats
+ Nombre d’Activitats
apps/client/src/app/components/admin-market-data/admin-market-data.html
113
@@ -1631,7 +1639,7 @@
Do you really want to delete this asset profile?
- Realment vol eliminar el perfil d'aquest actiu?
+ Realment vol eliminar el perfil d’aquest actiu?
apps/client/src/app/components/admin-market-data/admin-market-data.service.ts
18
@@ -1642,23 +1650,23 @@
Realment vol eliminar aquests perfils?
apps/client/src/app/components/admin-market-data/admin-market-data.service.ts
- 34
+ 36
Oops! Could not delete profiles.
- Oooh! No s'han pogut eliminar els perfils
+ Oooh! No s’han pogut eliminar els perfils
apps/client/src/app/components/admin-market-data/admin-market-data.service.ts
- 45
+ 49
Oops! Could not parse historical data.
- Oooh! No s'han pogut recopilar les dades históriques.
+ Oooh! No s’han pogut recopilar les dades históriques.
apps/client/src/app/components/admin-market-data/asset-profile-dialog/asset-profile-dialog.component.ts
- 232
+ 235
@@ -1666,7 +1674,7 @@
El preu de mercat actual és
apps/client/src/app/components/admin-market-data/asset-profile-dialog/asset-profile-dialog.component.ts
- 336
+ 339
@@ -1826,12 +1834,12 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 338
+ 339
Add Asset Profile
- Afegeix el Perfil de l'Actiu
+ Afegeix el Perfil de l’Actiu
apps/client/src/app/components/admin-market-data/create-asset-profile-dialog/create-asset-profile-dialog.html
7
@@ -1862,7 +1870,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 123
+ 124
@@ -1931,7 +1939,7 @@
User Count
- Número d'Usuaris
+ Número d’Usuaris
apps/client/src/app/components/admin-overview/admin-overview.html
13
@@ -1939,10 +1947,10 @@
Activity Count
- Número d'Activitats
+ Número d’Activitats
apps/client/src/app/components/admin-overview/admin-overview.html
- 23
+ 19
@@ -1950,7 +1958,7 @@
per Usuari
apps/client/src/app/components/admin-overview/admin-overview.html
- 33
+ 28
@@ -1958,7 +1966,7 @@
Tipus de Canvi
apps/client/src/app/components/admin-overview/admin-overview.html
- 39
+ 34
@@ -1966,7 +1974,7 @@
Afegir Divisa
apps/client/src/app/components/admin-overview/admin-overview.html
- 109
+ 104
@@ -1974,7 +1982,7 @@
Registrar Usuari
apps/client/src/app/components/admin-overview/admin-overview.html
- 115
+ 110
@@ -1982,7 +1990,7 @@
Mode Només Lecutra
apps/client/src/app/components/admin-overview/admin-overview.html
- 129
+ 124
@@ -1990,7 +1998,7 @@
Recollida de Dades
apps/client/src/app/components/admin-overview/admin-overview.html
- 141
+ 136
@@ -1998,7 +2006,7 @@
Missatge del Sistema
apps/client/src/app/components/admin-overview/admin-overview.html
- 153
+ 148
@@ -2006,7 +2014,7 @@
Estableix el Missatge
apps/client/src/app/components/admin-overview/admin-overview.html
- 175
+ 170
@@ -2014,7 +2022,7 @@
Coupons
apps/client/src/app/components/admin-overview/admin-overview.html
- 183
+ 178
@@ -2022,7 +2030,7 @@
Afegir
apps/client/src/app/components/admin-overview/admin-overview.html
- 243
+ 238
libs/ui/src/lib/account-balances/account-balances.component.html
@@ -2034,7 +2042,7 @@
Ordre
apps/client/src/app/components/admin-overview/admin-overview.html
- 251
+ 246
@@ -2042,7 +2050,7 @@
Depurar el Cache
apps/client/src/app/components/admin-overview/admin-overview.html
- 255
+ 250
@@ -2074,7 +2082,7 @@
apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
- 361
+ 368
apps/client/src/app/pages/accounts/accounts-page.html
@@ -2126,11 +2134,15 @@
apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
- 381
+ 393
+
+
+ apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
+ 430
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 382
+ 383
libs/ui/src/lib/assistant/assistant.html
@@ -2268,6 +2280,10 @@
apps/client/src/app/pages/portfolio/portfolio-page-routing.module.ts
41
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 95
+
Benchmark
@@ -2311,7 +2327,7 @@
Admin Control
- Panell d'Administració
+ Panell d’Administració
apps/client/src/app/components/header/header.component.html
68
@@ -2486,7 +2502,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 187
+ 188
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -2522,11 +2538,11 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 200
+ 201
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 270
+ 271
@@ -2563,10 +2579,10 @@
Report Data Glitch
- Informar d'un Problema amb les Dades
+ Informar d’un Problema amb les Dades
apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
- 399
+ 448
@@ -2574,7 +2590,7 @@
en Actiiu
apps/client/src/app/components/home-holdings/home-holdings.component.ts
- 36
+ 38
@@ -2582,7 +2598,7 @@
Finalitzat
apps/client/src/app/components/home-holdings/home-holdings.component.ts
- 37
+ 39
@@ -2606,7 +2622,7 @@
Gestionar Activitats
apps/client/src/app/components/home-holdings/home-holdings.html
- 60
+ 61
@@ -2618,7 +2634,7 @@
libs/ui/src/lib/i18n.ts
- 71
+ 93
@@ -2630,7 +2646,7 @@
libs/ui/src/lib/i18n.ts
- 72
+ 94
@@ -2722,7 +2738,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 10
+ 8
@@ -2738,7 +2754,7 @@
Total Amount
apps/client/src/app/components/investment-chart/investment-chart.component.ts
- 142
+ 141
@@ -2746,7 +2762,7 @@
Savings Rate
apps/client/src/app/components/investment-chart/investment-chart.component.ts
- 214
+ 213
@@ -2758,7 +2774,7 @@
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 255
+ 251
apps/client/src/app/pages/register/show-access-token-dialog/show-access-token-dialog.html
@@ -3234,7 +3250,7 @@
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 228
+ 224
@@ -3390,7 +3406,7 @@
Locale
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 127
+ 123
@@ -3398,7 +3414,7 @@
Date and number format
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 129
+ 125
@@ -3406,7 +3422,7 @@
Appearance
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 152
+ 148
@@ -3414,7 +3430,7 @@
Auto
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 166
+ 162
@@ -3422,7 +3438,7 @@
Light
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 167
+ 163
@@ -3430,7 +3446,7 @@
Dark
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 168
+ 164
@@ -3438,7 +3454,7 @@
Zen Mode
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 177
+ 173
apps/client/src/app/pages/features/features-page.html
@@ -3450,7 +3466,7 @@
Distraction-free experience for turbulent times
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 178
+ 174
@@ -3458,7 +3474,7 @@
Biometric Authentication
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 194
+ 190
@@ -3466,7 +3482,7 @@
Sign in with fingerprint
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 195
+ 191
@@ -3474,7 +3490,7 @@
Experimental Features
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 211
+ 207
@@ -3482,7 +3498,7 @@
Sneak peek at upcoming functionality
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 212
+ 208
@@ -3490,7 +3506,7 @@
Export Data
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 236
+ 232
@@ -3498,7 +3514,7 @@
Danger Zone
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 248
+ 244
@@ -3506,7 +3522,7 @@
Close Account
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 283
+ 279
@@ -3662,7 +3678,7 @@
Oops, cash balance transfer has failed.
apps/client/src/app/pages/accounts/accounts-page.component.ts
- 304
+ 308
@@ -4530,7 +4546,7 @@
Update activity
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 8
+ 10
@@ -4602,7 +4618,7 @@
Update Cash Balance
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 111
+ 112
@@ -4610,11 +4626,11 @@
Unit Price
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 212
+ 213
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 285
+ 286
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -4626,7 +4642,7 @@
Oops! Could not get the historical exchange rate from
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 239
+ 240
@@ -4634,11 +4650,11 @@
Fee
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 305
+ 306
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 329
+ 330
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -4650,7 +4666,7 @@
Oops! Could not get the historical exchange rate from
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 317
+ 318
@@ -4658,7 +4674,7 @@
Import Activities
apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.component.ts
- 44
+ 45
@@ -4690,7 +4706,7 @@
Validating data...
apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.component.ts
- 233
+ 234
@@ -4992,6 +5008,10 @@
apps/client/src/app/pages/portfolio/analysis/analysis-page.component.ts
56
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 89
+
Monthly
@@ -5094,7 +5114,7 @@
Investment Timeline
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 297
+ 296
@@ -5102,7 +5122,7 @@
Current Streak
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 318
+ 317
@@ -5110,7 +5130,7 @@
Longest Streak
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 327
+ 326
@@ -5118,7 +5138,7 @@
Dividend Timeline
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 356
+ 354
@@ -5522,7 +5542,11 @@
Switzerland
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 58
+ 60
+
+
+ libs/ui/src/lib/i18n.ts
+ 86
@@ -5530,7 +5554,11 @@
Global
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 59
+ 61
+
+
+ libs/ui/src/lib/i18n.ts
+ 14
@@ -5602,7 +5630,7 @@
Available in
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 87
+ 103
@@ -5610,35 +5638,35 @@
✅ Yes
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 115
+ 131
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 122
+ 138
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 134
+ 150
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 141
+ 157
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 153
+ 169
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 160
+ 176
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 172
+ 188
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 179
+ 195
@@ -5646,31 +5674,31 @@
❌ No
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 117
+ 133
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 136
+ 152
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 143
+ 159
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 155
+ 171
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 162
+ 178
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 174
+ 190
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 181
+ 197
@@ -5678,7 +5706,7 @@
❌ No
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 124
+ 140
@@ -5686,7 +5714,7 @@
Self-Hosting
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 129
+ 145
@@ -5694,7 +5722,7 @@
Use anonymously
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 148
+ 164
@@ -5702,7 +5730,7 @@
Free Plan
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 167
+ 183
@@ -5710,11 +5738,11 @@
Starting from
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 188
+ 204
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 193
+ 209
@@ -5722,11 +5750,11 @@
year
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 189
+ 205
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 195
+ 211
@@ -5734,7 +5762,7 @@
Notes
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 201
+ 217
@@ -5742,7 +5770,7 @@
Please note that the information provided in the Ghostfolio vs comparison table is based on our independent research and analysis. This website is not affiliated with or any other product mentioned in the comparison. As the landscape of personal finance tools evolves, it is essential to verify any specific details or changes directly from the respective product page. Data needs a refresh? Help us maintain accurate data on GitHub.
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 210
+ 226
@@ -5750,7 +5778,7 @@
Ready to take your investments to the next level?
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 223
+ 239
@@ -5758,7 +5786,7 @@
Effortlessly track, analyze, and visualize your wealth with Ghostfolio.
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 227
+ 243
@@ -5766,7 +5794,7 @@
Get Started
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 232
+ 248
@@ -5774,7 +5802,7 @@
Personal Finance Tools
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 308
+ 266
@@ -6298,7 +6326,7 @@
Grant
libs/ui/src/lib/i18n.ts
- 14
+ 15
@@ -6306,7 +6334,7 @@
Higher Risk
libs/ui/src/lib/i18n.ts
- 15
+ 16
@@ -6314,7 +6342,7 @@
This activity already exists.
libs/ui/src/lib/i18n.ts
- 16
+ 17
@@ -6322,7 +6350,7 @@
Japan
libs/ui/src/lib/i18n.ts
- 17
+ 80
@@ -6358,7 +6386,7 @@
libs/ui/src/lib/portfolio-proportion-chart/portfolio-proportion-chart.component.ts
- 400
+ 403
@@ -6614,7 +6642,7 @@
Extreme Fear
libs/ui/src/lib/i18n.ts
- 69
+ 91
@@ -6622,7 +6650,7 @@
Extreme Greed
libs/ui/src/lib/i18n.ts
- 70
+ 92
@@ -6630,7 +6658,7 @@
Neutral
libs/ui/src/lib/i18n.ts
- 73
+ 95
@@ -6662,11 +6690,11 @@
No data available
libs/ui/src/lib/portfolio-proportion-chart/portfolio-proportion-chart.component.ts
- 402
+ 405
libs/ui/src/lib/portfolio-proportion-chart/portfolio-proportion-chart.component.ts
- 415
+ 418
@@ -6677,6 +6705,262 @@
81
+
+ Alternative
+ Alternative
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 83
+
+
+
+ App
+ App
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 84
+
+
+
+ Budgeting
+ Budgeting
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 85
+
+
+
+ Community
+ Community
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 86
+
+
+
+ Family Office
+ Family Office
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 87
+
+
+
+ Investor
+ Investor
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 90
+
+
+
+ Open Source
+ Open Source
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 91
+
+
+
+ Personal Finance
+ Personal Finance
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 93
+
+
+
+ Privacy
+ Privacy
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 94
+
+
+
+ Software
+ Software
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 96
+
+
+
+ Tool
+ Tool
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 97
+
+
+
+ User Experience
+ User Experience
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 98
+
+
+
+ Wealth
+ Wealth
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 99
+
+
+
+ Wealth Management
+ Wealth Management
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 100
+
+
+
+ Australia
+ Australia
+
+ libs/ui/src/lib/i18n.ts
+ 69
+
+
+
+ Austria
+ Austria
+
+ libs/ui/src/lib/i18n.ts
+ 70
+
+
+
+ Belgium
+ Belgium
+
+ libs/ui/src/lib/i18n.ts
+ 71
+
+
+
+ Bulgaria
+ Bulgaria
+
+ libs/ui/src/lib/i18n.ts
+ 72
+
+
+
+ Canada
+ Canada
+
+ libs/ui/src/lib/i18n.ts
+ 73
+
+
+
+ Czech Republic
+ Czech Republic
+
+ libs/ui/src/lib/i18n.ts
+ 74
+
+
+
+ Finland
+ Finland
+
+ libs/ui/src/lib/i18n.ts
+ 75
+
+
+
+ France
+ France
+
+ libs/ui/src/lib/i18n.ts
+ 76
+
+
+
+ Germany
+ Germany
+
+ libs/ui/src/lib/i18n.ts
+ 77
+
+
+
+ India
+ India
+
+ libs/ui/src/lib/i18n.ts
+ 78
+
+
+
+ Italy
+ Italy
+
+ libs/ui/src/lib/i18n.ts
+ 79
+
+
+
+ Netherlands
+ Netherlands
+
+ libs/ui/src/lib/i18n.ts
+ 81
+
+
+
+ New Zealand
+ New Zealand
+
+ libs/ui/src/lib/i18n.ts
+ 82
+
+
+
+ Poland
+ Poland
+
+ libs/ui/src/lib/i18n.ts
+ 83
+
+
+
+ Romania
+ Romania
+
+ libs/ui/src/lib/i18n.ts
+ 84
+
+
+
+ South Africa
+ South Africa
+
+ libs/ui/src/lib/i18n.ts
+ 85
+
+
+
+ Thailand
+ Thailand
+
+ libs/ui/src/lib/i18n.ts
+ 87
+
+
+
+ United States
+ United States
+
+ libs/ui/src/lib/i18n.ts
+ 88
+
+
about
- about
+ o Ghostfolio
apps/client/src/app/app.component.ts
59
@@ -82,12 +82,12 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 23
+ 24
faq
- faq
+ faq
apps/client/src/app/app.component.ts
66
@@ -119,7 +119,7 @@
features
- features
+ Funkcje
apps/client/src/app/app.component.ts
67
@@ -178,12 +178,12 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 24
+ 25
license
- license
+ licencja
apps/client/src/app/app.component.ts
61
@@ -199,7 +199,7 @@
markets
- markets
+ rynki
apps/client/src/app/app.component.ts
68
@@ -231,7 +231,7 @@
pricing
- pricing
+ cennik
apps/client/src/app/app.component.ts
69
@@ -394,7 +394,7 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 26
+ 27
apps/client/src/app/pages/resources/resources-page.component.ts
@@ -411,7 +411,7 @@
Create Account
- Create Account
+ Utwórz Konto
apps/client/src/app/app.component.html
13
@@ -427,7 +427,7 @@
Personal Finance
- Personal Finance
+ Finanse Osobiste
apps/client/src/app/app.component.html
54
@@ -435,7 +435,7 @@
Markets
- Markets
+ Rynki
apps/client/src/app/app.component.html
58
@@ -491,7 +491,7 @@
Blog
- Blog
+ Blog
apps/client/src/app/app.component.html
70
@@ -579,7 +579,7 @@
Changelog
- Changelog
+ Dziennik Zmian
apps/client/src/app/app.component.html
74
@@ -607,7 +607,7 @@
Frequently Asked Questions (FAQ)
- Frequently Asked Questions (FAQ)
+ Często Zadawane Pytania (FAQ)
apps/client/src/app/app.component.html
80
@@ -631,7 +631,7 @@
Pricing
- Pricing
+ Cennik
apps/client/src/app/app.component.html
94
@@ -650,12 +650,12 @@
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 186
+ 202
Privacy Policy
- Privacy Policy
+ Polityka Prywatności
apps/client/src/app/app.component.html
100
@@ -667,38 +667,46 @@
Community
- Community
+ Społeczność
apps/client/src/app/app.component.html
118
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 85
+ 77
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 90
+ 83
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 94
+ 88
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 98
+ 92
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 102
+ 96
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 114
+ 100
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 118
+ 105
+
+
+ apps/client/src/app/components/user-account-settings/user-account-settings.html
+ 110
+
+
+ apps/client/src/app/components/user-account-settings/user-account-settings.html
+ 114
apps/client/src/app/pages/features/features-page.html
@@ -727,7 +735,7 @@
Grantee
- Grantee
+ Beneficjent
apps/client/src/app/components/access-table/access-table.component.html
11
@@ -755,7 +763,7 @@
Details
- Details
+ Szczegóły
apps/client/src/app/components/access-table/access-table.component.html
33
@@ -771,7 +779,7 @@
Do you really want to revoke this granted access?
- Do you really want to revoke this granted access?
+ Czy na pewno chcesz cofnąć przyznany dostęp?
apps/client/src/app/components/access-table/access-table.component.ts
50
@@ -779,7 +787,7 @@
Cash Balance
- Cash Balance
+ Saldo Gotówkowe
apps/client/src/app/components/account-detail-dialog/account-detail-dialog.html
45
@@ -906,7 +914,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 137
+ 138
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -950,7 +958,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 143
+ 144
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -974,31 +982,31 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 203
+ 204
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 206
+ 207
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 209
+ 210
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 273
+ 274
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 276
+ 277
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 279
+ 280
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 282
+ 283
libs/ui/src/lib/account-balances/account-balances.component.html
@@ -1023,7 +1031,7 @@
Edit
- Edit
+ Edytuj
apps/client/src/app/components/accounts-table/accounts-table.component.html
278
@@ -1034,7 +1042,7 @@
apps/client/src/app/components/admin-overview/admin-overview.html
- 83
+ 78
apps/client/src/app/components/admin-platform/admin-platform.component.html
@@ -1051,7 +1059,7 @@
Delete
- Delete
+ Usuń
apps/client/src/app/components/accounts-table/accounts-table.component.html
288
@@ -1066,11 +1074,11 @@
apps/client/src/app/components/admin-overview/admin-overview.html
- 93
+ 88
apps/client/src/app/components/admin-overview/admin-overview.html
- 210
+ 205
apps/client/src/app/components/admin-platform/admin-platform.component.html
@@ -1154,7 +1162,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 153
+ 154
@@ -1238,7 +1246,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 159
+ 160
libs/ui/src/lib/account-balances/account-balances.component.html
@@ -1251,7 +1259,7 @@
Market Price
- Market Price
+ Cena Rynkowa
apps/client/src/app/components/admin-market-data-detail/market-data-detail-dialog/market-data-detail-dialog.html
26
@@ -1298,7 +1306,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 426
+ 427
apps/client/src/app/pages/register/show-access-token-dialog/show-access-token-dialog.html
@@ -1338,7 +1346,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 433
+ 434
@@ -1378,7 +1386,7 @@
Filter by...
apps/client/src/app/components/admin-market-data/admin-market-data.component.ts
- 322
+ 328
@@ -1402,7 +1410,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 353
+ 354
@@ -1426,7 +1434,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 369
+ 370
@@ -1515,10 +1523,10 @@
Oops! Could not parse historical data.
- Oops! Could not parse historical data.
+ Ups! Nie udało się sparsować danych historycznych.
apps/client/src/app/components/admin-market-data/asset-profile-dialog/asset-profile-dialog.component.ts
- 232
+ 235
@@ -1654,7 +1662,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 338
+ 339
@@ -1690,7 +1698,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 123
+ 124
@@ -1762,7 +1770,7 @@
Activity Count
apps/client/src/app/components/admin-overview/admin-overview.html
- 23
+ 19
@@ -1770,7 +1778,7 @@
per User
apps/client/src/app/components/admin-overview/admin-overview.html
- 33
+ 28
@@ -1778,7 +1786,7 @@
Exchange Rates
apps/client/src/app/components/admin-overview/admin-overview.html
- 39
+ 34
@@ -1786,7 +1794,7 @@
Add Currency
apps/client/src/app/components/admin-overview/admin-overview.html
- 109
+ 104
@@ -1794,7 +1802,7 @@
User Signup
apps/client/src/app/components/admin-overview/admin-overview.html
- 115
+ 110
@@ -1802,7 +1810,7 @@
Read-only Mode
apps/client/src/app/components/admin-overview/admin-overview.html
- 129
+ 124
@@ -1810,7 +1818,7 @@
System Message
apps/client/src/app/components/admin-overview/admin-overview.html
- 153
+ 148
@@ -1818,7 +1826,7 @@
Set Message
apps/client/src/app/components/admin-overview/admin-overview.html
- 175
+ 170
@@ -1826,7 +1834,7 @@
Coupons
apps/client/src/app/components/admin-overview/admin-overview.html
- 183
+ 178
@@ -1834,7 +1842,7 @@
Add
apps/client/src/app/components/admin-overview/admin-overview.html
- 243
+ 238
libs/ui/src/lib/account-balances/account-balances.component.html
@@ -1846,7 +1854,7 @@
Housekeeping
apps/client/src/app/components/admin-overview/admin-overview.html
- 251
+ 246
@@ -1854,7 +1862,7 @@
Flush Cache
apps/client/src/app/components/admin-overview/admin-overview.html
- 255
+ 250
@@ -1902,7 +1910,7 @@
apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
- 361
+ 368
apps/client/src/app/pages/accounts/accounts-page.html
@@ -1954,11 +1962,15 @@
apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
- 381
+ 393
+
+
+ apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
+ 430
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 382
+ 383
libs/ui/src/lib/assistant/assistant.html
@@ -2087,7 +2099,7 @@
Portfolio
- Portfolio
+ Portfel
apps/client/src/app/components/benchmark-comparator/benchmark-comparator.component.ts
116
@@ -2096,6 +2108,10 @@
apps/client/src/app/pages/portfolio/portfolio-page-routing.module.ts
41
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 95
+
Benchmark
@@ -2219,7 +2235,7 @@
Oops! Incorrect Security Token.
- Oops! Incorrect Security Token.
+ Ups! Nieprawidłowy token bezpieczeństwa.
apps/client/src/app/components/header/header.component.ts
243
@@ -2234,7 +2250,7 @@
Manage Activities
apps/client/src/app/components/home-holdings/home-holdings.html
- 60
+ 61
@@ -2246,7 +2262,7 @@
libs/ui/src/lib/i18n.ts
- 71
+ 93
@@ -2258,7 +2274,7 @@
libs/ui/src/lib/i18n.ts
- 72
+ 94
@@ -2271,7 +2287,7 @@
Welcome to Ghostfolio
- Welcome to Ghostfolio
+ Witaj w Ghostfolio
apps/client/src/app/components/home-overview/home-overview.html
7
@@ -2279,7 +2295,7 @@
Ready to take control of your personal finances?
- Ready to take control of your personal finances?
+ Jesteś gotów przejąć kontrolę nad swoimi finansami osobistymi?
apps/client/src/app/components/home-overview/home-overview.html
8
@@ -2287,7 +2303,7 @@
Setup your accounts
- Setup your accounts
+ Skonfiguruj swoje konta
apps/client/src/app/components/home-overview/home-overview.html
15
@@ -2295,7 +2311,7 @@
Get a comprehensive financial overview by adding your bank and brokerage accounts.
- Get a comprehensive financial overview by adding your bank and brokerage accounts.
+ Uzyskaj kompleksowy przegląd finansowy, poprzez dodanie swoich rachunków bankowych i maklerskich.
apps/client/src/app/components/home-overview/home-overview.html
17
@@ -2303,7 +2319,7 @@
Capture your activities
- Capture your activities
+ Rejestruj swoje działania
apps/client/src/app/components/home-overview/home-overview.html
24
@@ -2311,7 +2327,7 @@
Record your investment activities to keep your portfolio up to date.
- Record your investment activities to keep your portfolio up to date.
+ Dokumentuj swoje działania inwestycyjne, aby zapewnić aktualność portfela.
apps/client/src/app/components/home-overview/home-overview.html
26
@@ -2319,7 +2335,7 @@
Monitor and analyze your portfolio
- Monitor and analyze your portfolio
+ Monitoruj i analizuj swój portfel
apps/client/src/app/components/home-overview/home-overview.html
33
@@ -2327,7 +2343,7 @@
Track your progress in real-time with comprehensive analysis and insights.
- Track your progress in real-time with comprehensive analysis and insights.
+ Śledź swój postęp w czasie rzeczywistym dzięki kompleksowym analizom i obserwacjom.
apps/client/src/app/components/home-overview/home-overview.html
35
@@ -2335,7 +2351,7 @@
Setup accounts
- Setup accounts
+ Konfiguracja kont
apps/client/src/app/components/home-overview/home-overview.html
44
@@ -2350,7 +2366,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 10
+ 8
@@ -2366,7 +2382,7 @@
Total Amount
apps/client/src/app/components/investment-chart/investment-chart.component.ts
- 142
+ 141
@@ -2374,7 +2390,7 @@
Savings Rate
apps/client/src/app/components/investment-chart/investment-chart.component.ts
- 214
+ 213
@@ -2386,7 +2402,7 @@
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 255
+ 251
apps/client/src/app/pages/register/show-access-token-dialog/show-access-token-dialog.html
@@ -2646,11 +2662,11 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 200
+ 201
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 270
+ 271
@@ -2706,7 +2722,7 @@
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 187
+ 188
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -2718,12 +2734,12 @@
Report Data Glitch
apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
- 399
+ 448
Are you an ambitious investor who needs the full picture?
- Are you an ambitious investor who needs the full picture?
+ Jesteś ambitnym inwestorem, który potrzebuje pełnego obrazu swojej działalności?
apps/client/src/app/components/subscription-interstitial-dialog/subscription-interstitial-dialog.html
12
@@ -2731,7 +2747,7 @@
Upgrade to Ghostfolio Premium today and gain access to exclusive features to enhance your investment experience:
- Upgrade to Ghostfolio Premium today and gain access to exclusive features to enhance your investment experience:
+ Przejdź na Ghostfolio Premium już dziś i uzyskaj dostęp do ekskluzywnych funkcji, które wzbogacą Twoje doświadczenie inwestycyjne:
apps/client/src/app/components/subscription-interstitial-dialog/subscription-interstitial-dialog.html
15
@@ -2739,7 +2755,7 @@
Portfolio Summary
- Portfolio Summary
+ Podsumowanie Portfela
apps/client/src/app/components/subscription-interstitial-dialog/subscription-interstitial-dialog.html
22
@@ -2791,7 +2807,7 @@
FIRE Calculator
- FIRE Calculator
+ FIRE Kalkulator
apps/client/src/app/components/subscription-interstitial-dialog/subscription-interstitial-dialog.html
34
@@ -2819,7 +2835,7 @@
and more Features...
- and more Features...
+ i więcej Funkcji ...
apps/client/src/app/components/subscription-interstitial-dialog/subscription-interstitial-dialog.html
42
@@ -2835,7 +2851,7 @@
Get the tools to effectively manage your finances and refine your personal investment strategy.
- Get the tools to effectively manage your finances and refine your personal investment strategy.
+ Uzyskaj narzędzia do skutecznego zarządzania swoimi finansami i udoskonal swoją osobistą strategię inwestycyjną.
apps/client/src/app/components/subscription-interstitial-dialog/subscription-interstitial-dialog.html
45
@@ -2843,7 +2859,7 @@
Skip
- Skip
+ Pomiń
apps/client/src/app/components/subscription-interstitial-dialog/subscription-interstitial-dialog.html
52
@@ -2963,7 +2979,7 @@
Could not redeem coupon code
- Could not redeem coupon code
+ Nie udało się zrealizować kodu kuponu
apps/client/src/app/components/user-account-membership/user-account-membership.component.ts
121
@@ -3031,7 +3047,7 @@
Settings
- Settings
+ Ustawienia
apps/client/src/app/components/user-account-settings/user-account-settings.html
2
@@ -3039,7 +3055,7 @@
Presenter View
- Presenter View
+ Widok Prezentera
apps/client/src/app/components/user-account-settings/user-account-settings.html
7
@@ -3055,7 +3071,7 @@
Base Currency
- Base Currency
+ Waluta Bazowa
apps/client/src/app/components/user-account-settings/user-account-settings.html
27
@@ -3063,7 +3079,7 @@
Language
- Language
+ Język
apps/client/src/app/components/user-account-settings/user-account-settings.html
48
@@ -3071,10 +3087,10 @@
Locale
- Locale
+ Ustawienia Regionalne
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 127
+ 123
@@ -3082,47 +3098,47 @@
Date and number format
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 129
+ 125
Appearance
- Appearance
+ Wygląd (tryb)
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 152
+ 148
Auto
- Auto
+ Auto
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 166
+ 162
Light
- Light
+ Jasny
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 167
+ 163
Dark
- Dark
+ Ciemny
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 168
+ 164
Zen Mode
- Zen Mode
+ Tryb Zen
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 177
+ 173
apps/client/src/app/pages/features/features-page.html
@@ -3134,64 +3150,64 @@
Distraction-free experience for turbulent times
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 178
+ 174
Biometric Authentication
- Biometric Authentication
+ Uwierzytelnianie Biometryczne
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 194
+ 190
Sign in with fingerprint
- Sign in with fingerprint
+ Zaloguj się za pomocą linii papilarnych
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 195
+ 191
Experimental Features
- Experimental Features
+ Funkcje Eksperymentalne
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 211
+ 207
Sneak peek at upcoming functionality
- Sneak peek at upcoming functionality
+ Włącz podgląd nadchodzących funkcji
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 212
+ 208
User ID
- User ID
+ ID Użytkownika
apps/client/src/app/components/user-account-access/create-or-update-access-dialog/create-or-update-access-dialog.html
45
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 228
+ 224
Export Data
- Export Data
+ Eksportuj Dane
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 236
+ 232
This feature is currently unavailable.
- This feature is currently unavailable.
+ Ta funkcja jest obecnie niedostępna.
apps/client/src/app/core/http-response.interceptor.ts
53
@@ -3199,7 +3215,7 @@
Please try again later.
- Please try again later.
+ Spróbuj ponownie później.
apps/client/src/app/core/http-response.interceptor.ts
55
@@ -3215,7 +3231,7 @@
Oops! Something went wrong.
- Oops! Something went wrong.
+ Ups! Coś poszło nie tak.
apps/client/src/app/core/http-response.interceptor.ts
78
@@ -3279,7 +3295,7 @@
Privacy Policy
- Privacy Policy
+ Polityka Prywatności
apps/client/src/app/pages/about/about-page.component.ts
62
@@ -3326,7 +3342,7 @@
Oops, cash balance transfer has failed.
apps/client/src/app/pages/accounts/accounts-page.component.ts
- 304
+ 308
@@ -3667,7 +3683,7 @@
Markets
- Markets
+ Rynki
apps/client/src/app/pages/home/home-page-routing.module.ts
38
@@ -4011,7 +4027,7 @@
Are you ready?
- Are you ready?
+ Czy jesteś gotów?
apps/client/src/app/pages/landing/landing-page.html
431
@@ -4027,7 +4043,7 @@
(Last 24 hours)
- (Last 24 hours)
+ (Ostatnie 24 godziny)
apps/client/src/app/pages/open/open-page.html
37
@@ -4035,7 +4051,7 @@
Active Users
- Active Users
+ Aktywni Użytkownicy
apps/client/src/app/pages/open/open-page.html
40
@@ -4047,7 +4063,7 @@
(Last 30 days)
- (Last 30 days)
+ (Ostatnie 30 dni)
apps/client/src/app/pages/open/open-page.html
48
@@ -4059,7 +4075,7 @@
New Users
- New Users
+ Nowi Użytkownicy
apps/client/src/app/pages/open/open-page.html
51
@@ -4083,7 +4099,7 @@
(Last 90 days)
- (Last 90 days)
+ (Ostatnie 90 dni)
apps/client/src/app/pages/open/open-page.html
127
@@ -4122,7 +4138,7 @@
Update activity
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 8
+ 10
@@ -4194,7 +4210,7 @@
Update Cash Balance
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 111
+ 112
@@ -4202,11 +4218,11 @@
Unit Price
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 212
+ 213
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 285
+ 286
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -4215,10 +4231,10 @@
Oops! Could not get the historical exchange rate from
- Oops! Could not get the historical exchange rate from
+ Ups! Nie udało się uzyskać historycznego kursu wymiany z
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 317
+ 318
@@ -4226,11 +4242,11 @@
Fee
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 305
+ 306
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 329
+ 330
libs/ui/src/lib/activities-table/activities-table.component.html
@@ -4239,10 +4255,10 @@
Oops! Could not get the historical exchange rate from
- Oops! Could not get the historical exchange rate from
+ Ups! Nie udało się uzyskać historycznego kursu wymiany z
apps/client/src/app/pages/portfolio/activities/create-or-update-activity-dialog/create-or-update-activity-dialog.html
- 239
+ 240
@@ -4250,7 +4266,7 @@
Import Activities
apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.component.ts
- 44
+ 45
@@ -4282,7 +4298,7 @@
Validating data...
apps/client/src/app/pages/portfolio/activities/import-activities-dialog/import-activities-dialog.component.ts
- 233
+ 234
@@ -4459,7 +4475,7 @@
Developed Markets
- Developed Markets
+ Rynki Rozwinięte
apps/client/src/app/pages/portfolio/allocations/allocations-page.html
222
@@ -4471,7 +4487,7 @@
Emerging Markets
- Emerging Markets
+ Rynki Wschodzące
apps/client/src/app/pages/portfolio/allocations/allocations-page.html
231
@@ -4483,7 +4499,7 @@
Other Markets
- Other Markets
+ Inne Rynki
apps/client/src/app/pages/portfolio/allocations/allocations-page.html
240
@@ -4495,7 +4511,7 @@
No data available
- No data available
+ Brak danych
apps/client/src/app/pages/portfolio/allocations/allocations-page.html
250
@@ -4567,7 +4583,7 @@
Monthly
- Monthly
+ Miesięcznie
apps/client/src/app/pages/portfolio/analysis/analysis-page.component.ts
50
@@ -4575,7 +4591,7 @@
Yearly
- Yearly
+ Corocznie
apps/client/src/app/pages/portfolio/analysis/analysis-page.component.ts
51
@@ -4583,7 +4599,7 @@
Analysis
- Analysis
+ Analiza
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
2
@@ -4618,7 +4634,7 @@
Investment Timeline
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 297
+ 296
@@ -4626,7 +4642,7 @@
Current Streak
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 318
+ 317
@@ -4634,7 +4650,7 @@
Longest Streak
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 327
+ 326
@@ -4642,7 +4658,7 @@
Dividend Timeline
apps/client/src/app/pages/portfolio/analysis/analysis-page.html
- 356
+ 354
@@ -4663,7 +4679,7 @@
Calculator
- Calculator
+ Kalkulator
apps/client/src/app/pages/portfolio/fire/fire-page.html
7
@@ -4671,7 +4687,7 @@
4% Rule
- 4% Rule
+ Zasada 4%
apps/client/src/app/pages/portfolio/fire/fire-page.html
40
@@ -4679,7 +4695,7 @@
Ghostfolio X-ray uses static analysis to identify potential issues and risks in your portfolio.
- Ghostfolio X-ray uses static analysis to identify potential issues and risks in your portfolio.
+ Ghostfolio X-ray wykorzystuje analizę statyczną do identyfikacji potencjalnych problemów i zagrożeń w Twoim portfelu.
apps/client/src/app/pages/portfolio/fire/fire-page.html
111
@@ -4723,7 +4739,7 @@
Pricing
- Pricing
+ Cennik
apps/client/src/app/pages/pricing/pricing-page-routing.module.ts
13
@@ -4731,7 +4747,7 @@
Pricing Plans
- Pricing Plans
+ Plany cenowe
apps/client/src/app/pages/pricing/pricing-page.html
4
@@ -4739,7 +4755,7 @@
Our official Ghostfolio Premium cloud offering is the easiest way to get started. Due to the time it saves, this will be the best option for most people. Revenue is used to cover the costs of the hosting infrastructure and to fund ongoing development.
- Our official Ghostfolio Premium cloud offering is the easiest way to get started. Due to the time it saves, this will be the best option for most people. Revenue is used to cover the costs of the hosting infrastructure and to fund ongoing development.
+ Nasza oficjalna chmurowa usługa Ghostfolio Premium jest najprostszym sposobem by rozpocząć przygodę z Ghostfolio. To najlepsza opcja dla większości osób ze względu na czas, jaki można dzięki niej zaoszczędzić. Uzyskany przychód jest wykorzystywany do pokrycia kosztów infrastruktury hostingowej i finansowania bieżącego rozwoju.
apps/client/src/app/pages/pricing/pricing-page.html
6
@@ -4747,7 +4763,7 @@
If you prefer to run Ghostfolio on your own infrastructure, please find the source code and further instructions on GitHub.
- If you prefer to run Ghostfolio on your own infrastructure, please find the source code and further instructions on GitHub.
+ Jeżeli wolisz uruchomić Ghostfolio na własnej infrastrukturze, możesz znaleźć kod źródłowy i dalsze instrukcje na naszym GitHubie.
apps/client/src/app/pages/pricing/pricing-page.html
26
@@ -4755,7 +4771,7 @@
For tech-savvy investors who prefer to run Ghostfolio on their own infrastructure.
- For tech-savvy investors who prefer to run Ghostfolio on their own infrastructure.
+ Dla inwestorów obeznanych technicznie, którzy wolą uruchomić Ghostfolio na własnej infrastrukturze.
apps/client/src/app/pages/pricing/pricing-page.html
38
@@ -4763,7 +4779,7 @@
Unlimited Transactions
- Unlimited Transactions
+ Nieograniczona Liczba Transakcji
apps/client/src/app/pages/pricing/pricing-page.html
45
@@ -4779,7 +4795,7 @@
Unlimited Accounts
- Unlimited Accounts
+ Nieograniczona Liczba Rachunków
apps/client/src/app/pages/pricing/pricing-page.html
49
@@ -4795,7 +4811,7 @@
Portfolio Performance
- Portfolio Performance
+ Efektywność Portfela
apps/client/src/app/pages/pricing/pricing-page.html
53
@@ -4811,7 +4827,7 @@
Data Import and Export
- Data Import and Export
+ Importowanie i Eksportowanie Danych
apps/client/src/app/pages/pricing/pricing-page.html
73
@@ -4827,7 +4843,7 @@
Community Support
- Community Support
+ Wsparcie Społeczności
apps/client/src/app/pages/pricing/pricing-page.html
90
@@ -4835,7 +4851,7 @@
Self-hosted, update manually.
- Self-hosted, update manually.
+ Samodzielny hosting, aktualizacja ręczna.
apps/client/src/app/pages/pricing/pricing-page.html
94
@@ -4843,7 +4859,7 @@
Free
- Free
+ Bezpłatnie
apps/client/src/app/pages/pricing/pricing-page.html
95
@@ -4855,7 +4871,7 @@
For new investors who are just getting started with trading.
- For new investors who are just getting started with trading.
+ Dla początkujących inwestorów, którzy dopiero zaczynają swoją przygodę z tradingiem.
apps/client/src/app/pages/pricing/pricing-page.html
123
@@ -4863,7 +4879,7 @@
Fully managed Ghostfolio cloud offering.
- Fully managed Ghostfolio cloud offering.
+ W pełni zarządzana oferta Ghostfolio w chmurze.
apps/client/src/app/pages/pricing/pricing-page.html
152
@@ -4875,7 +4891,7 @@
For ambitious investors who need the full picture of their financial assets.
- For ambitious investors who need the full picture of their financial assets.
+ Dla ambitnych inwestorów, którzy potrzebują pełnego obrazu swoich aktywów finansowych.
apps/client/src/app/pages/pricing/pricing-page.html
184
@@ -4883,7 +4899,7 @@
Email and Chat Support
- Email and Chat Support
+ Wsparcie przez E-mail i Czat
apps/client/src/app/pages/pricing/pricing-page.html
240
@@ -4891,7 +4907,7 @@
Renew Plan
- Renew Plan
+ Odnów Plan
apps/client/src/app/components/header/header.component.html
183
@@ -4907,7 +4923,7 @@
One-time payment, no auto-renewal.
- One-time payment, no auto-renewal.
+ Płatność jednorazowa, bez automatycznego odnawiania.
apps/client/src/app/pages/pricing/pricing-page.html
280
@@ -5138,7 +5154,7 @@
Available in
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 87
+ 103
@@ -5146,35 +5162,35 @@
✅ Yes
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 115
+ 131
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 122
+ 138
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 134
+ 150
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 141
+ 157
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 153
+ 169
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 160
+ 176
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 172
+ 188
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 179
+ 195
@@ -5182,31 +5198,31 @@
❌ No
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 117
+ 133
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 136
+ 152
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 143
+ 159
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 155
+ 171
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 162
+ 178
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 174
+ 190
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 181
+ 197
@@ -5214,7 +5230,7 @@
❌ No
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 124
+ 140
@@ -5222,7 +5238,7 @@
Self-Hosting
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 129
+ 145
@@ -5230,7 +5246,7 @@
Use anonymously
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 148
+ 164
@@ -5238,7 +5254,7 @@
Free Plan
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 167
+ 183
@@ -5246,7 +5262,7 @@
Notes
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 201
+ 217
@@ -5254,7 +5270,7 @@
Please note that the information provided in the Ghostfolio vs comparison table is based on our independent research and analysis. This website is not affiliated with or any other product mentioned in the comparison. As the landscape of personal finance tools evolves, it is essential to verify any specific details or changes directly from the respective product page. Data needs a refresh? Help us maintain accurate data on GitHub.
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 210
+ 226
@@ -5262,7 +5278,7 @@
Ready to take your investments to the next level?
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 223
+ 239
@@ -5270,7 +5286,7 @@
Effortlessly track, analyze, and visualize your wealth with Ghostfolio.
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 227
+ 243
@@ -5278,7 +5294,7 @@
Get Started
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 232
+ 248
@@ -5286,7 +5302,7 @@
Personal Finance Tools
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 308
+ 266
@@ -5294,7 +5310,11 @@
Switzerland
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 58
+ 60
+
+
+ libs/ui/src/lib/i18n.ts
+ 86
@@ -5302,7 +5322,11 @@
Global
apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
- 59
+ 61
+
+
+ libs/ui/src/lib/i18n.ts
+ 14
@@ -5682,7 +5706,7 @@
Grant
libs/ui/src/lib/i18n.ts
- 14
+ 15
@@ -5690,7 +5714,7 @@
Higher Risk
libs/ui/src/lib/i18n.ts
- 15
+ 16
@@ -5698,7 +5722,7 @@
This activity already exists.
libs/ui/src/lib/i18n.ts
- 16
+ 17
@@ -5706,7 +5730,7 @@
Japan
libs/ui/src/lib/i18n.ts
- 17
+ 80
@@ -5742,7 +5766,7 @@
libs/ui/src/lib/portfolio-proportion-chart/portfolio-proportion-chart.component.ts
- 400
+ 403
@@ -5939,7 +5963,7 @@
Africa
- Africa
+ Afryka
libs/ui/src/lib/i18n.ts
61
@@ -5947,7 +5971,7 @@
Asia
- Asia
+ Azja
libs/ui/src/lib/i18n.ts
62
@@ -5955,7 +5979,7 @@
Europe
- Europe
+ Europa
libs/ui/src/lib/i18n.ts
63
@@ -5963,7 +5987,7 @@
North America
- North America
+ Ameryka Północna
libs/ui/src/lib/i18n.ts
64
@@ -5971,7 +5995,7 @@
Oceania
- Oceania
+ Oceania
libs/ui/src/lib/i18n.ts
65
@@ -5979,7 +6003,7 @@
South America
- South America
+ Ameryka Południowa
libs/ui/src/lib/i18n.ts
66
@@ -5990,7 +6014,7 @@
Extreme Fear
libs/ui/src/lib/i18n.ts
- 69
+ 91
@@ -5998,7 +6022,7 @@
Extreme Greed
libs/ui/src/lib/i18n.ts
- 70
+ 92
@@ -6006,7 +6030,7 @@
Neutral
libs/ui/src/lib/i18n.ts
- 73
+ 95
@@ -6035,14 +6059,14 @@
No data available
- No data available
+ Brak danych
libs/ui/src/lib/portfolio-proportion-chart/portfolio-proportion-chart.component.ts
- 402
+ 405
libs/ui/src/lib/portfolio-proportion-chart/portfolio-proportion-chart.component.ts
- 415
+ 418
@@ -6066,7 +6090,7 @@
Cash Balances
apps/client/src/app/components/account-detail-dialog/account-detail-dialog.html
- 115
+ 122
@@ -6074,11 +6098,11 @@
Starting from
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 188
+ 204
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 193
+ 209
@@ -6086,16 +6110,16 @@
year
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 189
+ 205
apps/client/src/app/pages/resources/personal-finance-tools/product-page.html
- 195
+ 211
Do you really want to delete this account balance?
- Do you really want to delete this account balance?
+ Czy na pewno chcesz usunąć saldo tego konta?
libs/ui/src/lib/account-balances/account-balances.component.ts
101
@@ -6111,7 +6135,7 @@
If a translation is missing, kindly support us in extending it here.
- If a translation is missing, kindly support us in extending it here.
+ Jeżeli brakuje jakiegoś tłumaczenia, uprzejmie prosimy o wsparcie w jego uzupełnieniu tutaj.
apps/client/src/app/components/user-account-settings/user-account-settings.html
50
@@ -6122,7 +6146,7 @@
The current market price is
apps/client/src/app/components/admin-market-data/asset-profile-dialog/asset-profile-dialog.component.ts
- 336
+ 339
@@ -6167,7 +6191,7 @@
Oops! Could not grant access.
- Oops! Could not grant access.
+ Ups! Nie udało się przyznać dostępu.
apps/client/src/app/components/user-account-access/create-or-update-access-dialog/create-or-update-access-dialog.component.ts
88
@@ -6212,6 +6236,10 @@
apps/client/src/app/pages/portfolio/analysis/analysis-page.component.ts
56
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 89
+
Absolute Asset Performance
@@ -6315,7 +6343,7 @@
Oops! A data provider is experiencing the hiccups.
- Oops! A data provider is experiencing the hiccups.
+ Ups! Dostawca danych zmaga się teraz z drobnymi przeszkodami.
apps/client/src/app/components/portfolio-performance/portfolio-performance.component.html
8
@@ -6331,7 +6359,7 @@
Reset Filters
- Reset Filters
+ Resetuj Filtry
libs/ui/src/lib/assistant/assistant.html
155
@@ -6363,7 +6391,7 @@
Apply Filters
- Apply Filters
+ Zastosuj Filtry
libs/ui/src/lib/assistant/assistant.html
165
@@ -6371,10 +6399,10 @@
Data Gathering
- Data Gathering
+ Gromadzenie Danych
apps/client/src/app/components/admin-overview/admin-overview.html
- 141
+ 136
@@ -6419,7 +6447,7 @@
FAQ
- FAQ
+ FAQ
apps/client/src/app/pages/faq/saas/saas-page-routing.module.ts
13
@@ -6431,7 +6459,7 @@
Oops! It looks like you’re making too many requests. Please slow down a bit.
- Oops! It looks like you’re making too many requests. Please slow down a bit.
+ Ups! Wygląda na to, że wykonujesz zbyt wiele zapytań. Proszę, zwolnij trochę.
apps/client/src/app/core/http-response.interceptor.ts
96
@@ -6450,7 +6478,7 @@
Active
apps/client/src/app/components/home-holdings/home-holdings.component.ts
- 36
+ 38
@@ -6458,7 +6486,7 @@
Closed
apps/client/src/app/components/home-holdings/home-holdings.component.ts
- 37
+ 39
@@ -6471,7 +6499,7 @@
Dividend Yield
- Dividend Yield
+ Dochód z Dywidendy
apps/client/src/app/components/holding-detail-dialog/holding-detail-dialog.html
191
@@ -6487,7 +6515,7 @@
Priority
- Priority
+ Priorytet
apps/client/src/app/components/admin-jobs/admin-jobs.html
62
@@ -6495,7 +6523,7 @@
This action is not allowed.
- This action is not allowed.
+ To działanie jest niedozwolone.
apps/client/src/app/core/http-response.interceptor.ts
61
@@ -6503,7 +6531,7 @@
Liquidity
- Liquidity
+ Płynność środków finansowych
libs/ui/src/lib/i18n.ts
44
@@ -6551,7 +6579,7 @@
Internationalization
- Internationalization
+ Internacjonalizacja
apps/client/src/app/app-routing.module.ts
79
@@ -6559,7 +6587,7 @@
Do you really want to close your Ghostfolio account?
- Do you really want to close your Ghostfolio account?
+ Czy na pewno chcesz zamknąć swoje konto Ghostfolio?
apps/client/src/app/components/user-account-settings/user-account-settings.component.ts
148
@@ -6570,7 +6598,7 @@
Danger Zone
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 248
+ 244
@@ -6578,7 +6606,7 @@
Close Account
apps/client/src/app/components/user-account-settings/user-account-settings.html
- 283
+ 279
@@ -6599,7 +6627,7 @@
Join now or check out the example account
- Join now or check out the example account
+ Dołącz teraz lub sprawdź przykładowe konto
apps/client/src/app/pages/landing/landing-page.html
434
@@ -6607,7 +6635,7 @@
Oops! There was an error setting up biometric authentication.
- Oops! There was an error setting up biometric authentication.
+ Ups! Wystąpił błąd podczas konfigurowania uwierzytelniania biometrycznego.
apps/client/src/app/components/user-account-settings/user-account-settings.component.ts
302
@@ -6615,7 +6643,7 @@
Show more
- Show more
+ Pokaż więcej
libs/ui/src/lib/top-holdings/top-holdings.component.html
81
@@ -6631,7 +6659,7 @@
Delete Profiles
- Delete Profiles
+ Usuń Profile
apps/client/src/app/components/admin-market-data/admin-market-data.html
190
@@ -6639,18 +6667,18 @@
Do you really want to delete these profiles?
- Do you really want to delete these profiles?
+ Czy na pewno chcesz usunąć te profile?
apps/client/src/app/components/admin-market-data/admin-market-data.service.ts
- 34
+ 36
Oops! Could not delete profiles.
- Oops! Could not delete profiles.
+ Ups! Nie udało się usunąć profili.
apps/client/src/app/components/admin-market-data/admin-market-data.service.ts
- 45
+ 49
@@ -6671,12 +6699,268 @@
Would you like to refine your personal investment strategy?
- Would you like to refine your personal investment strategy?
+ Chcesz udoskonalić swoją osobistą strategię inwestycyjną?
apps/client/src/app/pages/public/public-page.html
155
+
+ Alternative
+ Alternative
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 83
+
+
+
+ App
+ App
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 84
+
+
+
+ Budgeting
+ Budgeting
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 85
+
+
+
+ Community
+ Community
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 86
+
+
+
+ Family Office
+ Family Office
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 87
+
+
+
+ Investor
+ Investor
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 90
+
+
+
+ Open Source
+ Open Source
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 91
+
+
+
+ Personal Finance
+ Personal Finance
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 93
+
+
+
+ Privacy
+ Privacy
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 94
+
+
+
+ Software
+ Software
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 96
+
+
+
+ Tool
+ Tool
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 97
+
+
+
+ User Experience
+ User Experience
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 98
+
+
+
+ Wealth
+ Wealth
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 99
+
+
+
+ Wealth Management
+ Wealth Management
+
+ apps/client/src/app/pages/resources/personal-finance-tools/product-page.component.ts
+ 100
+
+
+
+ Australia
+ Australia
+
+ libs/ui/src/lib/i18n.ts
+ 69
+
+
+
+ Austria
+ Austria
+
+ libs/ui/src/lib/i18n.ts
+ 70
+
+
+
+ Belgium
+ Belgium
+
+ libs/ui/src/lib/i18n.ts
+ 71
+
+
+
+ Bulgaria
+ Bulgaria
+
+ libs/ui/src/lib/i18n.ts
+ 72
+
+
+
+ Canada
+ Canada
+
+ libs/ui/src/lib/i18n.ts
+ 73
+
+
+
+ Czech Republic
+ Czech Republic
+
+ libs/ui/src/lib/i18n.ts
+ 74
+
+
+
+ Finland
+ Finland
+
+ libs/ui/src/lib/i18n.ts
+ 75
+
+
+
+ France
+ France
+
+ libs/ui/src/lib/i18n.ts
+ 76
+
+
+
+ Germany
+ Germany
+
+ libs/ui/src/lib/i18n.ts
+ 77
+
+
+
+ India
+ India
+
+ libs/ui/src/lib/i18n.ts
+ 78
+
+
+
+ Italy
+ Italy
+
+ libs/ui/src/lib/i18n.ts
+ 79
+
+
+
+ Netherlands
+ Netherlands
+
+ libs/ui/src/lib/i18n.ts
+ 81
+
+
+
+ New Zealand
+ New Zealand
+
+ libs/ui/src/lib/i18n.ts
+ 82
+
+
+
+ Poland
+ Poland
+
+ libs/ui/src/lib/i18n.ts
+ 83
+
+
+
+ Romania
+ Romania
+
+ libs/ui/src/lib/i18n.ts
+ 84
+
+
+
+ South Africa
+ South Africa
+
+ libs/ui/src/lib/i18n.ts
+ 85
+
+
+
+ Thailand
+ Thailand
+
+ libs/ui/src/lib/i18n.ts
+ 87
+
+
+
+ United States
+ United States
+
+ libs/ui/src/lib/i18n.ts
+ 88
+
+