diff --git a/.gitignore b/.gitignore index 3d70248ba2..df6605d1a7 100644 --- a/.gitignore +++ b/.gitignore @@ -12,4 +12,6 @@ npm-debug.log* yarn-debug.log* yarn-error.log* -package-lock.json \ No newline at end of file +package-lock.json + + diff --git a/Procfile b/Procfile deleted file mode 100644 index dc14c0b63a..0000000000 --- a/Procfile +++ /dev/null @@ -1 +0,0 @@ -web: npm start --prefix backend \ No newline at end of file diff --git a/README.md b/README.md index 31466b54c2..60eb52d3e1 100644 --- a/README.md +++ b/README.md @@ -1,13 +1,11 @@ -# Final Project +# Nanwa - Forestry -Replace this readme with your own information about your project. +A comprehensive forestry management platform providing real-time tree monitoring, data analytics, and environmental impact tracking. The application enables forest managers to monitor tree health, track growth patterns, analyze carbon absorption, and manage forestry investments through an intuitive dashboard interface. -Start by briefly describing the assignment in a sentence or two. Keep it short and to the point. - -## The problem - -Describe how you approached to problem, and what tools and techniques you used to solve it. How did you plan? What technologies did you use? If you had more time, what would be next? +Built with React and modern web technologies, Nanwa offers interactive maps, detailed analytics charts, and comprehensive data export capabilities for sustainable forest management. ## View it live -Every project should be deployed somewhere. Be sure to include the link to the deployed project so that the viewer can click around and see what it's all about. \ No newline at end of file +[Live Demo](https://nanwa.netlify.app/) +[Live API](https://project-final-frontend-4bia.onrender.com) +Login credentials on the login page! diff --git a/backend/.dockerignore b/backend/.dockerignore new file mode 100644 index 0000000000..eeb29cd144 --- /dev/null +++ b/backend/.dockerignore @@ -0,0 +1,14 @@ +node_modules +npm-debug.log +.git +.gitignore +README.md +.env +.env.local +.env.development +.env.production +coverage +.nyc_output +.DS_Store +*.log +logs \ No newline at end of file diff --git a/backend/Dockerfile b/backend/Dockerfile new file mode 100644 index 0000000000..a66c745103 --- /dev/null +++ b/backend/Dockerfile @@ -0,0 +1,32 @@ +# Use the official Node.js LTS image +FROM node:18-alpine + +# Set working directory +WORKDIR /app + +# Copy package files +COPY package*.json ./ + +# Install dependencies +RUN npm ci --only=production + +# Copy source code +COPY . . + +# Create non-root user +RUN addgroup -g 1001 -S nodejs +RUN adduser -S nanwa -u 1001 + +# Change ownership of the app directory +RUN chown -R nanwa:nodejs /app +USER nanwa + +# Expose port +EXPOSE 8080 + +# Health check +HEALTHCHECK --interval=30s --timeout=3s --start-period=5s --retries=3 \ + CMD node healthcheck.js + +# Start the application +CMD ["npm", "start"] \ No newline at end of file diff --git a/backend/babel.config.js b/backend/babel.config.js new file mode 100644 index 0000000000..4c1bd08e24 --- /dev/null +++ b/backend/babel.config.js @@ -0,0 +1,9 @@ +export default { + presets: [ + ['@babel/preset-env', { + targets: { + node: 'current', + }, + }], + ], +}; \ No newline at end of file diff --git a/backend/controllers/authController.js b/backend/controllers/authController.js new file mode 100644 index 0000000000..d9076f6fdd --- /dev/null +++ b/backend/controllers/authController.js @@ -0,0 +1,316 @@ +import { validationResult } from 'express-validator'; +import { User, RefreshToken } from '../models/index.js'; +import { generateToken } from '../utils/jwt.js'; + +// Register new user +export const register = async (req, res) => { + try { + // Check for validation errors + const errors = validationResult(req); + if (!errors.isEmpty()) { + return res.status(400).json({ + success: false, + message: 'Validation failed', + errors: errors.array() + }); + } + + const { email, password, firstName, lastName } = req.body; + + // Check if user already exists + const existingUser = await User.findByEmail(email); + if (existingUser) { + return res.status(409).json({ + success: false, + message: 'User with this email already exists' + }); + } + + // Create new user + const user = new User({ + email, + password, + firstName, + lastName + }); + + await user.save(); + + // Generate JWT token + const token = generateToken(user._id); + + // Generate refresh token + const refreshToken = RefreshToken.generateToken( + user._id, + req.headers['user-agent'], + req.ip || req.connection.remoteAddress + ); + await refreshToken.save(); + + // Update last login + user.lastLogin = new Date(); + await user.save(); + + res.status(201).json({ + success: true, + message: 'User registered successfully', + data: { + token, + refreshToken: refreshToken.token, + user: user.toJSON() + } + }); + } catch (error) { + console.error('Registration error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Login user +export const login = async (req, res) => { + try { + // Check for validation errors + const errors = validationResult(req); + if (!errors.isEmpty()) { + return res.status(400).json({ + success: false, + message: 'Validation failed', + errors: errors.array() + }); + } + + const { email, password } = req.body; + + // Find user by email + const user = await User.findByEmail(email); + if (!user) { + return res.status(401).json({ + success: false, + message: 'Invalid email or password' + }); + } + + // Check if account is active + if (!user.isActive) { + return res.status(401).json({ + success: false, + message: 'Account is deactivated' + }); + } + + // Check password + const isPasswordValid = await user.comparePassword(password); + if (!isPasswordValid) { + return res.status(401).json({ + success: false, + message: 'Invalid email or password' + }); + } + + // Generate JWT token + const token = generateToken(user._id); + + // Generate refresh token + const refreshToken = RefreshToken.generateToken( + user._id, + req.headers['user-agent'], + req.ip || req.connection.remoteAddress + ); + await refreshToken.save(); + + // Update last login + user.lastLogin = new Date(); + await user.save(); + + res.json({ + success: true, + message: 'Login successful', + data: { + token, + refreshToken: refreshToken.token, + user: user.toJSON() + } + }); + } catch (error) { + console.error('Login error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Get current user profile +export const getProfile = async (req, res) => { + try { + res.json({ + success: true, + data: { + user: req.user.toJSON() + } + }); + } catch (error) { + console.error('Get profile error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Update user profile +export const updateProfile = async (req, res) => { + try { + // Check for validation errors + const errors = validationResult(req); + if (!errors.isEmpty()) { + return res.status(400).json({ + success: false, + message: 'Validation failed', + errors: errors.array() + }); + } + + const { firstName, lastName } = req.body; + const user = req.user; + + // Update allowed fields + if (firstName !== undefined) user.firstName = firstName; + if (lastName !== undefined) user.lastName = lastName; + + await user.save(); + + res.json({ + success: true, + message: 'Profile updated successfully', + data: { + user: user.toJSON() + } + }); + } catch (error) { + console.error('Update profile error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Refresh JWT token using refresh token +export const refreshToken = async (req, res) => { + try { + const { refreshToken: tokenValue } = req.body; + + if (!tokenValue) { + return res.status(400).json({ + success: false, + message: 'Refresh token is required' + }); + } + + // Find and validate refresh token + const storedToken = await RefreshToken.findValidToken(tokenValue); + if (!storedToken) { + return res.status(401).json({ + success: false, + message: 'Invalid or expired refresh token' + }); + } + + // Check if user is still active + if (!storedToken.userId.isActive) { + await storedToken.revoke(); + return res.status(401).json({ + success: false, + message: 'Account is deactivated' + }); + } + + // Generate new JWT token + const newAccessToken = generateToken(storedToken.userId._id); + + // Generate new refresh token and revoke the old one + const newRefreshToken = RefreshToken.generateToken( + storedToken.userId._id, + req.headers['user-agent'], + req.ip || req.connection.remoteAddress + ); + + await Promise.all([ + newRefreshToken.save(), + storedToken.revoke() + ]); + + res.json({ + success: true, + message: 'Token refreshed successfully', + data: { + token: newAccessToken, + refreshToken: newRefreshToken.token + } + }); + } catch (error) { + console.error('Refresh token error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Logout with refresh token revocation +export const logout = async (req, res) => { + try { + const { refreshToken: tokenValue } = req.body; + + if (tokenValue) { + // Find and revoke the refresh token + const storedToken = await RefreshToken.findValidToken(tokenValue); + if (storedToken) { + await storedToken.revoke(); + } + } + + res.json({ + success: true, + message: 'Logout successful' + }); + } catch (error) { + console.error('Logout error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Logout from all devices +export const logoutAll = async (req, res) => { + try { + const userId = req.user._id; + + // Revoke all refresh tokens for the user + await RefreshToken.revokeAllForUser(userId); + + res.json({ + success: true, + message: 'Logged out from all devices successfully' + }); + } catch (error) { + console.error('Logout all error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; \ No newline at end of file diff --git a/backend/controllers/chartController.js b/backend/controllers/chartController.js new file mode 100644 index 0000000000..fe14baa730 --- /dev/null +++ b/backend/controllers/chartController.js @@ -0,0 +1,353 @@ +import mongoose from 'mongoose'; +import { Tree, Forest } from '../models/index.js'; +import { buildTreeQuery } from '../utils/dashboardUtils.js'; + +/** + * Safely convert string ID to ObjectId + * @param {string} id - String ID to convert + * @returns {mongoose.Types.ObjectId|null} ObjectId or null if invalid + */ +const toObjectId = (id) => { + if (!id || typeof id !== 'string') return null; + try { + return new mongoose.Types.ObjectId(id.trim()); + } catch (error) { + console.warn('Invalid ObjectId format in chart controller:', id); + return null; + } +}; + +/** + * Build filter conditions for charts with multi-forest support + * @param {Object} query - Request query parameters + * @returns {Object} Filter conditions + */ +const buildChartFilters = (query) => { + return { + forestId: query.forestId, + forestIds: query.forestIds, + startDate: query.startDate, + endDate: query.endDate, + species: query.species, + isAlive: query.isAlive + }; +}; + +// Get survival rate data over time for charts +export const getSurvivalRateChart = async (req, res) => { + try { + const { groupBy = 'month' } = req.query; // day, week, month, year + + // Build filter conditions using utility function + const filters = buildChartFilters(req.query); + const matchConditions = buildTreeQuery(filters); + + // Define grouping format based on groupBy parameter + const dateFormats = { + day: '%Y-%m-%d', + week: '%Y-%U', + month: '%Y-%m', + year: '%Y' + }; + + const survivalData = await Tree.aggregate([ + { $match: matchConditions }, + { + $group: { + _id: { + period: { + $dateToString: { + format: dateFormats[groupBy] || dateFormats.month, + date: '$plantedDate' + } + } + }, + totalPlanted: { $sum: 1 }, + surviving: { + $sum: { + $cond: ['$isAlive', 1, 0] + } + } + } + }, + { + $addFields: { + survivalRate: { + $multiply: [ + { $divide: ['$surviving', '$totalPlanted'] }, + 100 + ] + } + } + }, + { + $project: { + period: '$_id.period', + totalPlanted: 1, + surviving: 1, + dead: { $subtract: ['$totalPlanted', '$surviving'] }, + survivalRate: { $round: ['$survivalRate', 2] }, + _id: 0 + } + }, + { $sort: { period: 1 } } + ]); + + res.json({ + success: true, + data: { + chartData: survivalData, + groupBy, + totalDataPoints: survivalData.length, + dateRange: { + start: startDate, + end: endDate + } + } + }); + } catch (error) { + console.error('Get survival rate chart error:', error); + res.status(500).json({ + success: false, + message: 'Failed to fetch survival rate chart data', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Get average height growth over time +export const getHeightGrowthChart = async (req, res) => { + try { + const { groupBy = 'month' } = req.query; + + // Build filter conditions using utility function + const filters = buildChartFilters(req.query); + const matchConditions = buildTreeQuery(filters); + + // Ensure we only get alive trees for height growth + matchConditions.isAlive = true; + + // Define grouping format + const dateFormats = { + day: '%Y-%m-%d', + week: '%Y-%U', + month: '%Y-%m', + year: '%Y' + }; + + const heightData = await Tree.aggregate([ + { $match: matchConditions }, + { $unwind: '$measurements' }, + { + $match: { + ...(filters.startDate && { 'measurements.measuredAt': { $gte: new Date(filters.startDate) } }), + ...(filters.endDate && { 'measurements.measuredAt': { $lte: new Date(filters.endDate) } }) + } + }, + { + $group: { + _id: { + period: { + $dateToString: { + format: dateFormats[groupBy] || dateFormats.month, + date: '$measurements.measuredAt' + } + }, + species: '$species' + }, + avgHeight: { $avg: '$measurements.height' }, + minHeight: { $min: '$measurements.height' }, + maxHeight: { $max: '$measurements.height' }, + measurementCount: { $sum: 1 }, + treeCount: { $addToSet: '$_id' } + } + }, + { + $addFields: { + uniqueTreeCount: { $size: '$treeCount' } + } + }, + { + $project: { + period: '$_id.period', + species: '$_id.species', + avgHeight: { $round: ['$avgHeight', 2] }, + minHeight: { $round: ['$minHeight', 2] }, + maxHeight: { $round: ['$maxHeight', 2] }, + measurementCount: 1, + uniqueTreeCount: 1, + _id: 0 + } + }, + { $sort: { period: 1, species: 1 } } + ]); + + // Group by period for easier frontend consumption + const groupedData = heightData.reduce((acc, curr) => { + const period = curr.period; + if (!acc[period]) { + acc[period] = { + period, + species: [], + totalMeasurements: 0, + totalTrees: 0 + }; + } + + acc[period].species.push({ + name: curr.species, + avgHeight: curr.avgHeight, + minHeight: curr.minHeight, + maxHeight: curr.maxHeight, + measurementCount: curr.measurementCount, + treeCount: curr.uniqueTreeCount + }); + + acc[period].totalMeasurements += curr.measurementCount; + acc[period].totalTrees += curr.uniqueTreeCount; + + return acc; + }, {}); + + const chartData = Object.values(groupedData); + + res.json({ + success: true, + data: { + chartData, + groupBy, + totalDataPoints: chartData.length, + filters + } + }); + } catch (error) { + console.error('Get height growth chart error:', error); + res.status(500).json({ + success: false, + message: 'Failed to fetch height growth chart data', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Get CO2 absorption trends over time (based on planted dates for historical trends) +export const getCO2AbsorptionChart = async (req, res) => { + try { + const { groupBy = 'year' } = req.query; + + // Build filter conditions using utility function + const filters = buildChartFilters(req.query); + const matchConditions = buildTreeQuery(filters); + + // Ensure we only get alive trees for CO2 absorption + matchConditions.isAlive = true; + + // Define grouping format + const dateFormats = { + day: '%Y-%m-%d', + week: '%Y-%U', + month: '%Y-%m', + year: '%Y' + }; + + const co2Data = await Tree.aggregate([ + { $match: matchConditions }, + { + $match: { + measurements: { $exists: true, $ne: [] } + } + }, + { + $addFields: { + // Get the latest (most recent) measurement for current CO2 absorption + latestMeasurement: { + $arrayElemAt: [ + { + $filter: { + input: '$measurements', + cond: { $ne: ['$$this.co2Absorption', null] } + } + }, + -1 + ] + } + } + }, + { + $match: { + 'latestMeasurement.co2Absorption': { $exists: true, $ne: null, $gt: 0 } + } + }, + { + $group: { + _id: { + period: { + $dateToString: { + format: dateFormats[groupBy] || dateFormats.month, + date: '$plantedDate' + } + } + }, + // Current CO2 absorption from trees planted in this period + totalCO2: { $sum: '$latestMeasurement.co2Absorption' }, + avgCO2: { $avg: '$latestMeasurement.co2Absorption' }, + minCO2: { $min: '$latestMeasurement.co2Absorption' }, + maxCO2: { $max: '$latestMeasurement.co2Absorption' }, + treeCount: { $sum: 1 }, + trees: { $addToSet: '$_id' } + } + }, + { + $addFields: { + avgCO2PerTree: { $divide: ['$totalCO2', '$treeCount'] } + } + }, + { + $project: { + period: '$_id.period', + totalCO2: { $round: ['$totalCO2', 2] }, + avgCO2: { $round: ['$avgCO2', 2] }, + minCO2: { $round: ['$minCO2', 2] }, + maxCO2: { $round: ['$maxCO2', 2] }, + avgCO2PerTree: { $round: ['$avgCO2PerTree', 2] }, + treeCount: 1, + _id: 0 + } + }, + { $sort: { period: 1 } } + ]); + + // Calculate cumulative CO2 absorption over time (historical buildup) + let cumulativeCO2 = 0; + const chartDataWithCumulative = co2Data.map(item => { + cumulativeCO2 += item.totalCO2; + return { + ...item, + cumulativeCO2: Math.round(cumulativeCO2 * 100) / 100 + }; + }); + + res.json({ + success: true, + data: { + chartData: chartDataWithCumulative, + groupBy, + totalDataPoints: chartDataWithCumulative.length, + summary: { + totalCO2Absorption: cumulativeCO2, + totalTrees: co2Data.reduce((sum, item) => sum + item.treeCount, 0), + avgCO2PerPeriod: Math.round((cumulativeCO2 / co2Data.length) * 100) / 100 + }, + filters + } + }); + } catch (error) { + console.error('Get CO2 absorption chart error:', error); + res.status(500).json({ + success: false, + message: 'Failed to fetch CO2 absorption chart data', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + diff --git a/backend/controllers/dashboardController.js b/backend/controllers/dashboardController.js new file mode 100644 index 0000000000..da0837d625 --- /dev/null +++ b/backend/controllers/dashboardController.js @@ -0,0 +1,305 @@ +import mongoose from 'mongoose'; +import { Forest, Tree, User } from '../models/index.js'; +// TODO: Consider caching frequently accessed dashboard data to reduce database queries +import { + buildTreeQuery, + buildForestQuery, + getRecentDateFilter, + handleDashboardError +} from '../utils/dashboardUtils.js'; +import { + getSpeciesDistributionPipeline, + getHeightStatsPipeline, + getCO2StatsPipeline, + getHealthDistributionPipeline, + getForestStatsPipeline, + // Investor-focused metrics + getPortfolioValuePipeline, + getROIStatsPipeline, + getCarbonCreditsPipeline, + getTimberValuePipeline, + getMaintenanceBudgetPipeline, + // Manager-focused metrics + getBiodiversityIndexPipeline, + getTreesAtRiskPipeline, + getFireRiskPipeline, + getSpeciesDiversityPipeline, + getSoilHealthPipeline, + getInfrastructureConditionPipeline +} from '../utils/aggregationHelpers.js'; +import { + formatDashboardResponse, + +} from '../utils/dataFormatters.js'; +import { generateDashboardSimulation } from '../utils/simulationEngine.js'; + +// Get comprehensive dashboard statistics +export const getDashboardStats = async (req, res) => { + try { + const filters = { + forestId: req.query.forestId, + forestIds: req.query.forestIds, // Support multi-forest filtering + startDate: req.query.startDate, + endDate: req.query.endDate, + species: req.query.species, + isAlive: req.query.isAlive + }; + + console.log('🔍 Dashboard stats request with filters:', filters); + + // Build query conditions using utility functions + const treeQuery = buildTreeQuery(filters); + const forestQuery = buildForestQuery(filters); + + console.log('🔍 Built tree query:', treeQuery); + console.log('🔍 Built forest query:', forestQuery); + + // Check total documents in collection (for debugging) + const totalTreesInDB = await Tree.countDocuments({}); + const totalForestsInDB = await Forest.countDocuments({}); + console.log('🔍 Total documents in DB - Trees:', totalTreesInDB, 'Forests:', totalForestsInDB); + console.log('🔍 Database connection info:', { + readyState: mongoose.connection.readyState, + name: mongoose.connection.name, + host: mongoose.connection.host, + collection: Tree.collection.collectionName + }); + + // Get basic counts + const [ + totalTrees, + aliveTrees, + totalForests, + totalUsers + ] = await Promise.all([ + Tree.countDocuments(treeQuery), + Tree.countDocuments({ ...treeQuery, isAlive: true }), + Forest.countDocuments(forestQuery), + User.countDocuments({ isActive: true }) + ]); + + console.log('📊 Basic counts:', { + totalTrees, + aliveTrees, + totalForests, + totalUsers + }); + + // Get recent activity (trees planted in last 30 days) + const thirtyDaysAgo = getRecentDateFilter(30); + const recentActivity = await Tree.countDocuments({ + ...treeQuery, + createdAt: { $gte: thirtyDaysAgo } + }); + + // Execute aggregation pipelines in parallel alongside simulation data + const [ + speciesDistribution, + heightStats, + co2Stats, + healthDistribution, + forestStats, + simulationData + ] = await Promise.all([ + Tree.aggregate(getSpeciesDistributionPipeline(treeQuery)), + Tree.aggregate(getHeightStatsPipeline(treeQuery)), + Tree.aggregate(getCO2StatsPipeline(treeQuery)), + Tree.aggregate(getHealthDistributionPipeline(treeQuery)), + Forest.aggregate(getForestStatsPipeline(forestQuery)), + // Generate simulation data using mathematical models + generateDashboardSimulation(filters) + ]); + + // Prepare raw data for formatting (combines real data with simulations) + const rawData = { + totalTrees, + aliveTrees, + totalForests, + totalUsers, + recentActivity, + speciesDistribution, + heightStats, + co2Stats, + healthDistribution, + forestStats, + // Enhanced metrics from simulation engine + investor: simulationData.investor, + ecological: simulationData.ecological, + simulationSummary: simulationData.summary + }; + + // Format and send response + const response = formatDashboardResponse(rawData, filters); + res.json(response); + + } catch (error) { + handleDashboardError(res, error, 'Failed to fetch dashboard statistics'); + } +}; + + +// Get enhanced dashboard statistics with simulated financial/ecological data +export const getEnhancedDashboardStats = async (req, res) => { + try { + const filters = { + forestId: req.query.forestId, + forestIds: req.query.forestIds, + startDate: req.query.startDate, + endDate: req.query.endDate, + species: req.query.species, + isAlive: req.query.isAlive + }; + + console.log('🔍 Enhanced dashboard stats request with filters:', filters); + + // Generate simulation data with error handling + let simulationData; + try { + simulationData = await generateDashboardSimulation(filters); + console.log('✅ Simulation data generated successfully'); + } catch (simulationError) { + console.error('❌ Simulation generation failed:', simulationError); + // Fallback to empty simulation structure + simulationData = { + investor: { + portfolio: { totalCurrentValue: 0, forestCount: 0, treeCount: 0, averageValue: 0 }, + timber: { totalValue: 0, averageValuePerTree: 0 }, + carbonCredits: { totalAvailable: 0, totalSold: 0, averagePrice: 0, totalRevenue: 0 }, + roi: { averageROI: 0, minROI: 0, maxROI: 0 }, + maintenance: { totalBudget: 0, totalSpent: 0, utilization: 0 } + }, + ecological: { + biodiversity: { speciesCount: 0, shannonIndex: 0, dominantSpecies: [] }, + environmental: { carbonSequestration: { annualTons: 0, lifetimeStorage: 0 } }, + riskAssessment: { treesAtRisk: 0, riskPercentage: 0, mainRiskFactors: [] }, + forestHealth: { overallScore: 0, aliveTrees: 0, totalTrees: 0 } + }, + summary: { totalTrees: 0, totalForests: 0, aliveTrees: 0, filters, lastUpdated: new Date().toISOString() } + }; + } + + // Get basic counts for verification + const treeQuery = buildTreeQuery(filters); + console.log('🔍 Tree query:', JSON.stringify(treeQuery, null, 2)); + + const [totalTrees, aliveTrees] = await Promise.all([ + Tree.countDocuments(treeQuery), + Tree.countDocuments({ ...treeQuery, isAlive: true }) + ]); + + console.log('📊 Tree counts:', { totalTrees, aliveTrees }); + + // If no trees found with filters, get total count without filters for demonstration + const fallbackTotalTrees = totalTrees === 0 ? await Tree.countDocuments({}) : totalTrees; + const fallbackAliveTrees = aliveTrees === 0 ? await Tree.countDocuments({ isAlive: true }) : aliveTrees; + + console.log('📊 Fallback counts:', { fallbackTotalTrees, fallbackAliveTrees }); + + // Use fallback counts if needed + const finalTotalTrees = Math.max(totalTrees, fallbackTotalTrees); + const finalAliveTrees = Math.max(aliveTrees, fallbackAliveTrees); + + // Calculate survival rate + const survivalRate = finalTotalTrees > 0 ? (finalAliveTrees / finalTotalTrees) * 100 : 85.0; + + // Calculate average height from measurements (use fallback counts) + const heightData = []; + const averageHeight = 15.5; // Realistic height for Swedish forests + + // Calculate CO2 absorption from simulation data + const totalCO2Absorption = simulationData.ecological?.environmental?.carbonSequestration?.annualTons || + Math.round(finalTotalTrees * 0.025); // Default 25kg per tree per year + + // Generate chart data for faster loading + const chartData = { + survivalRate: { + chartData: [ + { period: '2020', totalPlanted: Math.round(finalTotalTrees * 0.15), surviving: Math.round(finalTotalTrees * 0.13), survivalRate: 87 }, + { period: '2021', totalPlanted: Math.round(finalTotalTrees * 0.25), surviving: Math.round(finalTotalTrees * 0.22), survivalRate: 88 }, + { period: '2022', totalPlanted: Math.round(finalTotalTrees * 0.35), surviving: Math.round(finalTotalTrees * 0.31), survivalRate: 89 }, + { period: '2023', totalPlanted: Math.round(finalTotalTrees * 0.50), surviving: Math.round(finalTotalTrees * 0.44), survivalRate: 88 }, + { period: '2024', totalPlanted: finalTotalTrees, surviving: finalAliveTrees, survivalRate: Math.round(survivalRate) } + ], + totalDataPoints: 5 + }, + co2Absorption: { + chartData: [ + { period: 'Q1 2024', totalAbsorption: Math.round(totalCO2Absorption * 0.2), cumulativeTotal: Math.round(totalCO2Absorption * 0.2) }, + { period: 'Q2 2024', totalAbsorption: Math.round(totalCO2Absorption * 0.25), cumulativeTotal: Math.round(totalCO2Absorption * 0.45) }, + { period: 'Q3 2024', totalAbsorption: Math.round(totalCO2Absorption * 0.28), cumulativeTotal: Math.round(totalCO2Absorption * 0.73) }, + { period: 'Q4 2024', totalAbsorption: Math.round(totalCO2Absorption * 0.27), cumulativeTotal: totalCO2Absorption } + ], + totalDataPoints: 4 + }, + forestValue: { + chartData: [ + { period: '2020', currentValue: Math.round(simulationData.investor.portfolio.totalCurrentValue * 0.65), acquisitionCost: simulationData.investor.portfolio.totalAcquisitionCost }, + { period: '2021', currentValue: Math.round(simulationData.investor.portfolio.totalCurrentValue * 0.75), acquisitionCost: simulationData.investor.portfolio.totalAcquisitionCost }, + { period: '2022', currentValue: Math.round(simulationData.investor.portfolio.totalCurrentValue * 0.85), acquisitionCost: simulationData.investor.portfolio.totalAcquisitionCost }, + { period: '2023', currentValue: Math.round(simulationData.investor.portfolio.totalCurrentValue * 0.95), acquisitionCost: simulationData.investor.portfolio.totalAcquisitionCost }, + { period: '2024', currentValue: simulationData.investor.portfolio.totalCurrentValue, acquisitionCost: simulationData.investor.portfolio.totalAcquisitionCost } + ], + totalDataPoints: 5 + } + }; + + // Combine real counts with simulation data + const response = { + success: true, + data: { + // Frontend expects these specific paths: + overview: { + totalTrees: finalTotalTrees, + aliveTrees: finalAliveTrees, + survivalRate: Math.round(survivalRate * 100) / 100, + totalForests: simulationData.summary.totalForests + }, + height: { + average: Math.round(averageHeight * 10) / 10 + }, + co2: { + totalAbsorbed: totalCO2Absorption, + totalAbsorption: totalCO2Absorption, // Frontend expects this field name + annualAbsorption: totalCO2Absorption + }, + // Enhanced data structure for compatibility + basic: { + totalTrees: finalTotalTrees, + aliveTrees: finalAliveTrees, + survivalRate: Math.round(survivalRate * 100) / 100, + totalForests: simulationData.summary.totalForests + }, + // Enhanced financial metrics + investor: { + ...simulationData.investor, + portfolio: { + ...simulationData.investor.portfolio, + totalCurrentValue: Math.max(simulationData.investor.portfolio.totalCurrentValue, finalTotalTrees * 2500) // Minimum 2500 SEK per tree + }, + roi: { + ...simulationData.investor.roi, + averageROI: Math.max(simulationData.investor.roi.averageROI, 45.0) // Ensure ROI is at least 45% + } + }, + // Enhanced ecological metrics + ecological: { + ...simulationData.ecological, + biodiversity: { + ...simulationData.ecological.biodiversity, + speciesCount: Math.max(simulationData.ecological.biodiversity.speciesCount, 3) // At least 3 species + } + }, + // Chart data for instant loading + charts: chartData, + // Metadata + filters, + lastUpdated: simulationData.summary.lastUpdated + } + }; + + res.json(response); + + } catch (error) { + handleDashboardError(res, error, 'Failed to fetch enhanced dashboard statistics'); + } +}; \ No newline at end of file diff --git a/backend/controllers/exportController.js b/backend/controllers/exportController.js new file mode 100644 index 0000000000..dcd1c8725b --- /dev/null +++ b/backend/controllers/exportController.js @@ -0,0 +1,223 @@ +import { Tree, Forest } from '../models/index.js'; +import XLSX from 'xlsx'; +import { + processTreesForExport, + generateCSVContent, + generateExportFilename, + setCSVResponseHeaders, + setXLSXResponseHeaders, + calculateForestAnalytics, + generateExportStatistics, + transformTreeToExportRow +} from '../utils/exportHelpers.js'; +import { handleDashboardError } from '../utils/dashboardUtils.js'; + +// Export trees data to CSV +export const exportTreesCSV = async (req, res) => { + try { + // Extract all field selection parameters from query + const fieldOptions = { ...req.query }; + + // Get ALL trees from database + const trees = await Tree.find({}).lean(); + + // Manually populate forestId information for valid references + const forestIds = [...new Set(trees.map(t => t.forestId).filter(Boolean))]; + const forests = await Forest.find( + { _id: { $in: forestIds } }, + { name: 1, region: 1 } + ).lean(); + + const forestMap = new Map(forests.map(f => [f._id.toString(), f])); + + // Attach forest info to trees + trees.forEach(tree => { + if (tree.forestId) { + tree.forestId = forestMap.get(tree.forestId.toString()) || null; + } + }); + + + if (trees.length === 0) { + return res.status(404).json({ + success: false, + message: 'No trees found matching the criteria' + }); + } + + // Process trees data for export with all field options + const csvData = processTreesForExport(trees, fieldOptions); + + // Generate CSV content + const csvContent = generateCSVContent(csvData); + + // Set response headers and send + const filename = generateExportFilename('nanwa_trees_export', 'csv'); + setCSVResponseHeaders(res, filename, csvContent); + res.send(csvContent); + } catch (error) { + handleDashboardError(res, error, 'Failed to export data'); + } +}; + +// Export trees data to XLSX +export const exportTreesXLSX = async (req, res) => { + try { + // Extract all field selection parameters from query + const fieldOptions = { ...req.query }; + + // Get ALL trees from database + const trees = await Tree.find({}).lean(); + + // Manually populate forestId information for valid references + const forestIds = [...new Set(trees.map(t => t.forestId).filter(Boolean))]; + const forests = await Forest.find( + { _id: { $in: forestIds } }, + { name: 1, region: 1 } + ).lean(); + + const forestMap = new Map(forests.map(f => [f._id.toString(), f])); + + // Attach forest info to trees + trees.forEach(tree => { + if (tree.forestId) { + tree.forestId = forestMap.get(tree.forestId.toString()) || null; + } + }); + + + if (trees.length === 0) { + return res.status(404).json({ + success: false, + message: 'No trees found matching the criteria' + }); + } + + // Create workbook + const workbook = XLSX.utils.book_new(); + + // Prepare trees summary data using helper with all field options + const treesData = trees.map(tree => transformTreeToExportRow(tree, fieldOptions)); + + // Add trees summary sheet + const treesWorksheet = XLSX.utils.json_to_sheet(treesData); + XLSX.utils.book_append_sheet(workbook, treesWorksheet, 'Trees Summary'); + + // Add measurements sheet if requested + if (fieldOptions.measurements === 'true') { + const measurementsData = []; + + trees.forEach(tree => { + tree.measurements.forEach((measurement, index) => { + measurementsData.push({ + 'Tree ID': tree.treeId, + 'Forest Name': tree.forestId?.name || 'Unknown', + 'Species': tree.species, + 'Measurement #': index + 1, + 'Height (m)': measurement.height, + 'Diameter (cm)': measurement.diameter || '', + 'CO2 Absorption (kg)': measurement.co2Absorption || '', + 'Health Status': measurement.healthStatus, + 'Measurement Date': measurement.measuredAt.toISOString().split('T')[0], + 'Notes': measurement.notes || '' + }); + }); + }); + + if (measurementsData.length > 0) { + const measurementsWorksheet = XLSX.utils.json_to_sheet(measurementsData); + XLSX.utils.book_append_sheet(workbook, measurementsWorksheet, 'Measurements'); + } + } + + // Add statistics sheet using helper + const statistics = generateExportStatistics(trees); + const statsData = Object.entries(statistics).map(([key, value]) => ({ + 'Metric': key, + 'Value': value + })); + + const statsWorksheet = XLSX.utils.json_to_sheet(statsData); + XLSX.utils.book_append_sheet(workbook, statsWorksheet, 'Statistics'); + + // Convert workbook to buffer + const buffer = XLSX.write(workbook, { type: 'buffer', bookType: 'xlsx' }); + + // Set response headers and send + const filename = generateExportFilename('nanwa_trees_export', 'xlsx'); + setXLSXResponseHeaders(res, filename, buffer); + res.send(buffer); + } catch (error) { + handleDashboardError(res, error, 'Failed to export data'); + } +}; + + +// Export forest analytics data (not being used at the moment) +export const exportForestAnalytics = async (req, res) => { + try { + const { format = 'xlsx' } = req.query; + + // Get all forests with their tree statistics + const forests = await Forest.find({ isActive: true }).lean(); + + const forestAnalytics = await Promise.all( + forests.map(async (forest) => { + const totalTrees = await Tree.countDocuments({ forestId: forest._id }); + const aliveTrees = await Tree.countDocuments({ forestId: forest._id, isAlive: true }); + + // Get average height from latest measurements + const avgHeightResult = await Tree.aggregate([ + { $match: { forestId: forest._id, isAlive: true } }, + { $unwind: '$measurements' }, + { $sort: { 'measurements.measuredAt': -1 } }, + { $group: { + _id: '$_id', + latestHeight: { $first: '$measurements.height' } + }}, + { $group: { + _id: null, + avgHeight: { $avg: '$latestHeight' } + }} + ]); + + const avgHeight = avgHeightResult.length > 0 ? avgHeightResult[0].avgHeight : 0; + + // Get total CO2 absorption + const co2Result = await Tree.aggregate([ + { $match: { forestId: forest._id, isAlive: true } }, + { $unwind: '$measurements' }, + { $group: { + _id: null, + totalCO2: { $sum: '$measurements.co2Absorption' } + }} + ]); + + const totalCO2 = co2Result.length > 0 ? co2Result[0].totalCO2 : 0; + + // Use helper to calculate analytics + return calculateForestAnalytics(forest, totalTrees, aliveTrees, avgHeight, totalCO2); + }) + ); + + if (format === 'csv') { + // Export as CSV using helper + const csvContent = generateCSVContent(forestAnalytics); + const filename = generateExportFilename('nanwa_forest_analytics', 'csv'); + setCSVResponseHeaders(res, filename, csvContent); + res.send(csvContent); + } else { + // Export as XLSX + const workbook = XLSX.utils.book_new(); + const worksheet = XLSX.utils.json_to_sheet(forestAnalytics); + XLSX.utils.book_append_sheet(workbook, worksheet, 'Forest Analytics'); + + const buffer = XLSX.write(workbook, { type: 'buffer', bookType: 'xlsx' }); + const filename = generateExportFilename('nanwa_forest_analytics', 'xlsx'); + setXLSXResponseHeaders(res, filename, buffer); + res.send(buffer); + } + } catch (error) { + handleDashboardError(res, error, 'Failed to export forest analytics'); + } +}; \ No newline at end of file diff --git a/backend/controllers/forestController.js b/backend/controllers/forestController.js new file mode 100644 index 0000000000..d71a7e53f1 --- /dev/null +++ b/backend/controllers/forestController.js @@ -0,0 +1,373 @@ +import { validationResult } from 'express-validator'; +import { Forest, Tree } from '../models/index.js'; + +// Get all forests with optional filtering +export const getForests = async (req, res) => { + try { + const { + region, + startDate, + endDate, + page = 1, + limit = 10, + sortBy = 'name', + sortOrder = 'asc' + } = req.query; + + // Build query conditions + const queryConditions = { isActive: true }; + + if (region) { + queryConditions.region = new RegExp(region, 'i'); + } + + if (startDate || endDate) { + queryConditions.establishedDate = {}; + if (startDate) queryConditions.establishedDate.$gte = new Date(startDate); + if (endDate) queryConditions.establishedDate.$lte = new Date(endDate); + } + + // Calculate pagination + const skip = (page - 1) * limit; + const sortDirection = sortOrder === 'desc' ? -1 : 1; + + // Execute query with pagination + const forests = await Forest.find(queryConditions) + .sort({ [sortBy]: sortDirection }) + .skip(skip) + .limit(parseInt(limit)) + .populate('treeCount'); + + // Get total count for pagination + const totalCount = await Forest.countDocuments(queryConditions); + + res.json({ + success: true, + data: { + forests, + pagination: { + currentPage: parseInt(page), + totalPages: Math.ceil(totalCount / limit), + totalCount, + hasNextPage: page * limit < totalCount, + hasPrevPage: page > 1 + } + } + }); + } catch (error) { + console.error('Get forests error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Get single forest by ID +export const getForestById = async (req, res) => { + try { + const { id } = req.params; + + const forest = await Forest.findById(id).populate('treeCount'); + if (!forest || !forest.isActive) { + return res.status(404).json({ + success: false, + message: 'Forest not found' + }); + } + + // Get forest statistics + const totalTrees = await Tree.countDocuments({ forestId: id }); + const aliveTrees = await Tree.countDocuments({ forestId: id, isAlive: true }); + const survivalRate = totalTrees > 0 ? (aliveTrees / totalTrees) * 100 : 0; + + // Get average height from latest measurements + const avgHeightResult = await Tree.aggregate([ + { $match: { forestId: forest._id, isAlive: true } }, + { $unwind: '$measurements' }, + { $sort: { 'measurements.measuredAt': -1 } }, + { $group: { + _id: '$_id', + latestHeight: { $first: '$measurements.height' } + }}, + { $group: { + _id: null, + avgHeight: { $avg: '$latestHeight' } + }} + ]); + + const avgHeight = avgHeightResult.length > 0 ? avgHeightResult[0].avgHeight : 0; + + // Get total CO2 absorption + const co2Result = await Tree.aggregate([ + { $match: { forestId: forest._id, isAlive: true } }, + { $unwind: '$measurements' }, + { $group: { + _id: null, + totalCO2: { $sum: '$measurements.co2Absorption' } + }} + ]); + + const totalCO2 = co2Result.length > 0 ? co2Result[0].totalCO2 : 0; + + res.json({ + success: true, + data: { + forest, + statistics: { + totalTrees, + aliveTrees, + survivalRate: Math.round(survivalRate * 100) / 100, + averageHeight: Math.round(avgHeight * 100) / 100, + totalCO2Absorption: Math.round(totalCO2 * 100) / 100 + } + } + }); + } catch (error) { + console.error('Get forest by ID error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Create new forest (admin only) +export const createForest = async (req, res) => { + try { + // Check for validation errors + const errors = validationResult(req); + if (!errors.isEmpty()) { + return res.status(400).json({ + success: false, + message: 'Validation failed', + errors: errors.array() + }); + } + + const forest = new Forest(req.body); + await forest.save(); + + res.status(201).json({ + success: true, + message: 'Forest created successfully', + data: { forest } + }); + } catch (error) { + console.error('Create forest error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Update forest (admin only) +export const updateForest = async (req, res) => { + try { + // Check for validation errors + const errors = validationResult(req); + if (!errors.isEmpty()) { + return res.status(400).json({ + success: false, + message: 'Validation failed', + errors: errors.array() + }); + } + + const { id } = req.params; + const forest = await Forest.findByIdAndUpdate( + id, + { ...req.body, updatedAt: Date.now() }, + { new: true, runValidators: true } + ); + + if (!forest) { + return res.status(404).json({ + success: false, + message: 'Forest not found' + }); + } + + res.json({ + success: true, + message: 'Forest updated successfully', + data: { forest } + }); + } catch (error) { + console.error('Update forest error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Delete forest (admin only) +export const deleteForest = async (req, res) => { + try { + const { id } = req.params; + + // Soft delete by setting isActive to false + const forest = await Forest.findByIdAndUpdate( + id, + { isActive: false, updatedAt: Date.now() }, + { new: true } + ); + + if (!forest) { + return res.status(404).json({ + success: false, + message: 'Forest not found' + }); + } + + res.json({ + success: true, + message: 'Forest deleted successfully' + }); + } catch (error) { + console.error('Delete forest error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; + +// Get forest analytics +export const getForestAnalytics = async (req, res) => { + try { + const { id } = req.params; + const { startDate, endDate } = req.query; + + // Validate forest exists + const forest = await Forest.findById(id); + if (!forest || !forest.isActive) { + return res.status(404).json({ + success: false, + message: 'Forest not found' + }); + } + + // Build date filter + const dateFilter = {}; + if (startDate || endDate) { + dateFilter['measurements.measuredAt'] = {}; + if (startDate) dateFilter['measurements.measuredAt'].$gte = new Date(startDate); + if (endDate) dateFilter['measurements.measuredAt'].$lte = new Date(endDate); + } + + // Get survival rate over time + const survivalRateData = await Tree.aggregate([ + { $match: { forestId: forest._id } }, + { + $group: { + _id: { + year: { $year: '$plantedDate' }, + month: { $month: '$plantedDate' } + }, + totalPlanted: { $sum: 1 }, + surviving: { $sum: { $cond: ['$isAlive', 1, 0] } } + } + }, + { + $project: { + date: { + $dateFromParts: { + year: '$_id.year', + month: '$_id.month', + day: 1 + } + }, + totalPlanted: 1, + surviving: 1, + survivalRate: { + $multiply: [{ $divide: ['$surviving', '$totalPlanted'] }, 100] + } + } + }, + { $sort: { date: 1 } } + ]); + + // Get average height over time + const heightData = await Tree.aggregate([ + { $match: { forestId: forest._id, isAlive: true, ...dateFilter } }, + { $unwind: '$measurements' }, + { + $group: { + _id: { + year: { $year: '$measurements.measuredAt' }, + month: { $month: '$measurements.measuredAt' } + }, + avgHeight: { $avg: '$measurements.height' } + } + }, + { + $project: { + date: { + $dateFromParts: { + year: '$_id.year', + month: '$_id.month', + day: 1 + } + }, + avgHeight: { $round: ['$avgHeight', 2] } + } + }, + { $sort: { date: 1 } } + ]); + + // Get CO2 absorption over time + const co2Data = await Tree.aggregate([ + { $match: { forestId: forest._id, isAlive: true, ...dateFilter } }, + { $unwind: '$measurements' }, + { + $group: { + _id: { + year: { $year: '$measurements.measuredAt' }, + month: { $month: '$measurements.measuredAt' } + }, + totalCO2: { $sum: '$measurements.co2Absorption' } + } + }, + { + $project: { + date: { + $dateFromParts: { + year: '$_id.year', + month: '$_id.month', + day: 1 + } + }, + totalCO2: { $round: ['$totalCO2', 2] } + } + }, + { $sort: { date: 1 } } + ]); + + res.json({ + success: true, + data: { + forest, + analytics: { + survivalRate: survivalRateData, + averageHeight: heightData, + co2Absorption: co2Data + } + } + }); + } catch (error) { + console.error('Get forest analytics error:', error); + res.status(500).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); + } +}; \ No newline at end of file diff --git a/backend/controllers/treeController.js b/backend/controllers/treeController.js new file mode 100644 index 0000000000..92ab793b4d --- /dev/null +++ b/backend/controllers/treeController.js @@ -0,0 +1,269 @@ +import { Tree } from '../models/index.js'; +import { buildTreeQuery, buildPaginationOptions } from '../utils/dashboardUtils.js'; +import { + handleValidationErrors, + verifyForestExists, + transformTreesForMapping, + createMeasurementData, + handleTreeError, + sendTreeListResponse, + sendTreeResponse, + sendTreeMappingResponse, + sendMeasurementResponse, + sendMeasurementsHistoryResponse +} from '../utils/treeHelpers.js'; + +// Get all trees with optional filtering +export const getTrees = async (req, res) => { + try { + const { + forestId, + forestIds, + species, + isAlive, + startDate, + endDate, + page = 1, + limit = 20, + sortBy = 'plantedDate', + sortOrder = 'desc' + } = req.query; + + // Build query conditions using utility + const queryConditions = buildTreeQuery({ forestId, forestIds, species, isAlive, startDate, endDate }); + + // Build pagination options using utility + const { skip, sort } = buildPaginationOptions({ page, limit, sortBy, sortOrder }); + + // Execute query with pagination + const trees = await Tree.find(queryConditions) + .populate('forestId', 'name region') + .sort(sort) + .skip(skip) + .limit(parseInt(limit)); + + // Get total count for pagination + const totalCount = await Tree.countDocuments(queryConditions); + + sendTreeListResponse(res, trees, totalCount, page, limit); + } catch (error) { + handleTreeError(res, error, 'Get trees'); + } +}; + +// Get single tree by ID +export const getTreeById = async (req, res) => { + try { + const { id } = req.params; + + const tree = await Tree.findById(id) + .populate('forestId', 'name region') + .populate('measurements.measuredBy', 'firstName lastName'); + + if (!tree) { + return res.status(404).json({ + success: false, + message: 'Tree not found' + }); + } + + sendTreeResponse(res, tree); + } catch (error) { + handleTreeError(res, error, 'Get tree by ID'); + } +}; + +// Create new tree (admin only) +export const createTree = async (req, res) => { + try { + // Check for validation errors + if (handleValidationErrors(req, res)) return; + + // Verify forest exists + const forest = await verifyForestExists(req.body.forestId); + if (!forest) { + return res.status(404).json({ + success: false, + message: 'Forest not found' + }); + } + + // Check if tree ID already exists + const existingTree = await Tree.findOne({ treeId: req.body.treeId }); + if (existingTree) { + return res.status(409).json({ + success: false, + message: 'Tree with this ID already exists' + }); + } + + const tree = new Tree(req.body); + await tree.save(); + + // Populate forest information + await tree.populate('forestId', 'name region'); + + sendTreeResponse(res, tree, 'Tree created successfully', 201); + } catch (error) { + handleTreeError(res, error, 'Create tree'); + } +}; + +// Update tree (admin only) +export const updateTree = async (req, res) => { + try { + const { id } = req.params; + + const tree = await Tree.findByIdAndUpdate( + id, + { ...req.body, updatedAt: Date.now() }, + { new: true, runValidators: true } + ).populate('forestId', 'name region'); + + if (!tree) { + return res.status(404).json({ + success: false, + message: 'Tree not found' + }); + } + + sendTreeResponse(res, tree, 'Tree updated successfully'); + } catch (error) { + handleTreeError(res, error, 'Update tree'); + } +}; + +// Delete tree (admin only) +export const deleteTree = async (req, res) => { + try { + const { id } = req.params; + + const tree = await Tree.findByIdAndDelete(id); + if (!tree) { + return res.status(404).json({ + success: false, + message: 'Tree not found' + }); + } + + res.json({ + success: true, + message: 'Tree deleted successfully' + }); + } catch (error) { + handleTreeError(res, error, 'Delete tree'); + } +}; + +// Add measurement to tree +export const addMeasurement = async (req, res) => { + try { + // Check for validation errors + if (handleValidationErrors(req, res)) return; + + const { id } = req.params; + const measurementData = createMeasurementData(req.body, req.user._id); + + const tree = await Tree.findById(id); + if (!tree) { + return res.status(404).json({ + success: false, + message: 'Tree not found' + }); + } + + tree.measurements.push(measurementData); + await tree.save(); + + // Populate the new measurement + await tree.populate('measurements.measuredBy', 'firstName lastName'); + + sendMeasurementResponse(res, tree); + } catch (error) { + handleTreeError(res, error, 'Add measurement'); + } +}; + +// Get tree measurements history +export const getTreeMeasurements = async (req, res) => { + try { + const { id } = req.params; + const { limit = 10 } = req.query; + + const tree = await Tree.findById(id) + .populate('measurements.measuredBy', 'firstName lastName'); + + if (!tree) { + return res.status(404).json({ + success: false, + message: 'Tree not found' + }); + } + + // Get latest measurements + const measurements = tree.getLatestMeasurements(parseInt(limit)); + + sendMeasurementsHistoryResponse(res, tree.treeId, measurements); + } catch (error) { + handleTreeError(res, error, 'Get tree measurements'); + } +}; + +// Get trees by forest with location data for mapping +export const getTreesByForest = async (req, res) => { + try { + const { forestId } = req.params; + const { isAlive = true } = req.query; + + // Verify forest exists + const forest = await verifyForestExists(forestId); + if (!forest) { + return res.status(404).json({ + success: false, + message: 'Forest not found' + }); + } + + const trees = await Tree.find({ + forestId, + isAlive: isAlive === 'true' + }).select('treeId location species measurements'); + + // Transform data for mapping + const treeMarkers = transformTreesForMapping(trees); + + sendTreeMappingResponse(res, forest, treeMarkers); + } catch (error) { + handleTreeError(res, error, 'Get trees by forest'); + } +}; + +// Mark tree as dead +export const markTreeDead = async (req, res) => { + try { + const { id } = req.params; + const { deathCause, deathDate } = req.body; + + const tree = await Tree.findByIdAndUpdate( + id, + { + isAlive: false, + deathDate: deathDate || new Date(), + deathCause, + updatedAt: Date.now() + }, + { new: true } + ).populate('forestId', 'name region'); + + if (!tree) { + return res.status(404).json({ + success: false, + message: 'Tree not found' + }); + } + + sendTreeResponse(res, tree, 'Tree marked as dead'); + } catch (error) { + handleTreeError(res, error, 'Mark tree dead'); + } +}; \ No newline at end of file diff --git a/backend/docs/README.md b/backend/docs/README.md new file mode 100644 index 0000000000..a56fe2be98 --- /dev/null +++ b/backend/docs/README.md @@ -0,0 +1,219 @@ +# Nanwa Forestry API Documentation + +## Overview +The Nanwa Forestry API provides comprehensive endpoints for forest and tree management, monitoring, and data visualization. This RESTful API supports authentication, real-time data tracking, and advanced analytics for forestry operations. + +## API Documentation Access + +### Interactive Documentation (Swagger UI) +Access the interactive API documentation at: +- **Development**: [http://localhost:8080/docs](http://localhost:8080/docs) +- **Staging**: [https://api-staging.nanwa-forestry.com/docs](https://api-staging.nanwa-forestry.com/docs) +- **Production**: [https://api.nanwa-forestry.com/docs](https://api.nanwa-forestry.com/docs) + +### OpenAPI Specification +Download the OpenAPI 3.0 specification: +- **JSON Format**: [/docs/json](http://localhost:8080/docs/json) +- **YAML Format**: [/docs/yaml](http://localhost:8080/docs/yaml) + +## API Endpoints Overview + +### Authentication (`/api/auth`) +- `POST /register` - Register new user account +- `POST /login` - User authentication +- `POST /logout` - Logout from current session +- `POST /refresh` - Refresh access token +- `GET /profile` - Get user profile information +- `PUT /profile` - Update user profile +- `POST /logout-all` - Logout from all devices + +### Dashboard Analytics (`/api/dashboard`) +- `GET /stats` - Comprehensive dashboard statistics +- `GET /quick-stats` - Quick stats for widgets +- `GET /forest-comparison` - Forest comparison data + +### Chart Data (`/api/charts`) +- `GET /survival-rate` - Tree survival rate over time +- `GET /height-growth` - Tree height growth trends +- `GET /co2-absorption` - CO2 absorption analytics +- `GET /health-status` - Tree health distribution +- `GET /combined` - All chart data combined + +### Forests (`/api/forests`) +- Forest management endpoints (CRUD operations) +- Geospatial queries and filtering +- Forest metadata and statistics + +### Trees (`/api/trees`) +- Tree management and tracking +- Measurement recording and history +- Species and health monitoring + +### Data Export (`/api/exports`) +- CSV and Excel data export +- Customizable data ranges and filters +- Bulk data downloads + +### Audit Logs (`/api/audit`) +- Activity tracking and logging +- Change history and accountability +- Administrative oversight + +### User Management (`/api/users`) +- User administration (admin only) +- Role management +- Account activation/deactivation + +## Authentication + +The API uses JWT (JSON Web Token) based authentication: + +```bash +# Login to get tokens +POST /api/auth/login +{ + "email": "user@example.com", + "password": "password123" +} + +# Use the access token in subsequent requests +Authorization: Bearer +``` + +## Rate Limiting + +The API implements rate limiting to prevent abuse: +- **General endpoints**: 100 requests per minute +- **Authentication endpoints**: 20 requests per minute +- **Data-heavy endpoints**: 50 requests per minute + +## Response Format + +All API responses follow a consistent format: + +```json +{ + "success": true, + "message": "Operation completed successfully", + "data": { + // Response data here + } +} +``` + +Error responses: +```json +{ + "success": false, + "message": "Error description", + "errors": [ + { + "field": "email", + "message": "Invalid email format" + } + ] +} +``` + +## Environment Endpoints + +| Environment | Base URL | Status | +|-------------|----------|--------| +| Development | `http://localhost:8080/api` | 🟢 Active | +| Staging | `https://api-staging.nanwa-forestry.com/api` | 🟡 Preview | +| Production | `https://api.nanwa-forestry.com/api` | 🔴 Not deployed | + +## Data Formats + +### Date Handling +- All dates are in ISO 8601 format: `YYYY-MM-DDTHH:mm:ss.sssZ` +- Query parameters accept simplified dates: `YYYY-MM-DD` + +### Geospatial Data +- Coordinates use GeoJSON format: `[longitude, latitude]` +- Geographic queries support radius and bounding box searches + +### Pagination +```json +{ + "page": 1, + "limit": 20, + "total": 150, + "pages": 8, + "data": [...] +} +``` + +## Examples + +### Get Dashboard Statistics +```bash +curl -H "Authorization: Bearer " \ + "http://localhost:8080/api/dashboard/stats?forestId=123&startDate=2024-01-01" +``` + +### Register New User +```bash +curl -X POST http://localhost:8080/api/auth/register \ + -H "Content-Type: application/json" \ + -d '{ + "firstName": "John", + "lastName": "Doe", + "email": "john.doe@example.com", + "password": "securePassword123", + "confirmPassword": "securePassword123" + }' +``` + +### Get Chart Data +```bash +curl -H "Authorization: Bearer " \ + "http://localhost:8080/api/charts/survival-rate?groupBy=month&forestId=123" +``` + +## Development + +### Prerequisites +- Node.js 18+ +- MongoDB 7.0+ +- npm or yarn + +### Setup +```bash +# Install dependencies +npm install + +# Set environment variables +cp .env.example .env + +# Start development server +npm run dev +``` + +### Testing the API +```bash +# Run all tests +npm test + +# Test specific endpoint +npm test -- --testNamePattern="auth" + +# Generate coverage report +npm run test:coverage +``` + +## Support + +- **Documentation Issues**: Submit issues to the [GitHub repository](https://github.com/your-org/nanwa-forestry) +- **API Support**: Contact [support@nanwa-forestry.com](mailto:support@nanwa-forestry.com) +- **Emergency**: For production issues, contact the on-call team + +## Changelog + +### Version 1.0.0 +- Initial API release +- Complete authentication system +- Dashboard analytics endpoints +- Chart data APIs with time-series support +- Comprehensive test coverage +- OpenAPI 3.0 documentation \ No newline at end of file diff --git a/backend/healthcheck.js b/backend/healthcheck.js new file mode 100644 index 0000000000..18930f7587 --- /dev/null +++ b/backend/healthcheck.js @@ -0,0 +1,28 @@ +import http from 'http'; + +const options = { + hostname: 'localhost', + port: process.env.PORT || 8080, + path: '/health', + method: 'GET', + timeout: 2000 +}; + +const req = http.request(options, (res) => { + if (res.statusCode === 200) { + process.exit(0); + } else { + process.exit(1); + } +}); + +req.on('error', () => { + process.exit(1); +}); + +req.on('timeout', () => { + req.destroy(); + process.exit(1); +}); + +req.end(); \ No newline at end of file diff --git a/backend/jest.config.js b/backend/jest.config.js new file mode 100644 index 0000000000..c3911d9768 --- /dev/null +++ b/backend/jest.config.js @@ -0,0 +1,25 @@ +export default { + testEnvironment: 'node', + transform: { + '^.+\\.js$': 'babel-jest' + }, + testMatch: [ + '**/tests/**/*.test.js', + '**/tests/**/*.spec.js' + ], + collectCoverageFrom: [ + 'controllers/**/*.js', + 'middleware/**/*.js', + 'models/**/*.js', + 'routes/**/*.js', + '!**/node_modules/**', + '!**/tests/**' + ], + coverageDirectory: 'coverage', + coverageReporters: ['text', 'lcov', 'html'], + setupFilesAfterEnv: ['/tests/setup.js'], + testTimeout: 30000, + transformIgnorePatterns: [ + 'node_modules/(?!(mongodb-memory-server)/)' + ] +}; \ No newline at end of file diff --git a/backend/middleware/auth.js b/backend/middleware/auth.js new file mode 100644 index 0000000000..9d6f9d936e --- /dev/null +++ b/backend/middleware/auth.js @@ -0,0 +1,74 @@ +import { User } from '../models/index.js'; +import { verifyToken, extractTokenFromHeader } from '../utils/jwt.js'; + +// Middleware to verify JWT token and authenticate user +export const authenticateToken = async (req, res, next) => { + try { + const token = extractTokenFromHeader(req.headers.authorization); + const decoded = verifyToken(token); + + // Find the user and attach to request + const user = await User.findById(decoded.userId); + if (!user) { + return res.status(401).json({ + success: false, + message: 'User not found' + }); + } + + if (!user.isActive) { + return res.status(401).json({ + success: false, + message: 'Account is deactivated' + }); + } + + // Attach user to request object + req.user = user; + next(); + } catch (error) { + return res.status(401).json({ + success: false, + message: error.message + }); + } +}; + +// Middleware to check if user is admin +export const requireAdmin = (req, res, next) => { + if (!req.user) { + return res.status(401).json({ + success: false, + message: 'Authentication required' + }); + } + + if (req.user.role !== 'admin') { + return res.status(403).json({ + success: false, + message: 'Admin access required' + }); + } + + next(); +}; + +// Optional authentication - doesn't fail if no token provided +export const optionalAuth = async (req, res, next) => { + try { + if (req.headers.authorization) { + const token = extractTokenFromHeader(req.headers.authorization); + const decoded = verifyToken(token); + + const user = await User.findById(decoded.userId); + if (user && user.isActive) { + req.user = user; + } + } + } catch (error) { + // Continue without authentication if token is invalid + console.log('Optional auth failed:', error.message); + } + + next(); +}; \ No newline at end of file diff --git a/backend/middleware/rateLimiter.js b/backend/middleware/rateLimiter.js new file mode 100644 index 0000000000..8a3dfe0b4a --- /dev/null +++ b/backend/middleware/rateLimiter.js @@ -0,0 +1,90 @@ +import rateLimit from 'express-rate-limit'; + +// General API rate limiter +export const generalLimiter = rateLimit({ + windowMs: 15 * 60 * 1000, // 15 minutes + max: parseInt(process.env.RATE_LIMIT) || 100, // Default to 100 requests per window + message: { + success: false, + message: 'Too many requests from this IP, please try again later.', + retryAfter: '15 minutes' + }, + standardHeaders: true, // Return rate limit info in the `RateLimit-*` headers + legacyHeaders: false, // Disable the `X-RateLimit-*` headers + handler: (req, res) => { + res.status(429).json({ + success: false, + message: 'Too many requests from this IP, please try again later.', + retryAfter: Math.ceil(req.rateLimit.resetTime / 1000), + limit: req.rateLimit.limit, + remaining: req.rateLimit.remaining + }); + } +}); + +// Stricter rate limiter for authentication endpoints +export const authLimiter = rateLimit({ + windowMs: 15 * 60 * 1000, // 15 minutes + max: 5, // Limit each IP to 5 requests per windowMs + message: { + success: false, + message: 'Too many authentication attempts from this IP, please try again later.', + retryAfter: '15 minutes' + }, + standardHeaders: true, + legacyHeaders: false, + skipSuccessfulRequests: true, // Don't count successful requests + handler: (req, res) => { + res.status(429).json({ + success: false, + message: 'Too many authentication attempts from this IP, please try again later.', + retryAfter: Math.ceil(req.rateLimit.resetTime / 1000), + limit: req.rateLimit.limit, + remaining: req.rateLimit.remaining + }); + } +}); + +// More lenient rate limiter for data retrieval endpoints +export const dataLimiter = rateLimit({ + windowMs: 1 * 60 * 1000, // 1 minute + max: parseInt(process.env.DATA_RATE_LIMIT) || 30, // Default to 30 requests per minute + message: { + success: false, + message: 'Too many data requests from this IP, please try again later.', + retryAfter: '1 minute' + }, + standardHeaders: true, + legacyHeaders: false, + handler: (req, res) => { + res.status(429).json({ + success: false, + message: 'Too many data requests from this IP, please try again later.', + retryAfter: Math.ceil(req.rateLimit.resetTime / 1000), + limit: req.rateLimit.limit, + remaining: req.rateLimit.remaining + }); + } +}); + +// Very strict rate limiter for admin operations +export const adminLimiter = rateLimit({ + windowMs: 5 * 60 * 1000, // 5 minutes + max: 10, // Limit each IP to 10 admin operations per 5 minutes + message: { + success: false, + message: 'Too many admin operations from this IP, please try again later.', + retryAfter: '5 minutes' + }, + standardHeaders: true, + legacyHeaders: false, + handler: (req, res) => { + res.status(429).json({ + success: false, + message: 'Too many admin operations from this IP, please try again later.', + retryAfter: Math.ceil(req.rateLimit.resetTime / 1000), + limit: req.rateLimit.limit, + remaining: req.rateLimit.remaining + }); + } +}); \ No newline at end of file diff --git a/backend/middleware/validation.js b/backend/middleware/validation.js new file mode 100644 index 0000000000..56c32a7ca2 --- /dev/null +++ b/backend/middleware/validation.js @@ -0,0 +1,158 @@ +import { body } from 'express-validator'; + +// Common validation patterns +const commonValidators = { + email: () => body('email') + .isEmail() + .normalizeEmail() + .withMessage('Please provide a valid email address'), + + password: () => body('password') + .isLength({ min: 6 }) + .withMessage('Password must be at least 6 characters long') + .matches(/^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)/) + .withMessage('Password must contain at least one lowercase letter, one uppercase letter, and one number'), + + passwordLogin: () => body('password') + .notEmpty() + .withMessage('Password is required'), + + firstName: (required = true) => { + const validator = body('firstName') + .trim() + .isLength({ min: 1, max: 50 }) + .withMessage('First name must be between 1 and 50 characters') + .matches(/^[a-zA-Z\s]+$/) + .withMessage('First name can only contain letters and spaces'); + + return required ? validator : validator.optional(); + }, + + lastName: (required = true) => { + const validator = body('lastName') + .trim() + .isLength({ min: 1, max: 50 }) + .withMessage('Last name must be between 1 and 50 characters') + .matches(/^[a-zA-Z\s]+$/) + .withMessage('Last name can only contain letters and spaces'); + + return required ? validator : validator.optional(); + }, + + coordinates: () => [ + body('location.coordinates') + .isArray({ min: 2, max: 2 }) + .withMessage('Coordinates must be an array of [longitude, latitude]'), + body('location.coordinates.*') + .isFloat({ min: -180, max: 180 }) + .withMessage('Coordinates must be valid numbers') + ], + + mongoId: (field) => body(field) + .isMongoId() + .withMessage(`${field} must be a valid MongoDB ObjectId`), + + positiveFloat: (field, message) => body(field) + .isFloat({ min: 0 }) + .withMessage(message || `${field} must be a positive number`), + + optionalPositiveFloat: (field, message) => body(field) + .optional() + .isFloat({ min: 0 }) + .withMessage(message || `${field} must be a positive number`), + + isoDate: (field, message) => body(field) + .isISO8601() + .withMessage(message || `${field} must be a valid date`), + + trimmedString: (field, min, max, message) => body(field) + .trim() + .isLength({ min, max }) + .withMessage(message || `${field} must be between ${min} and ${max} characters`), + + optionalString: (field, max, message) => body(field) + .optional() + .isLength({ max }) + .withMessage(message || `${field} cannot exceed ${max} characters`) +}; + +// Validation rules for user registration +export const validateRegister = [ + commonValidators.email(), + commonValidators.password(), + commonValidators.firstName(), + commonValidators.lastName() +]; + +// Validation rules for user login +export const validateLogin = [ + commonValidators.email(), + commonValidators.passwordLogin() +]; + +// Validation rules for profile update +export const validateProfileUpdate = [ + commonValidators.firstName(false), + commonValidators.lastName(false) +]; + +// Validation rules for forest creation +export const validateForestCreate = [ + commonValidators.trimmedString('name', 1, 100, 'Forest name must be between 1 and 100 characters'), + commonValidators.trimmedString('region', 1, 100, 'Region must be between 1 and 100 characters'), + ...commonValidators.coordinates(), + commonValidators.positiveFloat('area', 'Area must be a positive number'), + body('areaUnit') + .optional() + .isIn(['hectares', 'acres', 'sq_km', 'sq_miles']) + .withMessage('Area unit must be one of: hectares, acres, sq_km, sq_miles'), + commonValidators.isoDate('establishedDate', 'Established date must be a valid date'), + commonValidators.optionalString('description', 500, 'Description cannot exceed 500 characters') +]; + +// Validation rules for tree creation +export const validateTreeCreate = [ + commonValidators.trimmedString('treeId', 1, 50, 'Tree ID must be between 1 and 50 characters'), + commonValidators.mongoId('forestId'), + commonValidators.trimmedString('species', 1, 100, 'Species must be between 1 and 100 characters'), + ...commonValidators.coordinates(), + commonValidators.isoDate('plantedDate', 'Planted date must be a valid date') +]; + +// Validation rules for tree measurement +export const validateMeasurement = [ + commonValidators.positiveFloat('height', 'Height must be a positive number'), + commonValidators.optionalPositiveFloat('diameter', 'Diameter must be a positive number'), + commonValidators.optionalPositiveFloat('co2Absorption', 'CO₂ absorption must be a positive number'), + body('healthStatus') + .optional() + .isIn(['excellent', 'good', 'fair', 'poor', 'critical']) + .withMessage('Health status must be one of: excellent, good, fair, poor, critical'), + commonValidators.optionalString('notes', 500, 'Notes cannot exceed 500 characters') +]; + +// Validation rules for admin user creation +export const validateUserCreate = [ + commonValidators.email(), + commonValidators.password(), + commonValidators.firstName(), + commonValidators.lastName(), + body('role') + .optional() + .isIn(['user', 'admin']) + .withMessage('Role must be either user or admin') +]; + +// Validation rules for admin user update +export const validateUserUpdate = [ + commonValidators.firstName(false), + commonValidators.lastName(false), + body('role') + .optional() + .isIn(['user', 'admin']) + .withMessage('Role must be either user or admin'), + body('isActive') + .optional() + .isBoolean() + .withMessage('isActive must be a boolean value') +]; \ No newline at end of file diff --git a/backend/models/Forest.js b/backend/models/Forest.js new file mode 100644 index 0000000000..f47d1f960d --- /dev/null +++ b/backend/models/Forest.js @@ -0,0 +1,629 @@ +import mongoose from 'mongoose'; + +const forestSchema = new mongoose.Schema({ + name: { + type: String, + required: [true, 'Forest name is required'], + trim: true, + maxlength: [100, 'Forest name cannot exceed 100 characters'] + }, + region: { + type: String, + required: [true, 'Region is required'], + trim: true, + maxlength: [100, 'Region name cannot exceed 100 characters'] + }, + + // Owner relationship + owner: { + type: mongoose.Schema.Types.ObjectId, + ref: 'Owner', + required: [true, 'Forest owner is required'] + }, + + // Forest type classification + isPlantation: { + type: Boolean, + required: true, + default: false // false = natural forest, true = plantation + }, + + location: { + type: { + type: String, + enum: ['Point'], + required: true + }, + coordinates: { + type: [Number], + required: true, + validate: { + validator: function(v) { + return v.length === 2; + }, + message: 'Coordinates must be an array of [longitude, latitude]' + } + } + }, + area: { + type: Number, + required: [true, 'Forest area is required'], + min: [0, 'Area must be positive'] + }, + areaUnit: { + type: String, + enum: ['hectares', 'acres', 'sq_km', 'sq_miles'], + default: 'hectares' + }, + establishedDate: { + type: Date, + required: [true, 'Established date is required'] + }, + description: { + type: String, + maxlength: [500, 'Description cannot exceed 500 characters'] + }, + + // Financial tracking + financials: { + acquisitionCost: { + type: Number, + min: [0, 'Acquisition cost must be positive'] + }, + acquisitionDate: Date, + currentValue: { + type: Number, + min: [0, 'Current value must be positive'] + }, + lastValuationDate: Date, + currency: { + type: String, + default: 'SEK', + enum: ['SEK', 'EUR', 'USD'] + }, + maintenanceBudget: { + annual: { + type: Number, + min: [0, 'Annual budget must be positive'] + }, + allocated: { + type: Number, + min: [0, 'Allocated budget must be positive'] + }, + spent: { + type: Number, + min: [0, 'Spent amount must be positive'] + } + } + }, + + // Biodiversity tracking + biodiversity: { + species: [{ + name: String, + scientificName: String, + category: { + type: String, + enum: ['flora', 'fauna', 'fungi'] + }, + count: { + type: Number, + min: [0, 'Species count must be positive'] + }, + lastObserved: Date, + conservationStatus: { + type: String, + enum: ['common', 'uncommon', 'rare', 'endangered', 'critically_endangered'] + }, + notes: String + }], + biodiversityIndex: { + type: Number, + min: [0, 'Biodiversity index must be positive'], + max: [100, 'Biodiversity index cannot exceed 100'] + }, + lastSurveyDate: Date, + habitatTypes: [{ + type: String, + area: Number, // in hectares + condition: { + type: String, + enum: ['excellent', 'good', 'fair', 'poor'] + } + }] + }, + + // Environmental monitoring + environmental: { + waterSources: [{ + type: { + type: String, + enum: ['river', 'stream', 'pond', 'lake', 'wetland', 'spring'] + }, + name: String, + coordinates: { + type: { + type: String, + enum: ['Point'], + default: 'Point' + }, + coordinates: [Number] + }, + quality: { + type: String, + enum: ['excellent', 'good', 'fair', 'poor'] + }, + flowRate: Number, // liters per second + seasonal: Boolean + }], + fireRisk: { + current: { + type: String, + enum: ['low', 'moderate', 'high', 'extreme'] + }, + lastAssessment: Date, + mitigationMeasures: [String] + }, + pestInfestation: [{ + pestType: String, + species: String, + severity: { + type: String, + enum: ['none', 'minor', 'moderate', 'severe'] + }, + affectedArea: { + type: Number, + min: [0, 'Affected area must be positive'] + }, + discoveredDate: Date, + treatmentApplied: String, + treatmentDate: Date, + status: { + type: String, + enum: ['active', 'treated', 'resolved'] + } + }], + soilHealth: { + phLevel: { + type: Number, + min: [0, 'pH must be positive'], + max: [14, 'pH cannot exceed 14'] + }, + organicMatter: Number, // percentage + compaction: { + type: String, + enum: ['low', 'moderate', 'high'] + }, + erosionRisk: { + type: String, + enum: ['low', 'moderate', 'high'] + }, + lastTested: Date + } + }, + + // Carbon tracking and climate benefits + carbonMetrics: { + totalCarbonStored: { + type: Number, + min: [0, 'Carbon stored must be positive'] + }, + annualSequestration: { + type: Number, + min: [0, 'Annual sequestration must be positive'] + }, + carbonCredits: { + issued: { + type: Number, + min: [0, 'Issued credits must be positive'] + }, + sold: { + type: Number, + min: [0, 'Sold credits must be positive'] + }, + available: { + type: Number, + min: [0, 'Available credits must be positive'] + }, + pricePerCredit: { + type: Number, + min: [0, 'Price per credit must be positive'] + } + }, + lastCalculation: Date, + calculationMethod: String + }, + + // Infrastructure and access + infrastructure: { + roads: [{ + name: String, + type: { + type: String, + enum: ['paved', 'gravel', 'dirt', 'trail', 'logging_road'] + }, + length: { + type: Number, + min: [0, 'Road length must be positive'] + }, + width: Number, // meters + condition: { + type: String, + enum: ['excellent', 'good', 'fair', 'poor'] + }, + lastMaintenance: Date, + accessLevel: { + type: String, + enum: ['public', 'restricted', 'private'] + } + }], + facilities: [{ + type: { + type: String, + enum: ['office', 'storage', 'equipment_shed', 'observation_tower', 'rest_area', 'research_station'] + }, + name: String, + coordinates: { + type: { + type: String, + enum: ['Point'], + default: 'Point' + }, + coordinates: [Number] + }, + condition: { + type: String, + enum: ['excellent', 'good', 'fair', 'poor'] + }, + builtDate: Date, + lastMaintenance: Date, + capacity: Number + }], + fencing: { + totalLength: Number, // kilometers + type: String, + condition: { + type: String, + enum: ['excellent', 'good', 'fair', 'poor'] + }, + lastInspection: Date + } + }, + + // Sustainable harvesting plans + harvestPlan: { + zones: [{ + name: String, + area: { + type: Number, + min: [0, 'Zone area must be positive'] + }, + plannedHarvestYear: Number, + harvestMethod: { + type: String, + enum: ['clear_cut', 'selective', 'shelterwood', 'continuous_cover'] + }, + sustainabilityRating: { + type: String, + enum: ['low', 'medium', 'high', 'certified'] + }, + estimatedYield: { + volume: { + type: Number, + min: [0, 'Volume must be positive'] + }, + value: { + type: Number, + min: [0, 'Value must be positive'] + } + }, + lastHarvest: Date, + rotationPeriod: Number // years + }], + certifications: [{ + type: { + type: String, + enum: ['FSC', 'PEFC', 'ISO14001', 'SVEASKOG', 'Other'] + }, + certificationNumber: String, + issuedDate: Date, + expiryDate: Date, + issuingBody: String, + status: { + type: String, + enum: ['active', 'expired', 'pending_renewal'] + } + }], + nextPlannedHarvest: Date, + harvestRotation: Number // years + }, + + isActive: { + type: Boolean, + default: true + }, + + // Enhanced metadata with forest-specific information + metadata: { + soilType: { + type: String, + enum: ['clay', 'loam', 'sand', 'peat', 'rocky', 'mixed'] + }, + climate: { + type: String, + enum: ['temperate', 'continental', 'oceanic', 'boreal'] + }, + elevation: { + type: Number, + min: [0, 'Elevation must be positive'] + }, + avgRainfall: { + type: Number, + min: [0, 'Average rainfall must be positive'] + }, + avgTemperature: Number, + growingSeason: { + startMonth: { + type: Number, + min: [1, 'Month must be between 1 and 12'], + max: [12, 'Month must be between 1 and 12'] + }, + endMonth: { + type: Number, + min: [1, 'Month must be between 1 and 12'], + max: [12, 'Month must be between 1 and 12'] + }, + lengthDays: Number + }, + dominantSpecies: [String], + forestAge: { + type: String, + enum: ['young', 'mature', 'old_growth', 'mixed_age'] + }, + managementIntensity: { + type: String, + enum: ['intensive', 'moderate', 'extensive', 'minimal'] + } + } +}, { + timestamps: true, + toJSON: { virtuals: true }, + toObject: { virtuals: true } +}); + +// Create geospatial index for location-based queries +forestSchema.index({ location: '2dsphere' }); + +// Index for common query patterns +forestSchema.index({ region: 1, isActive: 1 }); +forestSchema.index({ establishedDate: 1 }); +forestSchema.index({ owner: 1, isActive: 1 }); +forestSchema.index({ isPlantation: 1 }); +forestSchema.index({ 'metadata.forestAge': 1 }); +forestSchema.index({ 'carbonMetrics.totalCarbonStored': 1 }); + +// Virtual for tree count +forestSchema.virtual('treeCount', { + ref: 'Tree', + localField: '_id', + foreignField: 'forestId', + count: true +}); + +// Virtual for forest health percentage +forestSchema.virtual('healthPercentage').get(function() { + if (this.biodiversity && this.biodiversity.biodiversityIndex) { + return this.biodiversity.biodiversityIndex; + } + return null; +}); + +// Instance method to calculate forest health +forestSchema.methods.calculateHealth = async function() { + const Tree = mongoose.model('Tree'); + const trees = await Tree.find({ forestId: this._id }); + + if (trees.length === 0) return 'unknown'; + + const aliveTrees = trees.filter(tree => tree.isAlive); + const healthyTrees = aliveTrees.filter(tree => { + const latestMeasurement = tree.measurements && tree.measurements.length > 0 + ? tree.measurements[tree.measurements.length - 1] + : null; + return latestMeasurement && ['excellent', 'good'].includes(latestMeasurement.healthStatus); + }); + + const healthPercentage = aliveTrees.length > 0 ? (healthyTrees.length / aliveTrees.length) * 100 : 0; + + if (healthPercentage >= 80) return 'excellent'; + if (healthPercentage >= 60) return 'good'; + if (healthPercentage >= 40) return 'fair'; + return 'poor'; +}; + +// Instance method to calculate ecological benefits +forestSchema.methods.calculateEcologicalBenefits = async function() { + const Tree = mongoose.model('Tree'); + const trees = await Tree.find({ forestId: this._id, isAlive: true }); + + let totalBenefits = { + stormwaterIntercepted: { gallons: 0, value: 0 }, + co2Sequestered: { pounds: 0, value: 0 }, + co2Stored: { pounds: 0, value: 0 }, + airPollutantsRemoved: { pounds: 0, value: 0 }, + soilErosionPrevented: { tons: 0, value: 0 }, + wildlifeHabitatValue: { score: 0, value: 0 }, + totalAnnualValue: 0 + }; + + for (const tree of trees) { + if (tree.ecologicalBenefits) { + const benefits = tree.ecologicalBenefits; + + if (benefits.stormwaterIntercepted) { + totalBenefits.stormwaterIntercepted.gallons += benefits.stormwaterIntercepted.gallons || 0; + totalBenefits.stormwaterIntercepted.value += benefits.stormwaterIntercepted.value || 0; + } + + if (benefits.co2Sequestered) { + totalBenefits.co2Sequestered.pounds += benefits.co2Sequestered.pounds || 0; + totalBenefits.co2Sequestered.value += benefits.co2Sequestered.value || 0; + } + + if (benefits.co2Stored) { + totalBenefits.co2Stored.pounds += benefits.co2Stored.pounds || 0; + totalBenefits.co2Stored.value += benefits.co2Stored.value || 0; + } + + if (benefits.airPollutantsRemoved) { + totalBenefits.airPollutantsRemoved.pounds += benefits.airPollutantsRemoved.pounds || 0; + totalBenefits.airPollutantsRemoved.value += benefits.airPollutantsRemoved.value || 0; + } + } + } + + // Calculate forest-level benefits + const canopyCover = trees.reduce((total, tree) => { + return total + (tree.canopy && tree.canopy.area ? tree.canopy.area : 0); + }, 0); + + // Soil erosion prevention (based on canopy cover and slope) + const erosionPreventionRate = 0.5; // tons per square meter of canopy per year + totalBenefits.soilErosionPrevented.tons = canopyCover * erosionPreventionRate; + totalBenefits.soilErosionPrevented.value = totalBenefits.soilErosionPrevented.tons * 15; // $15 per ton + + // Wildlife habitat value (based on biodiversity and area) + const habitatScore = this.biodiversity && this.biodiversity.biodiversityIndex ? this.biodiversity.biodiversityIndex : 50; + totalBenefits.wildlifeHabitatValue.score = habitatScore * this.area; + totalBenefits.wildlifeHabitatValue.value = totalBenefits.wildlifeHabitatValue.score * 0.5; // $0.50 per score point + + // Calculate total annual value + totalBenefits.totalAnnualValue = + totalBenefits.stormwaterIntercepted.value + + totalBenefits.co2Sequestered.value + + totalBenefits.airPollutantsRemoved.value + + totalBenefits.soilErosionPrevented.value + + totalBenefits.wildlifeHabitatValue.value; + + return totalBenefits; +}; + +// Instance method to calculate forest value +forestSchema.methods.calculateForestValue = async function() { + const Tree = mongoose.model('Tree'); + const trees = await Tree.find({ forestId: this._id, isAlive: true }); + + let totalTimberValue = 0; + let totalCarbonValue = 0; + + for (const tree of trees) { + if (tree.economicValue) { + totalTimberValue += tree.economicValue.currentTimberValue || 0; + totalCarbonValue += tree.economicValue.carbonCreditValue || 0; + } + } + + const ecologicalBenefits = await this.calculateEcologicalBenefits(); + + return { + timberValue: totalTimberValue, + carbonValue: totalCarbonValue, + ecologicalValue: ecologicalBenefits.totalAnnualValue * 10, // 10-year present value + totalValue: totalTimberValue + totalCarbonValue + (ecologicalBenefits.totalAnnualValue * 10) + }; +}; + +// Static method to find forests by owner +forestSchema.statics.findByOwner = function(ownerId) { + return this.find({ + owner: ownerId, + isActive: true + }).populate('owner'); +}; + +// Static method to find forests by region +forestSchema.statics.findByRegion = function(region) { + return this.find({ + region: new RegExp(region, 'i'), + isActive: true + }); +}; + +// Static method to find forests within a date range +forestSchema.statics.findByDateRange = function(startDate, endDate) { + return this.find({ + establishedDate: { + $gte: startDate, + $lte: endDate + }, + isActive: true + }); +}; + +// Static method to find forests by type +forestSchema.statics.findByType = function(isPlantation) { + return this.find({ + isPlantation: isPlantation, + isActive: true + }); +}; + +// Static method for aggregated forest statistics +forestSchema.statics.getAggregatedStats = function(ownerFilter = {}) { + const matchStage = { isActive: true, ...ownerFilter }; + + return this.aggregate([ + { $match: matchStage }, + { + $lookup: { + from: 'trees', + localField: '_id', + foreignField: 'forestId', + as: 'trees' + } + }, + { + $project: { + name: 1, + region: 1, + area: 1, + isPlantation: 1, + owner: 1, + totalTrees: { $size: '$trees' }, + aliveTrees: { + $size: { + $filter: { + input: '$trees', + cond: { $eq: ['$$this.isAlive', true] } + } + } + }, + averageHeight: { + $avg: { + $map: { + input: { + $filter: { + input: '$trees', + cond: { $eq: ['$$this.isAlive', true] } + } + }, + as: 'tree', + in: { + $let: { + vars: { + lastMeasurement: { $arrayElemAt: ['$$tree.measurements', -1] } + }, + in: '$$lastMeasurement.height' + } + } + } + } + } + } + } + ]); +}; + +const Forest = mongoose.model('Forest', forestSchema); + +export default Forest; \ No newline at end of file diff --git a/backend/models/Owner.js b/backend/models/Owner.js new file mode 100644 index 0000000000..63954a015f --- /dev/null +++ b/backend/models/Owner.js @@ -0,0 +1,266 @@ +import mongoose from 'mongoose'; + +const ownerSchema = new mongoose.Schema({ + name: { + type: String, + required: [true, 'Owner name is required'], + trim: true, + maxlength: [200, 'Owner name cannot exceed 200 characters'] + }, + + organizationType: { + type: String, + enum: ['government', 'private_company', 'ngo', 'individual', 'cooperative'], + required: [true, 'Organization type is required'] + }, + + contactInfo: { + email: { + type: String, + required: [true, 'Email is required'], + lowercase: true, + trim: true, + match: [/^\w+([.-]?\w+)*@\w+([.-]?\w+)*(\.\w{2,3})+$/, 'Please enter a valid email'] + }, + phone: { + type: String, + trim: true + }, + address: { + type: String, + trim: true, + maxlength: [500, 'Address cannot exceed 500 characters'] + }, + website: { + type: String, + trim: true + } + }, + + // Certifications and compliance + certifications: [{ + type: { + type: String, + enum: ['FSC', 'PEFC', 'ISO14001', 'ISO9001', 'SVEASKOG', 'Other'] + }, + certificationNumber: String, + issuedDate: Date, + expiryDate: Date, + issuingBody: String, + status: { + type: String, + enum: ['active', 'expired', 'pending_renewal'], + default: 'active' + } + }], + + // Financial information + financials: { + annualRevenue: { + type: Number, + min: [0, 'Annual revenue must be positive'] + }, + totalForestValue: { + type: Number, + min: [0, 'Total forest value must be positive'] + }, + currency: { + type: String, + default: 'SEK', + enum: ['SEK', 'EUR', 'USD'] + }, + insurancePolicies: [{ + provider: String, + policyNumber: String, + coverage: String, + coverageAmount: Number, + annualPremium: Number, + expiryDate: Date, + isActive: { + type: Boolean, + default: true + } + }] + }, + + // Sustainability goals and commitments + sustainabilityGoals: { + carbonNeutralTarget: Date, + biodiversityTargets: [{ + metric: { + type: String, + enum: ['species_diversity', 'habitat_area', 'old_growth_percentage', 'deadwood_volume'] + }, + currentValue: Number, + targetValue: Number, + unit: String, + deadline: Date, + description: String + }], + reforestationCommitment: { + treesPerYear: Number, + hectaresPerYear: Number, + startYear: Number, + endYear: Number + }, + communityEngagement: { + localEmployees: { + type: Number, + min: [0, 'Local employees count must be positive'] + }, + educationPrograms: { + type: Number, + min: [0, 'Education programs count must be positive'] + }, + publicAccessAreas: { + type: Number, + min: [0, 'Public access areas count must be positive'] + }, + communityInvestment: { + amount: Number, + currency: { + type: String, + default: 'SEK' + }, + description: String + } + } + }, + + // Forest management philosophy + managementApproach: { + primaryObjective: { + type: String, + enum: ['timber_production', 'conservation', 'recreation', 'carbon_sequestration', 'mixed_use'], + required: [true, 'Primary objective is required'] + }, + harvestingMethod: { + type: String, + enum: ['clear_cut', 'selective_harvest', 'shelterwood', 'continuous_cover', 'mixed_methods'] + }, + sustainabilityPrinciples: [{ + principle: String, + description: String, + implementationStatus: { + type: String, + enum: ['implemented', 'in_progress', 'planned'], + default: 'planned' + } + }] + }, + + // Relationships + forests: [{ + type: mongoose.Schema.Types.ObjectId, + ref: 'Forest' + }], + + users: [{ + type: mongoose.Schema.Types.ObjectId, + ref: 'User' + }], + + // Status and metadata + isActive: { + type: Boolean, + default: true + }, + + establishedDate: { + type: Date + }, + + description: { + type: String, + maxlength: [1000, 'Description cannot exceed 1000 characters'] + }, + + metadata: { + type: Map, + of: mongoose.Schema.Types.Mixed + } +}, { + timestamps: true, + toJSON: { virtuals: true }, + toObject: { virtuals: true } +}); + +// Virtual for forest count +ownerSchema.virtual('forestCount').get(function() { + return this.forests ? this.forests.length : 0; +}); + +// Virtual for user count +ownerSchema.virtual('userCount').get(function() { + return this.users ? this.users.length : 0; +}); + +// Instance method to calculate total forest area +ownerSchema.methods.calculateTotalArea = async function() { + const Forest = mongoose.model('Forest'); + const forests = await Forest.find({ owner: this._id }); + return forests.reduce((total, forest) => total + (forest.area || 0), 0); +}; + +// Instance method to calculate total tree count +ownerSchema.methods.calculateTotalTrees = async function() { + const Tree = mongoose.model('Tree'); + const Forest = mongoose.model('Forest'); + + const forests = await Forest.find({ owner: this._id }).select('_id'); + const forestIds = forests.map(f => f._id); + + return await Tree.countDocuments({ forestId: { $in: forestIds } }); +}; + +// Instance method to calculate aggregated ecological benefits +ownerSchema.methods.calculateEcologicalBenefits = async function() { + const Forest = mongoose.model('Forest'); + const forests = await Forest.find({ owner: this._id }); + + let totalBenefits = { + stormwaterIntercepted: { gallons: 0, value: 0 }, + co2Sequestered: { pounds: 0, value: 0 }, + co2Stored: { pounds: 0, value: 0 }, + airPollutantsRemoved: { pounds: 0, value: 0 }, + totalAnnualValue: 0 + }; + + for (const forest of forests) { + if (typeof forest.calculateEcologicalBenefits === 'function') { + const forestBenefits = await forest.calculateEcologicalBenefits(); + + totalBenefits.stormwaterIntercepted.gallons += forestBenefits.stormwaterIntercepted.gallons || 0; + totalBenefits.stormwaterIntercepted.value += forestBenefits.stormwaterIntercepted.value || 0; + totalBenefits.co2Sequestered.pounds += forestBenefits.co2Sequestered.pounds || 0; + totalBenefits.co2Sequestered.value += forestBenefits.co2Sequestered.value || 0; + totalBenefits.co2Stored.pounds += forestBenefits.co2Stored.pounds || 0; + totalBenefits.co2Stored.value += forestBenefits.co2Stored.value || 0; + totalBenefits.airPollutantsRemoved.pounds += forestBenefits.airPollutantsRemoved.pounds || 0; + totalBenefits.airPollutantsRemoved.value += forestBenefits.airPollutantsRemoved.value || 0; + totalBenefits.totalAnnualValue += forestBenefits.totalAnnualValue || 0; + } + } + + return totalBenefits; +}; + +// Static method to find owners with expiring certifications +ownerSchema.statics.findExpiringCertifications = function(daysFromNow = 90) { + const futureDate = new Date(); + futureDate.setDate(futureDate.getDate() + daysFromNow); + + return this.find({ + 'certifications.expiryDate': { $lte: futureDate }, + 'certifications.status': 'active' + }); +}; + +// Indexes for better query performance +ownerSchema.index({ name: 1 }); +ownerSchema.index({ organizationType: 1 }); +ownerSchema.index({ 'contactInfo.email': 1 }); +ownerSchema.index({ isActive: 1 }); +ownerSchema.index({ 'managementApproach.primaryObjective': 1 }); + +export const Owner = mongoose.model('Owner', ownerSchema); \ No newline at end of file diff --git a/backend/models/RefreshToken.js b/backend/models/RefreshToken.js new file mode 100644 index 0000000000..614d44bbb7 --- /dev/null +++ b/backend/models/RefreshToken.js @@ -0,0 +1,91 @@ +import mongoose from 'mongoose'; +import crypto from 'crypto'; + +const refreshTokenSchema = new mongoose.Schema({ + token: { + type: String, + required: true, + unique: true + }, + userId: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User', + required: true + }, + isActive: { + type: Boolean, + default: true + }, + userAgent: { + type: String, + maxlength: 500 + }, + ipAddress: { + type: String, + maxlength: 45 // IPv6 addresses can be up to 45 characters + }, + createdAt: { + type: Date, + default: Date.now + }, + expiresAt: { + type: Date, + required: true + } +}); + +// Create TTL index for automatic document deletion +refreshTokenSchema.index({ expiresAt: 1 }, { expireAfterSeconds: 0 }); + +// Index for efficient token lookups +refreshTokenSchema.index({ token: 1, isActive: 1 }); +refreshTokenSchema.index({ userId: 1, isActive: 1 }); + +// Static method to generate a new refresh token +refreshTokenSchema.statics.generateToken = function(userId, userAgent = null, ipAddress = null) { + const token = crypto.randomBytes(64).toString('hex'); + const expiresAt = new Date(); + expiresAt.setDate(expiresAt.getDate() + 30); // 30 days from now + + return new this({ + token, + userId, + userAgent, + ipAddress, + expiresAt + }); +}; + +// Static method to find valid token +refreshTokenSchema.statics.findValidToken = function(token) { + return this.findOne({ + token, + isActive: true, + expiresAt: { $gt: new Date() } + }).populate('userId'); +}; + +// Instance method to revoke token +refreshTokenSchema.methods.revoke = function() { + this.isActive = false; + return this.save(); +}; + +// Static method to revoke all tokens for a user +refreshTokenSchema.statics.revokeAllForUser = function(userId) { + return this.updateMany( + { userId, isActive: true }, + { isActive: false } + ); +}; + +// Static method to cleanup expired tokens (optional, since TTL index handles this) +refreshTokenSchema.statics.cleanupExpired = function() { + return this.deleteMany({ + expiresAt: { $lte: new Date() } + }); +}; + +const RefreshToken = mongoose.model('RefreshToken', refreshTokenSchema); + +export default RefreshToken; \ No newline at end of file diff --git a/backend/models/Tree.js b/backend/models/Tree.js new file mode 100644 index 0000000000..eb55484892 --- /dev/null +++ b/backend/models/Tree.js @@ -0,0 +1,694 @@ +import mongoose from 'mongoose'; + +const measurementSchema = new mongoose.Schema({ + height: { + type: Number, + required: [true, 'Height measurement is required'], + min: [0, 'Height must be positive'] + }, + diameter: { + type: Number, + min: [0, 'Diameter must be positive'] + }, + co2Absorption: { + type: Number, + min: [0, 'CO₂ absorption must be positive'] + }, + healthStatus: { + type: String, + enum: ['excellent', 'good', 'fair', 'poor', 'critical'], + default: 'good' + }, + notes: { + type: String, + maxlength: [500, 'Notes cannot exceed 500 characters'] + }, + measuredBy: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User' + }, + measuredAt: { + type: Date, + default: Date.now + } +}); + +const treeSchema = new mongoose.Schema({ + treeId: { + type: String, + required: [true, 'Tree ID is required'], + unique: true, + trim: true + }, + forestId: { + type: mongoose.Schema.Types.ObjectId, + ref: 'Forest', + required: [true, 'Forest ID is required'] + }, + species: { + type: String, + required: [true, 'Tree species is required'], + trim: true, + maxlength: [100, 'Species name cannot exceed 100 characters'] + }, + location: { + type: { + type: String, + enum: ['Point'], + required: true + }, + coordinates: { + type: [Number], + required: true, + validate: { + validator: function(v) { + return v.length === 2; + }, + message: 'Coordinates must be an array of [longitude, latitude]' + } + } + }, + plantedDate: { + type: Date, + required: [true, 'Planted date is required'] + }, + isAlive: { + type: Boolean, + default: true + }, + deathDate: { + type: Date + }, + deathCause: { + type: String, + maxlength: [200, 'Death cause cannot exceed 200 characters'] + }, + measurements: [measurementSchema], + + // Genetic and source tracking + genetics: { + seedSource: { + type: String, + trim: true + }, + cultivar: { + type: String, + trim: true + }, + parentTreeId: { + type: String, + trim: true + }, + geneticDiversity: { + type: String, + enum: ['native', 'improved', 'hybrid', 'exotic'] + }, + provenanceRegion: String + }, + + // Growth predictions and modeling + growthModel: { + expectedHeightAt15Years: { + type: Number, + min: [0, 'Expected height must be positive'] + }, + expectedDiameterAt15Years: { + type: Number, + min: [0, 'Expected diameter must be positive'] + }, + growthRate: { + type: String, + enum: ['slow', 'moderate', 'fast'] + }, + siteIndex: { + type: Number, + min: [0, 'Site index must be positive'], + max: [50, 'Site index cannot exceed 50'] + }, + maturityAge: Number, // years + maxHeight: Number, // meters + maxDiameter: Number // cm + }, + + // Maintenance and treatment history + maintenance: { + fertilization: [{ + date: Date, + type: { + type: String, + enum: ['organic', 'mineral', 'slow_release', 'liquid'] + }, + npkRatio: String, // e.g., "10-10-10" + amount: Number, // kg or grams + appliedBy: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User' + }, + notes: String + }], + pestControl: [{ + date: Date, + pestType: String, + treatment: String, + method: { + type: String, + enum: ['biological', 'chemical', 'mechanical', 'integrated'] + }, + effectiveness: { + type: String, + enum: ['successful', 'partial', 'unsuccessful'] + }, + followUpRequired: Boolean, + appliedBy: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User' + } + }], + damageReports: [{ + date: Date, + type: { + type: String, + enum: ['wind', 'snow', 'ice', 'animal', 'disease', 'human', 'fire', 'drought', 'flood'] + }, + severity: { + type: String, + enum: ['minor', 'moderate', 'severe'] + }, + description: String, + photoUrls: [String], + treatmentRequired: Boolean, + recoveryExpected: Boolean + }], + pruning: { + lastPruned: Date, + nextScheduled: Date, + history: [{ + date: Date, + type: { + type: String, + enum: ['structural', 'health', 'clearance', 'aesthetic', 'safety'] + }, + prunedBy: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User' + }, + notes: String, + photoUrls: [String] + }] + } + }, + + // Economic value tracking + economicValue: { + currentTimberValue: { + type: Number, + min: [0, 'Timber value must be positive'] + }, + carbonCreditValue: { + type: Number, + min: [0, 'Carbon credit value must be positive'] + }, + lastValuation: Date, + valuationMethod: String, + marketPricePerCubicMeter: Number + }, + + // Tree canopy characteristics (for forest benefit calculations) + canopy: { + diameter: { + type: Number, + min: [0, 'Canopy diameter must be positive'] + }, + area: { + type: Number, + min: [0, 'Canopy area must be positive'] + }, + density: { + type: String, + enum: ['sparse', 'moderate', 'dense'] + }, + condition: { + type: String, + enum: ['excellent', 'good', 'fair', 'poor', 'dead'] + }, + leafArea: Number, // leaf area index + seasonalChanges: { + springBudBreak: Date, + fallColorChange: Date, + leafDrop: Date + } + }, + + // Forest ecosystem benefits (calculated annually) + ecologicalBenefits: { + // Stormwater management + stormwaterIntercepted: { + gallons: { + type: Number, + min: [0, 'Stormwater gallons must be positive'] + }, + value: { + type: Number, + min: [0, 'Stormwater value must be positive'] + }, + lastCalculated: Date + }, + // Carbon sequestration and storage + co2Sequestered: { + pounds: { + type: Number, + min: [0, 'CO2 sequestered must be positive'] + }, + value: { + type: Number, + min: [0, 'CO2 value must be positive'] + }, + lastCalculated: Date + }, + co2Stored: { + pounds: { + type: Number, + min: [0, 'CO2 stored must be positive'] + }, + value: { + type: Number, + min: [0, 'CO2 stored value must be positive'] + }, + lastCalculated: Date + }, + // Air quality improvement + airPollutantsRemoved: { + pounds: { + type: Number, + min: [0, 'Air pollutants removed must be positive'] + }, + value: { + type: Number, + min: [0, 'Air pollutants value must be positive'] + }, + pollutantBreakdown: { + ozone: Number, + no2: Number, + so2: Number, + pm10: Number, + pm25: Number + }, + lastCalculated: Date + }, + // Soil and water benefits + soilStabilization: { + rootArea: Number, // square meters + erosionPrevention: Number, // tons of soil per year + value: Number, + lastCalculated: Date + }, + // Wildlife habitat provision + wildlifeHabitat: { + nestingSites: Number, + foodProduction: Number, // kg of seeds/fruit per year + shelterValue: Number, + biodiversitySupport: { + type: String, + enum: ['low', 'moderate', 'high'] + }, + lastCalculated: Date + } + }, + + // Environmental factors affecting tree growth and benefits + environmentalFactors: { + microclimate: { + avgTemperature: Number, + humidity: Number, + windExposure: { + type: String, + enum: ['sheltered', 'moderate', 'exposed'] + } + }, + siteConditions: { + slope: { + type: String, + enum: ['flat', 'gentle', 'moderate', 'steep'] + }, + aspect: { + type: String, + enum: ['north', 'south', 'east', 'west', 'northeast', 'northwest', 'southeast', 'southwest'] + }, + drainage: { + type: String, + enum: ['well_drained', 'moderately_drained', 'poorly_drained', 'waterlogged'] + }, + competitionIndex: { + type: Number, + min: [0, 'Competition index must be positive'], + max: [10, 'Competition index cannot exceed 10'] + } + }, + forestPosition: { + type: String, + enum: ['canopy', 'understory', 'suppressed', 'dominant', 'co_dominant'] + } + }, + + images: [{ + url: String, + caption: String, + uploadedAt: { + type: Date, + default: Date.now + } + }], + + metadata: { + soilCondition: { + type: String, + enum: ['excellent', 'good', 'moderate', 'poor'] + }, + sunlightExposure: { + type: String, + enum: ['full_sun', 'partial_sun', 'partial_shade', 'full_shade'] + }, + waterAccess: { + type: String, + enum: ['excellent', 'good', 'moderate', 'poor'] + }, + seedlingSource: String, + plantingMethod: { + type: String, + enum: ['hand_planted', 'machine_planted', 'direct_seeded', 'natural_regeneration'] + }, + initialSpacing: Number, // meters between trees + managementObjective: { + type: String, + enum: ['timber', 'conservation', 'recreation', 'carbon', 'wildlife', 'mixed'] + } + }, + createdAt: { + type: Date, + default: Date.now + }, + updatedAt: { + type: Date, + default: Date.now + } +}); + +// Create geospatial index for location-based queries +treeSchema.index({ location: '2dsphere' }); + +// Index for common query patterns +treeSchema.index({ forestId: 1, isAlive: 1 }); +treeSchema.index({ species: 1 }); +treeSchema.index({ plantedDate: 1 }); +treeSchema.index({ treeId: 1 }); +treeSchema.index({ 'measurements.measuredAt': -1 }); +treeSchema.index({ 'genetics.geneticDiversity': 1 }); +treeSchema.index({ 'growthModel.growthRate': 1 }); +treeSchema.index({ 'canopy.condition': 1 }); +treeSchema.index({ 'environmentalFactors.forestPosition': 1 }); +treeSchema.index({ 'economicValue.currentTimberValue': 1 }); + +// Update the updatedAt field before saving +treeSchema.pre('save', function(next) { + this.updatedAt = Date.now(); + next(); +}); + +// Virtual for current height (latest measurement) +treeSchema.virtual('currentHeight').get(function() { + if (this.measurements && this.measurements.length > 0) { + const latestMeasurement = this.measurements + .sort((a, b) => b.measuredAt - a.measuredAt)[0]; + return latestMeasurement.height; + } + return null; +}); + +// Virtual for current health status +treeSchema.virtual('currentHealthStatus').get(function() { + if (this.measurements && this.measurements.length > 0) { + const latestMeasurement = this.measurements + .sort((a, b) => b.measuredAt - a.measuredAt)[0]; + return latestMeasurement.healthStatus; + } + return 'unknown'; +}); + +// Virtual for age in days +treeSchema.virtual('ageInDays').get(function() { + const now = new Date(); + const planted = new Date(this.plantedDate); + return Math.floor((now - planted) / (1000 * 60 * 60 * 24)); +}); + +// Virtual for total CO₂ absorption +treeSchema.virtual('totalCO2Absorption').get(function() { + if (this.measurements && this.measurements.length > 0) { + return this.measurements.reduce((total, measurement) => { + return total + (measurement.co2Absorption || 0); + }, 0); + } + return 0; +}); + +// Method to add a new measurement +treeSchema.methods.addMeasurement = function(measurementData) { + this.measurements.push(measurementData); + return this.save(); +}; + +// Method to get latest measurements (default: 10) +treeSchema.methods.getLatestMeasurements = function(limit = 10) { + return this.measurements + .sort((a, b) => b.measuredAt - a.measuredAt) + .slice(0, limit); +}; + +// Method to calculate tree age in years +treeSchema.methods.getAgeInYears = function() { + const now = new Date(); + const planted = new Date(this.plantedDate); + return Math.floor((now - planted) / (1000 * 60 * 60 * 24 * 365.25)); +}; + +// Method to calculate current ecological benefits +treeSchema.methods.calculateEcologicalBenefits = function() { + if (!this.canopy || !this.canopy.area) return null; + + const canopyArea = this.canopy.area; + const treeAge = this.getAgeInYears(); + const species = this.species.toLowerCase(); + + // Species-specific factors for Swedish forests + const speciesFactors = { + 'pine': { stormwaterRate: 0.8, co2Rate: 25, pollutantRate: 0.15, soilRate: 0.6 }, + 'spruce': { stormwaterRate: 1.0, co2Rate: 30, pollutantRate: 0.18, soilRate: 0.7 }, + 'birch': { stormwaterRate: 0.6, co2Rate: 15, pollutantRate: 0.12, soilRate: 0.4 }, + 'oak': { stormwaterRate: 1.2, co2Rate: 35, pollutantRate: 0.22, soilRate: 0.8 }, + 'fir': { stormwaterRate: 0.9, co2Rate: 28, pollutantRate: 0.16, soilRate: 0.65 } + }; + + const factors = speciesFactors[species] || speciesFactors['pine']; // Default to pine + + // Calculate annual benefits + const benefits = { + stormwater: { + gallons: canopyArea * factors.stormwaterRate * 200, // 200 gallons per m² base rate + value: 0 + }, + co2: { + sequestered: canopyArea * factors.co2Rate * Math.min(treeAge, 10) / 10, // Peak at 10 years + stored: canopyArea * factors.co2Rate * treeAge, + value: 0 + }, + airPollutants: { + pounds: canopyArea * factors.pollutantRate * 6.6, // 6.6 lbs per m² base rate + value: 0 + }, + soilStabilization: { + erosionPrevention: canopyArea * factors.soilRate * 0.5, // tons per year + value: 0 + } + }; + + // Calculate monetary values (Swedish pricing) + benefits.stormwater.value = benefits.stormwater.gallons * 0.008; // ~0.08 SEK per gallon + benefits.co2.value = (benefits.co2.sequestered + benefits.co2.stored * 0.1) * 0.15; // ~1.5 SEK per kg CO2 + benefits.airPollutants.value = benefits.airPollutants.pounds * 35; // ~350 SEK per pound + benefits.soilStabilization.value = benefits.soilStabilization.erosionPrevention * 120; // ~1200 SEK per ton + + return benefits; +}; + +// Method to calculate timber value +treeSchema.methods.calculateTimberValue = function() { + const latestMeasurement = this.getLatestMeasurements(1)[0]; + if (!latestMeasurement || !this.isAlive) return 0; + + const height = latestMeasurement.height; + const diameter = latestMeasurement.diameter || height * 0.1; // Estimate diameter if not available + + // Volume calculation (simplified cone formula) + const volume = Math.PI * Math.pow(diameter / 200, 2) * height * 0.5; // cubic meters + + // Species-specific timber prices (SEK per cubic meter) + const timberPrices = { + 'pine': 600, + 'spruce': 650, + 'birch': 400, + 'oak': 1200, + 'fir': 580 + }; + + const species = this.species.toLowerCase(); + const pricePerCubicMeter = timberPrices[species] || timberPrices['pine']; + + return volume * pricePerCubicMeter; +}; + +// Method to assess tree health based on multiple factors +treeSchema.methods.assessOverallHealth = function() { + if (!this.isAlive) return 'dead'; + + const latestMeasurement = this.getLatestMeasurements(1)[0]; + let healthScore = 100; + + // Factor in latest health status + if (latestMeasurement) { + const healthPoints = { + 'excellent': 100, + 'good': 80, + 'fair': 60, + 'poor': 40, + 'critical': 20 + }; + healthScore = healthPoints[latestMeasurement.healthStatus] || 50; + } + + // Factor in canopy condition + if (this.canopy && this.canopy.condition) { + const canopyPoints = { + 'excellent': 0, + 'good': -5, + 'fair': -15, + 'poor': -25, + 'dead': -50 + }; + healthScore += canopyPoints[this.canopy.condition] || 0; + } + + // Factor in recent damage reports + if (this.maintenance && this.maintenance.damageReports) { + const recentDamage = this.maintenance.damageReports.filter(report => + new Date(report.date) > new Date(Date.now() - 365 * 24 * 60 * 60 * 1000) + ); + + recentDamage.forEach(damage => { + const severityPenalty = { + 'minor': -5, + 'moderate': -15, + 'severe': -30 + }; + healthScore += severityPenalty[damage.severity] || 0; + }); + } + + // Convert score to category + if (healthScore >= 90) return 'excellent'; + if (healthScore >= 75) return 'good'; + if (healthScore >= 55) return 'fair'; + if (healthScore >= 30) return 'poor'; + return 'critical'; +}; + +// Static method to find trees by forest +treeSchema.statics.findByForest = function(forestId) { + return this.find({ forestId, isAlive: true }); +}; + +// Static method to find trees by species +treeSchema.statics.findBySpecies = function(species) { + return this.find({ + species: new RegExp(species, 'i'), + isAlive: true + }); +}; + +// Static method to find trees by health status +treeSchema.statics.findByHealthStatus = function(healthStatus) { + return this.find({ + 'measurements.healthStatus': healthStatus, + isAlive: true + }); +}; + +// Static method to calculate survival rate for a forest +treeSchema.statics.calculateSurvivalRate = async function(forestId, dateRange = null) { + const matchCondition = { forestId }; + + if (dateRange) { + matchCondition.plantedDate = { + $gte: dateRange.start, + $lte: dateRange.end + }; + } + + const totalTrees = await this.countDocuments(matchCondition); + const aliveTrees = await this.countDocuments({ ...matchCondition, isAlive: true }); + + return totalTrees > 0 ? (aliveTrees / totalTrees) * 100 : 0; +}; + +// Static method to get forest statistics +treeSchema.statics.getForestStatistics = async function(forestId) { + const trees = await this.find({ forestId }); + const aliveTrees = trees.filter(tree => tree.isAlive); + + if (trees.length === 0) { + return { + totalTrees: 0, + aliveTrees: 0, + survivalRate: 0, + averageAge: 0, + averageHeight: 0, + speciesDistribution: {}, + totalTimberValue: 0, + totalEcologicalValue: 0 + }; + } + + // Calculate species distribution + const speciesCount = {}; + trees.forEach(tree => { + speciesCount[tree.species] = (speciesCount[tree.species] || 0) + 1; + }); + + // Calculate averages for alive trees + const totalAge = aliveTrees.reduce((sum, tree) => sum + tree.getAgeInYears(), 0); + const avgAge = aliveTrees.length > 0 ? totalAge / aliveTrees.length : 0; + + const heights = aliveTrees.map(tree => { + const latest = tree.getLatestMeasurements(1)[0]; + return latest ? latest.height : 0; + }); + const avgHeight = heights.length > 0 ? heights.reduce((a, b) => a + b, 0) / heights.length : 0; + + // Calculate economic values + const timberValues = aliveTrees.map(tree => tree.calculateTimberValue()); + const totalTimberValue = timberValues.reduce((a, b) => a + b, 0); + + return { + totalTrees: trees.length, + aliveTrees: aliveTrees.length, + survivalRate: (aliveTrees.length / trees.length) * 100, + averageAge: Math.round(avgAge * 10) / 10, + averageHeight: Math.round(avgHeight * 10) / 10, + speciesDistribution: speciesCount, + totalTimberValue: Math.round(totalTimberValue), + totalEcologicalValue: 0 // Will be calculated by forest-level aggregation + }; +}; + +const Tree = mongoose.model('Tree', treeSchema); + +export default Tree; \ No newline at end of file diff --git a/backend/models/User.js b/backend/models/User.js new file mode 100644 index 0000000000..ccc9c5f955 --- /dev/null +++ b/backend/models/User.js @@ -0,0 +1,136 @@ +import mongoose from 'mongoose'; +import bcrypt from 'bcryptjs'; + +const userSchema = new mongoose.Schema({ + email: { + type: String, + required: [true, 'Email is required'], + unique: true, + lowercase: true, + trim: true, + match: [/^\w+([.-]?\w+)*@\w+([.-]?\w+)*(\.\w{2,3})+$/, 'Please enter a valid email'] + }, + password: { + type: String, + required: [true, 'Password is required'], + minlength: [6, 'Password must be at least 6 characters'] + }, + firstName: { + type: String, + required: [true, 'First name is required'], + trim: true, + maxlength: [50, 'First name cannot exceed 50 characters'] + }, + lastName: { + type: String, + required: [true, 'Last name is required'], + trim: true, + maxlength: [50, 'Last name cannot exceed 50 characters'] + }, + role: { + type: String, + enum: ['user', 'admin'], + default: 'user' + }, + + // Owner relationship - null for admin users (they can access all data) + owner: { + type: mongoose.Schema.Types.ObjectId, + ref: 'Owner', + default: null // Admin users have no owner, regular users belong to an owner + }, + + isActive: { + type: Boolean, + default: true + }, + lastLogin: { + type: Date + }, + createdAt: { + type: Date, + default: Date.now + }, + updatedAt: { + type: Date, + default: Date.now + } +}); + +// Hash password before saving +userSchema.pre('save', async function(next) { + // Only hash the password if it has been modified (or is new) + if (!this.isModified('password')) return next(); + + try { + // Hash password with cost of 12 + const salt = await bcrypt.genSalt(12); + this.password = await bcrypt.hash(this.password, salt); + next(); + } catch (error) { + next(error); + } +}); + +// Update the updatedAt field before saving +userSchema.pre('save', function(next) { + this.updatedAt = Date.now(); + next(); +}); + +// Instance method to check password +userSchema.methods.comparePassword = async function(candidatePassword) { + return bcrypt.compare(candidatePassword, this.password); +}; + +// Instance method to check if user is admin +userSchema.methods.isAdmin = function() { + return this.role === 'admin'; +}; + +// Instance method to check if user can access owner data +userSchema.methods.canAccessOwner = function(ownerId) { + // Admin users can access all owners + if (this.isAdmin()) return true; + + // Regular users can only access their own owner's data + return this.owner && this.owner.toString() === ownerId.toString(); +}; + +// Instance method to check if user can access forest data +userSchema.methods.canAccessForest = async function(forestId) { + // Admin users can access all forests + if (this.isAdmin()) return true; + + // Regular users can only access forests owned by their owner + const Forest = mongoose.model('Forest'); + const forest = await Forest.findById(forestId); + + if (!forest || !this.owner) return false; + + return forest.owner.toString() === this.owner.toString(); +}; + +// Instance method to get user info without password +userSchema.methods.toJSON = function() { + const userObject = this.toObject(); + delete userObject.password; + return userObject; +}; + +// Static method to find user by email +userSchema.statics.findByEmail = function(email) { + return this.findOne({ email: email.toLowerCase() }); +}; + +// Static method to find users by owner +userSchema.statics.findByOwner = function(ownerId) { + return this.find({ owner: ownerId, isActive: true }); +}; + +// Add index for owner-based queries +userSchema.index({ owner: 1, isActive: 1 }); + +const User = mongoose.model('User', userSchema); + +export default User; \ No newline at end of file diff --git a/backend/models/index.js b/backend/models/index.js new file mode 100644 index 0000000000..b452b1bf32 --- /dev/null +++ b/backend/models/index.js @@ -0,0 +1,13 @@ +import User from './User.js'; +import Forest from './Forest.js'; +import Tree from './Tree.js'; +import RefreshToken from './RefreshToken.js'; +import { Owner } from './Owner.js'; + +export { + User, + Forest, + Tree, + RefreshToken, + Owner +}; \ No newline at end of file diff --git a/backend/package.json b/backend/package.json index 08f29f2448..2bbc03d4f6 100644 --- a/backend/package.json +++ b/backend/package.json @@ -2,9 +2,14 @@ "name": "project-final-backend", "version": "1.0.0", "description": "Server part of final project", + "type": "module", "scripts": { "start": "babel-node server.js", - "dev": "nodemon server.js --exec babel-node" + "dev": "nodemon server.js --exec babel-node", + "seed": "babel-node utils/seeder.js", + "test": "NODE_ENV=test jest", + "test:watch": "NODE_ENV=test jest --watch", + "test:coverage": "NODE_ENV=test jest --coverage" }, "author": "", "license": "ISC", @@ -12,9 +17,22 @@ "@babel/core": "^7.17.9", "@babel/node": "^7.16.8", "@babel/preset-env": "^7.16.11", + "bcryptjs": "^3.0.2", "cors": "^2.8.5", + "csv-writer": "^1.6.0", + "dotenv": "^17.2.0", "express": "^4.17.3", + "express-rate-limit": "^7.5.1", + "express-validator": "^7.2.1", + "jsonwebtoken": "^9.0.2", "mongoose": "^8.4.0", - "nodemon": "^3.0.1" + "nodemon": "^3.0.1", + "xlsx": "^0.18.5" + }, + "devDependencies": { + "babel-jest": "^30.0.4", + "jest": "^30.0.4", + "mongodb-memory-server": "^10.1.4", + "supertest": "^7.1.3" } -} \ No newline at end of file +} diff --git a/backend/render.yaml b/backend/render.yaml new file mode 100644 index 0000000000..d419268374 --- /dev/null +++ b/backend/render.yaml @@ -0,0 +1,31 @@ +services: + - type: web + name: nanwa-backend + env: node + region: oregon + plan: free + buildCommand: npm install + startCommand: npm start + healthCheckPath: /health + autoDeploy: false + envVars: + - key: NODE_ENV + value: production + - key: PORT + value: 8080 + - key: MONGODB_URI + fromDatabase: + name: nanwa-db + property: connectionString + - key: JWT_SECRET + generateValue: true + - key: RATE_LIMIT + value: 100 + - key: FRONTEND_URL + value: https://nanwa-frontend.onrender.com + +databases: + - name: nanwa-db + databaseName: nanwa + region: oregon + plan: free \ No newline at end of file diff --git a/backend/routes/auth.js b/backend/routes/auth.js new file mode 100644 index 0000000000..57b5841cf0 --- /dev/null +++ b/backend/routes/auth.js @@ -0,0 +1,293 @@ +import express from 'express'; +import { + register, + login, + getProfile, + updateProfile, + logout, + refreshToken, + logoutAll +} from '../controllers/authController.js'; +import { + validateRegister, + validateLogin, + validateProfileUpdate +} from '../middleware/validation.js'; +import { authenticateToken } from '../middleware/auth.js'; +import { authLimiter } from '../middleware/rateLimiter.js'; + +const router = express.Router(); + +/** + * @swagger + * /auth/register: + * post: + * summary: Register a new user + * tags: [Authentication] + * security: [] + * requestBody: + * required: true + * content: + * application/json: + * schema: + * type: object + * required: + * - firstName + * - lastName + * - email + * - password + * - confirmPassword + * properties: + * firstName: + * type: string + * maxLength: 50 + * lastName: + * type: string + * maxLength: 50 + * email: + * type: string + * format: email + * password: + * type: string + * minLength: 6 + * confirmPassword: + * type: string + * minLength: 6 + * responses: + * 201: + * description: User registered successfully + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * message: + * type: string + * data: + * type: object + * properties: + * user: + * $ref: '#/components/schemas/User' + * accessToken: + * type: string + * refreshToken: + * type: string + * 400: + * description: Validation error + * content: + * application/json: + * schema: + * $ref: '#/components/schemas/Error' + * 409: + * description: Email already exists + * 429: + * description: Too many requests + */ +router.post('/register', authLimiter, validateRegister, register); + +/** + * @swagger + * /auth/login: + * post: + * summary: Login user + * tags: [Authentication] + * security: [] + * requestBody: + * required: true + * content: + * application/json: + * schema: + * type: object + * required: + * - email + * - password + * properties: + * email: + * type: string + * format: email + * password: + * type: string + * responses: + * 200: + * description: Login successful + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * data: + * type: object + * properties: + * user: + * $ref: '#/components/schemas/User' + * accessToken: + * type: string + * refreshToken: + * type: string + * 401: + * description: Invalid credentials + * 429: + * description: Too many requests + */ +router.post('/login', authLimiter, validateLogin, login); + + +/** + * @swagger + * /auth/logout: + * post: + * summary: Logout user + * tags: [Authentication] + * security: [] + * requestBody: + * required: true + * content: + * application/json: + * schema: + * type: object + * properties: + * refreshToken: + * type: string + * responses: + * 200: + * description: Logout successful + * content: + * application/json: + * schema: + * $ref: '#/components/schemas/SuccessResponse' + */ +router.post('/logout', logout); + +/** + * @swagger + * /auth/refresh: + * post: + * summary: Refresh access token + * tags: [Authentication] + * security: [] + * requestBody: + * required: true + * content: + * application/json: + * schema: + * type: object + * required: + * - refreshToken + * properties: + * refreshToken: + * type: string + * responses: + * 200: + * description: Token refreshed successfully + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * data: + * type: object + * properties: + * accessToken: + * type: string + * refreshToken: + * type: string + * 401: + * description: Invalid refresh token + */ +router.post('/refresh', authLimiter, refreshToken); + +/** + * @swagger + * /auth/profile: + * get: + * summary: Get user profile + * tags: [Authentication] + * responses: + * 200: + * description: Profile retrieved successfully + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * data: + * $ref: '#/components/schemas/User' + * 401: + * description: Authentication required + */ +router.get('/profile', authenticateToken, getProfile); + +/** + * @swagger + * /auth/profile: + * put: + * summary: Update user profile + * tags: [Authentication] + * requestBody: + * required: true + * content: + * application/json: + * schema: + * type: object + * properties: + * firstName: + * type: string + * maxLength: 50 + * lastName: + * type: string + * maxLength: 50 + * email: + * type: string + * format: email + * responses: + * 200: + * description: Profile updated successfully + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * data: + * $ref: '#/components/schemas/User' + * 400: + * description: Validation error + * 401: + * description: Authentication required + */ +router.put('/profile', authenticateToken, validateProfileUpdate, updateProfile); + +/** + * @swagger + * /auth/logout-all: + * post: + * summary: Logout from all devices + * tags: [Authentication] + * responses: + * 200: + * description: Logged out from all devices + * content: + * application/json: + * schema: + * $ref: '#/components/schemas/SuccessResponse' + * 401: + * description: Authentication required + */ +router.post('/logout-all', authenticateToken, logoutAll); + + +export default router; \ No newline at end of file diff --git a/backend/routes/charts.js b/backend/routes/charts.js new file mode 100644 index 0000000000..073d00d274 --- /dev/null +++ b/backend/routes/charts.js @@ -0,0 +1,172 @@ +import express from 'express'; +import { + getSurvivalRateChart, + getHeightGrowthChart, + getCO2AbsorptionChart +} from '../controllers/chartController.js'; +import { authenticateToken } from '../middleware/auth.js'; +// import { dataLimiter } from '../middleware/rateLimiter.js'; + +const router = express.Router(); + +// All chart routes require authentication +router.use(authenticateToken); + +// Rate limiting disabled for development +// router.use(dataLimiter); + +/** + * @swagger + * components: + * schemas: + * ChartDataPoint: + * type: object + * properties: + * period: + * type: string + * description: Time period (format depends on groupBy parameter) + * value: + * type: number + * description: Data value for this period + */ + +/** + * @swagger + * /charts/survival-rate: + * get: + * summary: Get survival rate chart data over time + * tags: [Charts] + * parameters: + * - in: query + * name: forestId + * schema: + * type: string + * description: Filter by specific forest ID + * - in: query + * name: startDate + * schema: + * type: string + * format: date + * description: Start date for data range + * - in: query + * name: endDate + * schema: + * type: string + * format: date + * description: End date for data range + * - in: query + * name: groupBy + * schema: + * type: string + * enum: [day, week, month, year] + * default: month + * description: Time grouping for data points + * responses: + * 200: + * description: Survival rate chart data + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * data: + * type: object + * properties: + * chartData: + * type: array + * items: + * type: object + * properties: + * period: + * type: string + * totalPlanted: + * type: integer + * surviving: + * type: integer + * dead: + * type: integer + * survivalRate: + * type: number + * format: float + * groupBy: + * type: string + * totalDataPoints: + * type: integer + */ +router.get('/survival-rate', getSurvivalRateChart); + +/** + * @swagger + * /charts/height-growth: + * get: + * summary: Get height growth chart data over time + * tags: [Charts] + * parameters: + * - in: query + * name: forestId + * schema: + * type: string + * description: Filter by specific forest ID + * - in: query + * name: species + * schema: + * type: string + * description: Filter by tree species + * - in: query + * name: startDate + * schema: + * type: string + * format: date + * description: Start date for data range + * - in: query + * name: endDate + * schema: + * type: string + * format: date + * description: End date for data range + * - in: query + * name: groupBy + * schema: + * type: string + * enum: [day, week, month, year] + * default: month + * responses: + * 200: + * description: Height growth chart data + */ +router.get('/height-growth', getHeightGrowthChart); + +/** + * @swagger + * /charts/co2-absorption: + * get: + * summary: Get CO2 absorption chart data over time + * tags: [Charts] + * parameters: + * - in: query + * name: forestId + * schema: + * type: string + * description: Filter by specific forest ID + * - in: query + * name: species + * schema: + * type: string + * description: Filter by tree species + * - in: query + * name: groupBy + * schema: + * type: string + * enum: [day, week, month, year] + * default: month + * responses: + * 200: + * description: CO2 absorption chart data with cumulative totals + */ +router.get('/co2-absorption', getCO2AbsorptionChart); + + +export default router; \ No newline at end of file diff --git a/backend/routes/dashboard.js b/backend/routes/dashboard.js new file mode 100644 index 0000000000..c5ae99ab19 --- /dev/null +++ b/backend/routes/dashboard.js @@ -0,0 +1,182 @@ +import express from 'express'; +import { + getDashboardStats, + getEnhancedDashboardStats +} from '../controllers/dashboardController.js'; +import { authenticateToken } from '../middleware/auth.js'; +// import { dataLimiter } from '../middleware/rateLimiter.js'; + +const router = express.Router(); + + +// All dashboard routes require authentication +router.use(authenticateToken); + +// Rate limiting disabled for development +// router.use(dataLimiter); + +/** + * @swagger + * /dashboard/stats: + * get: + * summary: Get comprehensive dashboard statistics + * tags: [Dashboard] + * parameters: + * - in: query + * name: forestId + * schema: + * type: string + * description: Filter by specific forest ID + * - in: query + * name: startDate + * schema: + * type: string + * format: date + * description: Start date for filtering (YYYY-MM-DD) + * - in: query + * name: endDate + * schema: + * type: string + * format: date + * description: End date for filtering (YYYY-MM-DD) + * - in: query + * name: species + * schema: + * type: string + * description: Filter by tree species + * responses: + * 200: + * description: Dashboard statistics retrieved successfully + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * data: + * type: object + * properties: + * trees: + * type: object + * properties: + * total: + * type: integer + * alive: + * type: integer + * dead: + * type: integer + * survivalRate: + * type: number + * format: float + * forests: + * type: object + * properties: + * total: + * type: integer + * active: + * type: integer + * totalArea: + * type: number + * format: float + * users: + * type: object + * properties: + * total: + * type: integer + * active: + * type: integer + * growth: + * type: object + * properties: + * avgHeight: + * type: number + * format: float + * avgDiameter: + * type: number + * format: float + * totalCO2Absorbed: + * type: number + * format: float + * 401: + * description: Authentication required + * 429: + * description: Too many requests + */ +router.get('/stats', getDashboardStats); + + +/** + * @swagger + * /dashboard/enhanced-stats: + * get: + * summary: Get enhanced dashboard statistics with simulated financial and ecological data + * tags: [Dashboard] + * parameters: + * - in: query + * name: forestId + * schema: + * type: string + * description: Filter by specific forest ID + * - in: query + * name: forestIds + * schema: + * type: string + * description: Filter by multiple forest IDs (comma-separated) + * - in: query + * name: startDate + * schema: + * type: string + * format: date + * description: Start date for filtering (YYYY-MM-DD) + * - in: query + * name: endDate + * schema: + * type: string + * format: date + * description: End date for filtering (YYYY-MM-DD) + * - in: query + * name: species + * schema: + * type: string + * description: Filter by tree species + * - in: query + * name: isAlive + * schema: + * type: boolean + * description: Filter by tree alive status + * responses: + * 200: + * description: Enhanced dashboard statistics with simulations + * content: + * application/json: + * schema: + * type: object + * properties: + * success: + * type: boolean + * example: true + * data: + * type: object + * properties: + * basic: + * type: object + * properties: + * totalTrees: + * type: integer + * aliveTrees: + * type: integer + * survivalRate: + * type: number + * format: float + * investor: + * type: object + * description: Simulated financial metrics + * ecological: + * type: object + * description: Simulated ecological metrics + */ +router.get('/enhanced-stats', getEnhancedDashboardStats); + +export default router; \ No newline at end of file diff --git a/backend/routes/exports.js b/backend/routes/exports.js new file mode 100644 index 0000000000..609bf62da6 --- /dev/null +++ b/backend/routes/exports.js @@ -0,0 +1,21 @@ +import express from 'express'; +import { + exportTreesCSV, + exportTreesXLSX, + exportForestAnalytics +} from '../controllers/exportController.js'; +import { authenticateToken } from '../middleware/auth.js'; + +const router = express.Router(); + +// All export routes require authentication +router.use(authenticateToken); + +// Trees export endpoints +router.get('/trees/csv', exportTreesCSV); +router.get('/trees/xlsx', exportTreesXLSX); + +// Forest analytics export +router.get('/forest-analytics', exportForestAnalytics); + +export default router; \ No newline at end of file diff --git a/backend/routes/forests.js b/backend/routes/forests.js new file mode 100644 index 0000000000..27b63872aa --- /dev/null +++ b/backend/routes/forests.js @@ -0,0 +1,25 @@ +import express from 'express'; +import { + getForests, + getForestById, + createForest, + updateForest, + deleteForest, + getForestAnalytics +} from '../controllers/forestController.js'; +import { validateForestCreate } from '../middleware/validation.js'; +import { authenticateToken, requireAdmin } from '../middleware/auth.js'; + +const router = express.Router(); + +// Public routes (with optional authentication) +router.get('/', getForests); +router.get('/:id', getForestById); +router.get('/:id/analytics', getForestAnalytics); + +// Protected routes (admin only) +router.post('/', authenticateToken, requireAdmin, validateForestCreate, createForest); +router.put('/:id', authenticateToken, requireAdmin, validateForestCreate, updateForest); +router.delete('/:id', authenticateToken, requireAdmin, deleteForest); + +export default router; \ No newline at end of file diff --git a/backend/routes/trees.js b/backend/routes/trees.js new file mode 100644 index 0000000000..ae92826f66 --- /dev/null +++ b/backend/routes/trees.js @@ -0,0 +1,36 @@ +import express from 'express'; +import { + getTrees, + getTreeById, + createTree, + updateTree, + deleteTree, + addMeasurement, + getTreeMeasurements, + getTreesByForest, + markTreeDead +} from '../controllers/treeController.js'; +import { + validateTreeCreate, + validateMeasurement +} from '../middleware/validation.js'; +import { authenticateToken, requireAdmin } from '../middleware/auth.js'; + +const router = express.Router(); + +// Public routes +router.get('/', getTrees); +router.get('/:id', getTreeById); +router.get('/:id/measurements', getTreeMeasurements); +router.get('/forest/:forestId', getTreesByForest); + +// Protected routes (authenticated users) +router.post('/:id/measurements', authenticateToken, validateMeasurement, addMeasurement); + +// Admin only routes +router.post('/', authenticateToken, requireAdmin, validateTreeCreate, createTree); +router.put('/:id', authenticateToken, requireAdmin, updateTree); +router.delete('/:id', authenticateToken, requireAdmin, deleteTree); +router.patch('/:id/mark-dead', authenticateToken, requireAdmin, markTreeDead); + +export default router; \ No newline at end of file diff --git a/backend/scripts/populateEnhancedMetrics.js b/backend/scripts/populateEnhancedMetrics.js new file mode 100644 index 0000000000..1356105a6d --- /dev/null +++ b/backend/scripts/populateEnhancedMetrics.js @@ -0,0 +1,154 @@ +import mongoose from 'mongoose'; +import dotenv from 'dotenv'; +import { Forest, Tree } from '../models/index.js'; + +dotenv.config(); + +const populateEnhancedMetrics = async () => { + try { + // Connect to MongoDB + await mongoose.connect(process.env.MONGO_URL); + console.log('Connected to MongoDB'); + + // Update all forests with sample financial and environmental data + const forests = await Forest.find({}); + console.log(`Found ${forests.length} forests to update`); + + for (const forest of forests) { + // Add financial data if missing + if (!forest.financials || !forest.financials.currentValue) { + forest.financials = { + acquisitionCost: Math.floor(Math.random() * 5000000) + 1000000, // 1M - 6M SEK + acquisitionDate: new Date(2020, 0, 1), + currentValue: Math.floor(Math.random() * 7000000) + 1500000, // 1.5M - 8.5M SEK + lastValuationDate: new Date(), + currency: 'SEK', + maintenanceBudget: { + annual: Math.floor(Math.random() * 500000) + 100000, + allocated: Math.floor(Math.random() * 400000) + 80000, + spent: Math.floor(Math.random() * 300000) + 50000 + } + }; + } + + // Add carbon metrics if missing + if (!forest.carbonMetrics || !forest.carbonMetrics.totalCarbonStored) { + forest.carbonMetrics = { + totalCarbonStored: Math.floor(Math.random() * 10000) + 1000, // tons + annualSequestration: Math.floor(Math.random() * 500) + 50, + carbonCredits: { + issued: Math.floor(Math.random() * 1000) + 100, + sold: Math.floor(Math.random() * 500), + available: Math.floor(Math.random() * 500) + 100, + pricePerCredit: Math.floor(Math.random() * 500) + 100 // 100-600 SEK + }, + lastCalculation: new Date(), + calculationMethod: 'IPCC Standard' + }; + } + + // Add biodiversity data if missing + if (!forest.biodiversity || !forest.biodiversity.biodiversityIndex) { + forest.biodiversity = { + biodiversityIndex: Math.floor(Math.random() * 60) + 40, // 40-100 + species: [ + { + name: 'Pine', + scientificName: 'Pinus sylvestris', + category: 'flora', + count: Math.floor(Math.random() * 1000) + 100, + conservationStatus: 'common' + }, + { + name: 'Spruce', + scientificName: 'Picea abies', + category: 'flora', + count: Math.floor(Math.random() * 800) + 100, + conservationStatus: 'common' + }, + { + name: 'Brown Bear', + scientificName: 'Ursus arctos', + category: 'fauna', + count: Math.floor(Math.random() * 5) + 1, + conservationStatus: 'rare' + } + ], + lastSurveyDate: new Date() + }; + } + + // Add environmental data if missing + if (!forest.environmental || !forest.environmental.fireRisk) { + const riskLevels = ['low', 'moderate', 'high', 'extreme']; + forest.environmental = { + fireRisk: { + current: riskLevels[Math.floor(Math.random() * riskLevels.length)], + lastAssessment: new Date(), + mitigationMeasures: ['Firebreaks maintained', 'Regular monitoring', 'Water sources mapped'] + }, + soilHealth: { + phLevel: Math.random() * 2 + 5.5, // 5.5 - 7.5 + organicMatter: Math.random() * 5 + 2, // 2-7% + compaction: ['low', 'moderate', 'high'][Math.floor(Math.random() * 3)], + erosionRisk: ['low', 'moderate', 'high'][Math.floor(Math.random() * 3)], + lastTested: new Date() + } + }; + } + + await forest.save(); + console.log(`Updated forest: ${forest.name}`); + } + + // Update trees with economic value data + const trees = await Tree.find({}).limit(1000); // Update first 1000 trees + console.log(`Found ${trees.length} trees to update`); + + for (const tree of trees) { + // Add economic value if missing + if (!tree.economicValue || !tree.economicValue.currentTimberValue) { + tree.economicValue = { + currentTimberValue: Math.floor(Math.random() * 5000) + 500, // 500-5500 SEK per tree + carbonCreditValue: Math.floor(Math.random() * 200) + 50, // 50-250 SEK + lastValuation: new Date(), + valuationMethod: 'Market-based', + marketPricePerCubicMeter: Math.floor(Math.random() * 200) + 400 // 400-600 SEK/m³ + }; + } + + // Ensure latest measurement has health status + if (tree.measurements && tree.measurements.length > 0) { + const latestMeasurement = tree.measurements[tree.measurements.length - 1]; + if (!latestMeasurement.healthStatus) { + const healthStatuses = ['excellent', 'good', 'fair', 'poor', 'critical']; + // Weighted towards good health + const weights = [0.2, 0.4, 0.25, 0.1, 0.05]; + const random = Math.random(); + let sum = 0; + for (let i = 0; i < weights.length; i++) { + sum += weights[i]; + if (random < sum) { + latestMeasurement.healthStatus = healthStatuses[i]; + break; + } + } + } + } + + await tree.save(); + } + + console.log('✅ Successfully populated enhanced metrics'); + console.log(`Updated ${forests.length} forests and ${trees.length} trees`); + + } catch (error) { + console.error('Error populating enhanced metrics:', error); + } finally { + await mongoose.disconnect(); + console.log('Disconnected from MongoDB'); + } +}; + +// Run the script +populateEnhancedMetrics(); \ No newline at end of file diff --git a/backend/server.js b/backend/server.js index 070c875189..bc50082fb6 100644 --- a/backend/server.js +++ b/backend/server.js @@ -1,22 +1,192 @@ import express from "express"; import cors from "cors"; import mongoose from "mongoose"; +import dotenv from "dotenv"; +import { createServer } from "http"; + +// Import routes +import authRoutes from "./routes/auth.js"; +import forestRoutes from "./routes/forests.js"; +import treeRoutes from "./routes/trees.js"; +import exportRoutes from "./routes/exports.js"; +import dashboardRoutes from "./routes/dashboard.js"; +import chartRoutes from "./routes/charts.js"; + +// Import middleware +// import { generalLimiter } from "./middleware/rateLimiter.js"; // TEMPORARILY DISABLED FOR DEVELOPMENT + + +// Load environment variables +dotenv.config(); + +// Debug environment variables +console.log('🔧 Environment check:'); +console.log('- NODE_ENV:', process.env.NODE_ENV); +console.log('- PORT:', process.env.PORT); +console.log('- MONGO_URL:', process.env.MONGO_URL ? 'Set' : 'Missing'); +console.log('- JWT_SECRET:', process.env.JWT_SECRET ? 'Set' : 'Missing'); + +// MongoDB connection +const mongoUrl = process.env.MONGO_URL || "mongodb://localhost/nanwa-forestry"; +console.log('🔗 Connecting to MongoDB:', mongoUrl.includes('mongodb+srv') ? 'Atlas connection' : 'Local connection'); + +// Don't crash the app if MongoDB fails initially - let it retry +mongoose.connect(mongoUrl) + .then(() => console.log('✅ MongoDB connected successfully')) + .catch((err) => { + console.error('❌ MongoDB connection failed:', err.message); + console.log('⚠️ Server will continue running but database operations will fail'); + // Don't exit - let the app start and show the error in health check + }); -const mongoUrl = process.env.MONGO_URL || "mongodb://localhost/final-project"; -mongoose.connect(mongoUrl); mongoose.Promise = Promise; +// MongoDB connection events +mongoose.connection.on('connected', () => { + console.log('✅ Connected to MongoDB'); +}); + +mongoose.connection.on('error', (err) => { + console.error('❌ MongoDB connection error:', err); +}); + +mongoose.connection.on('disconnected', () => { + console.log('📴 Disconnected from MongoDB'); +}); + const port = process.env.PORT || 8080; const app = express(); -app.use(cors()); -app.use(express.json()); +// Configure trust proxy for production deployment (Render, Heroku, etc.) +if (process.env.NODE_ENV === 'production') { + app.set('trust proxy', 1); +} + +const server = createServer(app); + +// Define allowed origins for CORS +const allowedOrigins = [ + "http://localhost:3000", + "http://localhost:5173", // Vite dev server + "https://entitree.netlify.app", + process.env.FRONTEND_URL +].filter(Boolean); + +console.log('🔗 Allowed CORS origins:', allowedOrigins); + + +// CORS configuration - simplified for debugging +app.use(cors({ + origin: true, // Allow all origins temporarily + credentials: true, + methods: ['GET', 'POST', 'PUT', 'DELETE', 'OPTIONS'], + allowedHeaders: ['Content-Type', 'Authorization'] +})); +app.use(express.json({ limit: '10mb' })); +app.use(express.urlencoded({ extended: true, limit: '10mb' })); + +// Rate limiting middleware - TEMPORARILY DISABLED FOR DEVELOPMENT +// app.use('/api/', generalLimiter); // TODO: RE-ENABLE IN PRODUCTION FOR SECURITY +// Request logging middleware +app.use((req, res, next) => { + console.log(`${new Date().toISOString()} - ${req.method} ${req.path}`); + next(); +}); + +// Health check endpoint app.get("/", (req, res) => { - res.send("Hello Technigo!"); + res.json({ + success: true, + message: "Nanwa Forestry API is running", + version: "1.0.0", + timestamp: new Date().toISOString() + }); +}); + +app.get("/health", (req, res) => { + res.json({ + success: true, + status: "healthy", + database: mongoose.connection.readyState === 1 ? "connected" : "disconnected", + uptime: process.uptime(), + timestamp: new Date().toISOString() + }); +}); + +// Health check endpoints +app.get('/', (req, res) => { + res.json({ + status: 'Server is running', + timestamp: new Date().toISOString(), + environment: process.env.NODE_ENV, + mongodb: mongoose.connection.readyState === 1 ? 'Connected' : 'Disconnected' + }); +}); + +app.get('/api', (req, res) => { + res.json({ + status: 'API is running', + timestamp: new Date().toISOString(), + mongodb: mongoose.connection.readyState === 1 ? 'Connected' : 'Disconnected', + environment: { + NODE_ENV: process.env.NODE_ENV, + MONGO_URL: process.env.MONGO_URL ? 'Set' : 'Missing', + JWT_SECRET: process.env.JWT_SECRET ? 'Set' : 'Missing', + PORT: process.env.PORT + } + }); +}); + + + +// API Routes +app.use("/api/auth", authRoutes); +app.use("/api/forests", forestRoutes); +app.use("/api/trees", treeRoutes); +app.use("/api/exports", exportRoutes); +app.use("/api/dashboard", dashboardRoutes); +app.use("/api/charts", chartRoutes); + +// 404 handler +app.use("*", (req, res) => { + res.status(404).json({ + success: false, + message: "Endpoint not found", + path: req.originalUrl + }); +}); + +// Global error handler +app.use((error, req, res, next) => { + console.error('Global error handler:', error); + + res.status(error.status || 500).json({ + success: false, + message: error.message || "Internal server error", + error: process.env.NODE_ENV === 'development' ? error.stack : undefined + }); +}); + +// Graceful shutdown +process.on('SIGTERM', () => { + console.log('👋 SIGTERM received, shutting down gracefully'); + mongoose.connection.close(() => { + console.log('📴 MongoDB connection closed'); + process.exit(0); + }); +}); + +process.on('SIGINT', () => { + console.log('👋 SIGINT received, shutting down gracefully'); + mongoose.connection.close(() => { + console.log('📴 MongoDB connection closed'); + process.exit(0); + }); }); // Start the server -app.listen(port, () => { - console.log(`Server running on http://localhost:${port}`); +server.listen(port, () => { + console.log(`🚀 Server running on http://localhost:${port}`); + console.log(`📊 Environment: ${process.env.NODE_ENV || 'development'}`); }); diff --git a/backend/tests/controllers/auth.test.js b/backend/tests/controllers/auth.test.js new file mode 100644 index 0000000000..7644040225 --- /dev/null +++ b/backend/tests/controllers/auth.test.js @@ -0,0 +1,235 @@ +import request from 'supertest'; +import express from 'express'; +import authRoutes from '../../routes/auth.js'; +import { createTestUser, expectValidationError } from '../helpers/testHelpers.js'; + +// Create test app +const app = express(); +app.use(express.json()); +app.use('/api/auth', authRoutes); + +describe('Auth Controller', () => { + describe('POST /api/auth/register', () => { + it('should register a new user successfully', async () => { + const userData = { + firstName: 'Jane', + lastName: 'Smith', + email: 'jane.smith@test.com', + password: 'password123', + confirmPassword: 'password123' + }; + + const response = await request(app) + .post('/api/auth/register') + .send(userData); + + expect(response.status).toBe(201); + expect(response.body.success).toBe(true); + expect(response.body.data.user).toBeDefined(); + expect(response.body.data.user.email).toBe(userData.email); + expect(response.body.data.user.password).toBeUndefined(); + expect(response.body.data.accessToken).toBeDefined(); + expect(response.body.data.refreshToken).toBeDefined(); + }); + + it('should fail with invalid email format', async () => { + const userData = { + firstName: 'Jane', + lastName: 'Smith', + email: 'invalid-email', + password: 'password123', + confirmPassword: 'password123' + }; + + const response = await request(app) + .post('/api/auth/register') + .send(userData); + + expectValidationError(response, 'email'); + }); + + it('should fail when passwords do not match', async () => { + const userData = { + firstName: 'Jane', + lastName: 'Smith', + email: 'jane.smith@test.com', + password: 'password123', + confirmPassword: 'different-password' + }; + + const response = await request(app) + .post('/api/auth/register') + .send(userData); + + expect(response.status).toBe(400); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('match'); + }); + + it('should fail when user already exists', async () => { + await createTestUser({ email: 'existing@test.com' }); + + const userData = { + firstName: 'Jane', + lastName: 'Smith', + email: 'existing@test.com', + password: 'password123', + confirmPassword: 'password123' + }; + + const response = await request(app) + .post('/api/auth/register') + .send(userData); + + expect(response.status).toBe(409); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('already exists'); + }); + }); + + describe('POST /api/auth/login', () => { + beforeEach(async () => { + await createTestUser({ + email: 'test@example.com', + password: 'password123' + }); + }); + + it('should login successfully with valid credentials', async () => { + const response = await request(app) + .post('/api/auth/login') + .send({ + email: 'test@example.com', + password: 'password123' + }); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data.user).toBeDefined(); + expect(response.body.data.accessToken).toBeDefined(); + expect(response.body.data.refreshToken).toBeDefined(); + }); + + it('should fail with invalid email', async () => { + const response = await request(app) + .post('/api/auth/login') + .send({ + email: 'nonexistent@example.com', + password: 'password123' + }); + + expect(response.status).toBe(401); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('Invalid credentials'); + }); + + it('should fail with invalid password', async () => { + const response = await request(app) + .post('/api/auth/login') + .send({ + email: 'test@example.com', + password: 'wrongpassword' + }); + + expect(response.status).toBe(401); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('Invalid credentials'); + }); + + it('should fail with missing credentials', async () => { + const response = await request(app) + .post('/api/auth/login') + .send({}); + + expectValidationError(response); + }); + }); + + describe('POST /api/auth/refresh', () => { + let refreshToken; + + beforeEach(async () => { + const user = await createTestUser({ + email: 'refresh-test@example.com', + password: 'password123' + }); + + const loginResponse = await request(app) + .post('/api/auth/login') + .send({ + email: 'refresh-test@example.com', + password: 'password123' + }); + + refreshToken = loginResponse.body.data.refreshToken; + }); + + it('should refresh token successfully', async () => { + const response = await request(app) + .post('/api/auth/refresh') + .send({ refreshToken }); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data.accessToken).toBeDefined(); + expect(response.body.data.refreshToken).toBeDefined(); + }); + + it('should fail with invalid refresh token', async () => { + const response = await request(app) + .post('/api/auth/refresh') + .send({ refreshToken: 'invalid-token' }); + + expect(response.status).toBe(401); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('Invalid refresh token'); + }); + + it('should fail with missing refresh token', async () => { + const response = await request(app) + .post('/api/auth/refresh') + .send({}); + + expectValidationError(response, 'refreshToken'); + }); + }); + + describe('POST /api/auth/logout', () => { + let refreshToken; + + beforeEach(async () => { + const user = await createTestUser({ + email: 'logout-test@example.com', + password: 'password123' + }); + + const loginResponse = await request(app) + .post('/api/auth/login') + .send({ + email: 'logout-test@example.com', + password: 'password123' + }); + + refreshToken = loginResponse.body.data.refreshToken; + }); + + it('should logout successfully', async () => { + const response = await request(app) + .post('/api/auth/logout') + .send({ refreshToken }); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.message).toContain('Logged out successfully'); + }); + + it('should handle logout with invalid token gracefully', async () => { + const response = await request(app) + .post('/api/auth/logout') + .send({ refreshToken: 'invalid-token' }); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + }); + }); +}); \ No newline at end of file diff --git a/backend/tests/controllers/charts.test.js b/backend/tests/controllers/charts.test.js new file mode 100644 index 0000000000..a10c034aa9 --- /dev/null +++ b/backend/tests/controllers/charts.test.js @@ -0,0 +1,240 @@ +import request from 'supertest'; +import express from 'express'; +import chartRoutes from '../../routes/charts.js'; +import { createTestUser, createTestTrees, createTestTree, generateTestToken } from '../helpers/testHelpers.js'; + +// Create test app +const app = express(); +app.use(express.json()); +app.use('/api/charts', chartRoutes); + +describe('Chart Controller', () => { + let user, authToken; + + beforeEach(async () => { + user = await createTestUser(); + authToken = generateTestToken(user._id, user.role); + + // Create test trees with measurements for chart data + await createTestTrees(5); + }); + + describe('GET /api/charts/survival-rate', () => { + it('should return survival rate chart data', async () => { + const response = await request(app) + .get('/api/charts/survival-rate') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data).toBeDefined(); + expect(response.body.data.chartData).toBeInstanceOf(Array); + expect(response.body.data.groupBy).toBeDefined(); + expect(response.body.data.totalDataPoints).toBeGreaterThanOrEqual(0); + }); + + it('should support different groupBy parameters', async () => { + const response = await request(app) + .get('/api/charts/survival-rate') + .query({ groupBy: 'week' }) + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.data.groupBy).toBe('week'); + }); + + it('should filter by forest ID', async () => { + const trees = await createTestTrees(3); + const forestId = trees[0].forestId; + + const response = await request(app) + .get('/api/charts/survival-rate') + .query({ forestId: forestId.toString() }) + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + }); + + it('should fail without authentication', async () => { + const response = await request(app) + .get('/api/charts/survival-rate'); + + expect(response.status).toBe(401); + }); + }); + + describe('GET /api/charts/height-growth', () => { + beforeEach(async () => { + // Create tree with multiple measurements for height growth + await createTestTree({ + measurements: [ + { + height: 1.5, + diameter: 0.08, + healthStatus: 'healthy', + measuredAt: new Date('2023-01-01'), + co2Absorption: 8.0 + }, + { + height: 2.1, + diameter: 0.12, + healthStatus: 'healthy', + measuredAt: new Date('2023-06-01'), + co2Absorption: 12.5 + } + ] + }); + }); + + it('should return height growth chart data', async () => { + const response = await request(app) + .get('/api/charts/height-growth') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data.chartData).toBeInstanceOf(Array); + }); + + it('should filter by species', async () => { + const response = await request(app) + .get('/api/charts/height-growth') + .query({ species: 'Pine' }) + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.data.filters.species).toBe('Pine'); + }); + + it('should include date range filtering', async () => { + const startDate = '2023-01-01'; + const endDate = '2023-12-31'; + + const response = await request(app) + .get('/api/charts/height-growth') + .query({ startDate, endDate }) + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.data.filters.startDate).toBe(startDate); + expect(response.body.data.filters.endDate).toBe(endDate); + }); + }); + + describe('GET /api/charts/co2-absorption', () => { + it('should return CO2 absorption chart data', async () => { + const response = await request(app) + .get('/api/charts/co2-absorption') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data.chartData).toBeInstanceOf(Array); + expect(response.body.data.summary).toBeDefined(); + }); + + it('should include cumulative CO2 data', async () => { + const response = await request(app) + .get('/api/charts/co2-absorption') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + if (response.body.data.chartData.length > 0) { + expect(response.body.data.chartData[0]).toHaveProperty('cumulativeCO2'); + } + }); + + it('should provide summary statistics', async () => { + const response = await request(app) + .get('/api/charts/co2-absorption') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.data.summary).toHaveProperty('totalCO2Absorption'); + expect(response.body.data.summary).toHaveProperty('totalMeasurements'); + expect(response.body.data.summary).toHaveProperty('uniqueTrees'); + }); + }); + + describe('GET /api/charts/health-status', () => { + it('should return health status distribution chart data', async () => { + const response = await request(app) + .get('/api/charts/health-status') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data.chartData).toBeInstanceOf(Array); + }); + + it('should include health distribution breakdown', async () => { + const response = await request(app) + .get('/api/charts/health-status') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + if (response.body.data.chartData.length > 0) { + expect(response.body.data.chartData[0]).toHaveProperty('healthDistribution'); + expect(response.body.data.chartData[0]).toHaveProperty('totalMeasurements'); + } + }); + }); + + describe('GET /api/charts/combined', () => { + it('should return combined chart data for dashboard', async () => { + const response = await request(app) + .get('/api/charts/combined') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data).toHaveProperty('survivalRate'); + expect(response.body.data).toHaveProperty('heightGrowth'); + expect(response.body.data).toHaveProperty('co2Absorption'); + expect(response.body.data).toHaveProperty('healthStatus'); + expect(response.body.data).toHaveProperty('generatedAt'); + }); + + it('should include filters in combined response', async () => { + const forestId = (await createTestTrees(2))[0].forestId; + + const response = await request(app) + .get('/api/charts/combined') + .query({ forestId: forestId.toString(), groupBy: 'week' }) + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.data.filters.forestId).toBe(forestId.toString()); + expect(response.body.data.groupBy).toBe('week'); + }); + + it('should handle empty data gracefully', async () => { + // Clear all test data + const { Tree } = await import('../../models/index.js'); + await Tree.deleteMany({}); + + const response = await request(app) + .get('/api/charts/combined') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + }); + }); + + describe('Rate Limiting', () => { + it('should apply rate limiting to chart endpoints', async () => { + const requests = Array(15).fill().map(() => + request(app) + .get('/api/charts/survival-rate') + .set('Authorization', `Bearer ${authToken}`) + ); + + const responses = await Promise.all(requests); + const rateLimitedResponses = responses.filter(r => r.status === 429); + + expect(rateLimitedResponses.length).toBeGreaterThan(0); + }); + }); +}); \ No newline at end of file diff --git a/backend/tests/controllers/dashboard.test.js b/backend/tests/controllers/dashboard.test.js new file mode 100644 index 0000000000..b946903a3e --- /dev/null +++ b/backend/tests/controllers/dashboard.test.js @@ -0,0 +1,129 @@ +import request from 'supertest'; +import express from 'express'; +import dashboardRoutes from '../../routes/dashboard.js'; +import { createTestUser, createTestTrees, generateTestToken } from '../helpers/testHelpers.js'; + +// Create test app +const app = express(); +app.use(express.json()); +app.use('/api/dashboard', dashboardRoutes); + +describe('Dashboard Controller', () => { + let user, authToken; + + beforeEach(async () => { + user = await createTestUser(); + authToken = generateTestToken(user._id, user.role); + + // Create test data for dashboard stats + await createTestTrees(10); + }); + + describe('GET /api/dashboard/stats', () => { + it('should return dashboard statistics for authenticated user', async () => { + const response = await request(app) + .get('/api/dashboard/stats') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data).toBeDefined(); + expect(response.body.data.trees).toBeDefined(); + expect(response.body.data.forests).toBeDefined(); + expect(response.body.data.users).toBeDefined(); + expect(response.body.data.trees.total).toBeGreaterThan(0); + expect(response.body.data.trees.alive).toBeDefined(); + expect(response.body.data.trees.dead).toBeDefined(); + expect(response.body.data.trees.survivalRate).toBeDefined(); + }); + + it('should fail without authentication', async () => { + const response = await request(app) + .get('/api/dashboard/stats'); + + expect(response.status).toBe(401); + expect(response.body.success).toBe(false); + }); + + it('should return filtered stats by forest', async () => { + const trees = await createTestTrees(5); + const forestId = trees[0].forestId; + + const response = await request(app) + .get('/api/dashboard/stats') + .query({ forestId: forestId.toString() }) + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data.filters.forestId).toBe(forestId.toString()); + }); + + it('should return stats within date range', async () => { + const startDate = '2021-01-01'; + const endDate = '2023-12-31'; + + const response = await request(app) + .get('/api/dashboard/stats') + .query({ startDate, endDate }) + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data.filters.startDate).toBe(startDate); + expect(response.body.data.filters.endDate).toBe(endDate); + }); + }); + + describe('GET /api/dashboard/overview', () => { + it('should return comprehensive dashboard overview', async () => { + const response = await request(app) + .get('/api/dashboard/overview') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.success).toBe(true); + expect(response.body.data).toBeDefined(); + expect(response.body.data.summary).toBeDefined(); + expect(response.body.data.recentActivities).toBeDefined(); + expect(response.body.data.topPerformingForests).toBeDefined(); + expect(response.body.data.speciesDistribution).toBeDefined(); + }); + + it('should include recent activities', async () => { + const response = await request(app) + .get('/api/dashboard/overview') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.data.recentActivities).toBeInstanceOf(Array); + }); + + it('should include species distribution', async () => { + const response = await request(app) + .get('/api/dashboard/overview') + .set('Authorization', `Bearer ${authToken}`); + + expect(response.status).toBe(200); + expect(response.body.data.speciesDistribution).toBeInstanceOf(Array); + expect(response.body.data.speciesDistribution.length).toBeGreaterThan(0); + }); + }); + + describe('Rate Limiting', () => { + it('should apply rate limiting to dashboard endpoints', async () => { + // Make multiple rapid requests to trigger rate limiting + const requests = Array(20).fill().map(() => + request(app) + .get('/api/dashboard/stats') + .set('Authorization', `Bearer ${authToken}`) + ); + + const responses = await Promise.all(requests); + const rateLimitedResponses = responses.filter(r => r.status === 429); + + // At least some requests should be rate limited + expect(rateLimitedResponses.length).toBeGreaterThan(0); + }); + }); +}); \ No newline at end of file diff --git a/backend/tests/helpers/testHelpers.js b/backend/tests/helpers/testHelpers.js new file mode 100644 index 0000000000..3129968a8a --- /dev/null +++ b/backend/tests/helpers/testHelpers.js @@ -0,0 +1,136 @@ +import jwt from 'jsonwebtoken'; +import bcrypt from 'bcryptjs'; +import { User, Forest, Tree } from '../../models/index.js'; + +// Create test user with hashed password +export const createTestUser = async (userData = {}) => { + const defaultUser = { + firstName: 'John', + lastName: 'Doe', + email: 'john.doe@test.com', + password: 'password123', + role: 'user', + isActive: true + }; + + const user = new User({ ...defaultUser, ...userData }); + await user.save(); + return user; +}; + +// Create test admin user +export const createTestAdmin = async (userData = {}) => { + return createTestUser({ + ...userData, + role: 'admin', + email: 'admin@test.com' + }); +}; + +// Generate valid JWT token for testing +export const generateTestToken = (userId, role = 'user') => { + return jwt.sign( + { userId, role }, + process.env.JWT_SECRET, + { expiresIn: '1h' } + ); +}; + +// Create test forest +export const createTestForest = async (forestData = {}) => { + const defaultForest = { + name: 'Test Forest', + region: 'Test Region', // Added required region field + location: { + type: 'Point', + coordinates: [18.0686, 59.3293] // Stockholm coordinates + }, + description: 'A test forest for unit testing', + area: 100.5, + establishedDate: new Date('2020-01-01'), + isActive: true + }; + + const forest = new Forest({ ...defaultForest, ...forestData }); + await forest.save(); + return forest; +}; + +// Create test tree +export const createTestTree = async (treeData = {}) => { + const forest = treeData.forestId || await createTestForest(); + + const defaultTree = { + forestId: forest._id || forest, + species: 'Pine', + plantedDate: new Date('2021-01-01'), + location: { + type: 'Point', + coordinates: [18.0686, 59.3293] + }, + isAlive: true, + measurements: [{ + height: 2.5, + diameter: 0.1, + healthStatus: 'healthy', + co2Absorption: 10.5, + measuredAt: new Date(), + notes: 'Initial measurement' + }] + }; + + const tree = new Tree({ ...defaultTree, ...treeData }); + await tree.save(); + return tree; +}; + +// Create multiple test trees for bulk operations +export const createTestTrees = async (count = 5, forestId = null) => { + const forest = forestId || (await createTestForest())._id; + const trees = []; + + for (let i = 0; i < count; i++) { + const tree = await createTestTree({ + forestId: forest, + species: i % 2 === 0 ? 'Pine' : 'Oak', + isAlive: i < Math.floor(count * 0.8) // 80% survival rate + }); + trees.push(tree); + } + + return trees; +}; + +// Clean specific collections +export const cleanCollection = async (collectionName) => { + const collection = mongoose.connection.collections[collectionName]; + if (collection) { + await collection.deleteMany({}); + } +}; + +// Helper for API error validation +export const expectValidationError = (response, field = null) => { + expect(response.status).toBe(400); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('validation'); + + if (field) { + expect(response.body.errors).toBeDefined(); + expect(response.body.errors.some(err => err.path === field)).toBe(true); + } +}; + +// Helper for API authentication error +export const expectAuthError = (response) => { + expect(response.status).toBe(401); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('authentication'); +}; + +// Helper for API authorization error +export const expectAuthorizationError = (response) => { + expect(response.status).toBe(403); + expect(response.body.success).toBe(false); + expect(response.body.message).toContain('authorization'); +}; \ No newline at end of file diff --git a/backend/tests/middleware/auth.test.js b/backend/tests/middleware/auth.test.js new file mode 100644 index 0000000000..05bd158d88 --- /dev/null +++ b/backend/tests/middleware/auth.test.js @@ -0,0 +1,178 @@ +import jwt from 'jsonwebtoken'; +import { authenticateToken, requireAdmin } from '../../middleware/auth.js'; +import { createTestUser, generateTestToken } from '../helpers/testHelpers.js'; + +describe('Auth Middleware', () => { + let mockReq, mockRes, mockNext; + + beforeEach(() => { + mockReq = { + headers: {}, + user: null + }; + + mockRes = { + status: jest.fn().mockReturnThis(), + json: jest.fn().mockReturnThis() + }; + + mockNext = jest.fn(); + }); + + describe('authenticateToken', () => { + let user, validToken; + + beforeEach(async () => { + user = await createTestUser(); + validToken = generateTestToken(user._id, user.role); + }); + + it('should authenticate valid token', async () => { + mockReq.headers.authorization = `Bearer ${validToken}`; + + await authenticateToken(mockReq, mockRes, mockNext); + + expect(mockNext).toHaveBeenCalledWith(); + expect(mockReq.user).toBeDefined(); + expect(mockReq.user._id.toString()).toBe(user._id.toString()); + expect(mockReq.user.role).toBe(user.role); + }); + + it('should reject request without authorization header', async () => { + await authenticateToken(mockReq, mockRes, mockNext); + + expect(mockRes.status).toHaveBeenCalledWith(401); + expect(mockRes.json).toHaveBeenCalledWith({ + success: false, + message: expect.stringContaining('token') + }); + expect(mockNext).not.toHaveBeenCalled(); + }); + + it('should reject malformed authorization header', async () => { + mockReq.headers.authorization = 'InvalidFormat'; + + await authenticateToken(mockReq, mockRes, mockNext); + + expect(mockRes.status).toHaveBeenCalledWith(401); + expect(mockRes.json).toHaveBeenCalledWith({ + success: false, + message: 'Invalid token format' + }); + expect(mockNext).not.toHaveBeenCalled(); + }); + + it('should reject invalid token', async () => { + mockReq.headers.authorization = 'Bearer invalid-token'; + + await authenticateToken(mockReq, mockRes, mockNext); + + expect(mockRes.status).toHaveBeenCalledWith(401); + expect(mockRes.json).toHaveBeenCalledWith({ + success: false, + message: 'Invalid or expired token' + }); + expect(mockNext).not.toHaveBeenCalled(); + }); + + it('should reject expired token', async () => { + const expiredToken = jwt.sign( + { userId: user._id, role: user.role }, + process.env.JWT_SECRET, + { expiresIn: '-1h' } // Already expired + ); + + mockReq.headers.authorization = `Bearer ${expiredToken}`; + + await authenticateToken(mockReq, mockRes, mockNext); + + expect(mockRes.status).toHaveBeenCalledWith(401); + expect(mockRes.json).toHaveBeenCalledWith({ + success: false, + message: 'Invalid or expired token' + }); + expect(mockNext).not.toHaveBeenCalled(); + }); + + it('should handle token without Bearer prefix', async () => { + mockReq.headers.authorization = validToken; + + await authenticateToken(mockReq, mockRes, mockNext); + + expect(mockRes.status).toHaveBeenCalledWith(401); + expect(mockRes.json).toHaveBeenCalledWith({ + success: false, + message: 'Invalid token format' + }); + expect(mockNext).not.toHaveBeenCalled(); + }); + }); + + describe('requireAdmin', () => { + let user, adminUser; + + beforeEach(async () => { + user = await createTestUser({ role: 'user' }); + adminUser = await createTestUser({ + email: 'admin@test.com', + role: 'admin' + }); + }); + + it('should allow access for admin user', async () => { + mockReq.user = adminUser; + + await requireAdmin(mockReq, mockRes, mockNext); + + expect(mockNext).toHaveBeenCalledWith(); + expect(mockRes.status).not.toHaveBeenCalled(); + }); + + it('should deny access for non-admin user', async () => { + mockReq.user = user; + + await requireAdmin(mockReq, mockRes, mockNext); + + expect(mockRes.status).toHaveBeenCalledWith(403); + expect(mockRes.json).toHaveBeenCalledWith({ + success: false, + message: 'Admin access required' + }); + expect(mockNext).not.toHaveBeenCalled(); + }); + + it('should deny access when no user in request', async () => { + // No mockReq.user set + + await requireAdmin(mockReq, mockRes, mockNext); + + expect(mockRes.status).toHaveBeenCalledWith(401); + expect(mockRes.json).toHaveBeenCalledWith({ + success: false, + message: 'Authentication required' + }); + expect(mockNext).not.toHaveBeenCalled(); + }); + }); + + describe('Middleware Integration', () => { + it('should work together in sequence', async () => { + const user = await createTestUser({ role: 'admin' }); + const token = generateTestToken(user._id, user.role); + + mockReq.headers.authorization = `Bearer ${token}`; + + // First authenticate + await authenticateToken(mockReq, mockRes, mockNext); + expect(mockNext).toHaveBeenCalledTimes(1); + expect(mockReq.user).toBeDefined(); + + // Reset next function + mockNext.mockClear(); + + // Then check admin role + await requireAdmin(mockReq, mockRes, mockNext); + expect(mockNext).toHaveBeenCalledTimes(1); + }); + }); +}); \ No newline at end of file diff --git a/backend/tests/models/user.test.js b/backend/tests/models/user.test.js new file mode 100644 index 0000000000..cae4c38d89 --- /dev/null +++ b/backend/tests/models/user.test.js @@ -0,0 +1,234 @@ +import { User } from '../../models/index.js'; +import bcrypt from 'bcryptjs'; + +describe('User Model', () => { + describe('User Creation', () => { + it('should create a user with valid data', async () => { + const userData = { + firstName: 'John', + lastName: 'Doe', + email: 'john.doe@test.com', + password: 'password123', + role: 'user' + }; + + const user = new User(userData); + const savedUser = await user.save(); + + expect(savedUser._id).toBeDefined(); + expect(savedUser.firstName).toBe(userData.firstName); + expect(savedUser.lastName).toBe(userData.lastName); + expect(savedUser.email).toBe(userData.email); + expect(savedUser.role).toBe(userData.role); + expect(savedUser.isActive).toBe(true); // Default value + expect(savedUser.createdAt).toBeDefined(); + expect(savedUser.updatedAt).toBeDefined(); + }); + + it('should hash password before saving', async () => { + const userData = { + firstName: 'Jane', + lastName: 'Smith', + email: 'jane.smith@test.com', + password: 'plaintext123' + }; + + const user = new User(userData); + const savedUser = await user.save(); + + expect(savedUser.password).not.toBe(userData.password); + expect(savedUser.password.length).toBeGreaterThan(50); // Bcrypt hash length + + // Verify password can be compared + const isMatch = await bcrypt.compare(userData.password, savedUser.password); + expect(isMatch).toBe(true); + }); + + it('should hash password even if it looks pre-hashed', async () => { + const hashedPassword = await bcrypt.hash('password123', 10); + const userData = { + firstName: 'Test', + lastName: 'User', + email: 'test@example.com', + password: hashedPassword + }; + + const user = new User(userData); + const savedUser = await user.save(); + + // The password will be hashed again since the pre-save hook always runs + expect(savedUser.password).not.toBe(hashedPassword); + expect(savedUser.password.length).toBeGreaterThan(50); + }); + }); + + describe('Validation', () => { + it('should require firstName', async () => { + const userData = { + lastName: 'Doe', + email: 'john.doe@test.com', + password: 'password123' + }; + + const user = new User(userData); + + await expect(user.save()).rejects.toThrow(/firstName.*required/); + }); + + it('should require lastName', async () => { + const userData = { + firstName: 'John', + email: 'john.doe@test.com', + password: 'password123' + }; + + const user = new User(userData); + + await expect(user.save()).rejects.toThrow(/lastName.*required/); + }); + + it('should require valid email format', async () => { + const userData = { + firstName: 'John', + lastName: 'Doe', + email: 'invalid-email', + password: 'password123' + }; + + const user = new User(userData); + + await expect(user.save()).rejects.toThrow(/email.*valid/); + }); + + it('should require unique email', async () => { + const userData = { + firstName: 'John', + lastName: 'Doe', + email: 'duplicate@test.com', + password: 'password123' + }; + + // Create first user + const user1 = new User(userData); + await user1.save(); + + // Try to create second user with same email + const user2 = new User(userData); + + await expect(user2.save()).rejects.toThrow(/duplicate.*email/); + }); + + it('should require password', async () => { + const userData = { + firstName: 'John', + lastName: 'Doe', + email: 'john.doe@test.com' + }; + + const user = new User(userData); + + await expect(user.save()).rejects.toThrow(/password.*required/); + }); + + it('should validate role enum', async () => { + const userData = { + firstName: 'John', + lastName: 'Doe', + email: 'john.doe@test.com', + password: 'password123', + role: 'invalid-role' + }; + + const user = new User(userData); + + await expect(user.save()).rejects.toThrow(/role.*enum/); + }); + + it('should accept valid roles', async () => { + const roles = ['user', 'admin']; // Only these two roles are valid + + for (const role of roles) { + const userData = { + firstName: 'John', + lastName: 'Doe', + email: `john.${role}@test.com`, + password: 'password123', + role + }; + + const user = new User(userData); + const savedUser = await user.save(); + + expect(savedUser.role).toBe(role); + } + }); + }); + + describe('Instance Methods', () => { + let user; + + beforeEach(async () => { + user = new User({ + firstName: 'John', + lastName: 'Doe', + email: 'john.doe@test.com', + password: 'password123' + }); + await user.save(); + }); + + it('should compare password correctly', async () => { + const isMatch = await user.comparePassword('password123'); + expect(isMatch).toBe(true); + }); + + it('should return false for incorrect password', async () => { + const isMatch = await user.comparePassword('wrongpassword'); + expect(isMatch).toBe(false); + }); + + it('should exclude password in JSON', () => { + const userJSON = user.toJSON(); + expect(userJSON.password).toBeUndefined(); + expect(userJSON.firstName).toBe('John'); + expect(userJSON.lastName).toBe('Doe'); + }); + }); + + describe('Static Methods', () => { + beforeEach(async () => { + await User.create({ + firstName: 'John', + lastName: 'Doe', + email: 'john.doe@test.com', + password: 'password123' + }); + }); + + it('should find user by email', async () => { + const user = await User.findByEmail('john.doe@test.com'); + expect(user).toBeDefined(); + expect(user.email).toBe('john.doe@test.com'); + }); + + it('should return null for non-existent email', async () => { + const user = await User.findByEmail('nonexistent@test.com'); + expect(user).toBeNull(); + }); + + it('should find active users', async () => { + // Create inactive user + await User.create({ + firstName: 'Jane', + lastName: 'Smith', + email: 'jane.smith@test.com', + password: 'password123', + isActive: false + }); + + const activeUsers = await User.find({ isActive: true }); + expect(activeUsers.length).toBe(1); + expect(activeUsers[0].email).toBe('john.doe@test.com'); + }); + }); +}); \ No newline at end of file diff --git a/backend/tests/setup.js b/backend/tests/setup.js new file mode 100644 index 0000000000..b920cd5c00 --- /dev/null +++ b/backend/tests/setup.js @@ -0,0 +1,40 @@ +import { MongoMemoryServer } from 'mongodb-memory-server'; +import mongoose from 'mongoose'; +import dotenv from 'dotenv'; + +// Load test environment variables +dotenv.config({ path: '.env.test' }); + +let mongoServer; + +// Setup MongoDB Memory Server before all tests +beforeAll(async () => { + mongoServer = await MongoMemoryServer.create(); + const mongoUri = mongoServer.getUri(); + + await mongoose.connect(mongoUri); + + // Set test environment + process.env.NODE_ENV = 'test'; + process.env.JWT_SECRET = 'test-jwt-secret-key-for-testing'; + process.env.JWT_REFRESH_SECRET = 'test-refresh-secret-key-for-testing'; +}); + +// Clean up database after each test +afterEach(async () => { + const collections = mongoose.connection.collections; + for (const key in collections) { + const collection = collections[key]; + await collection.deleteMany({}); + } +}); + +// Cleanup after all tests +afterAll(async () => { + await mongoose.connection.dropDatabase(); + await mongoose.connection.close(); + await mongoServer.stop(); +}); + +// Global test timeout +jest.setTimeout(30000); \ No newline at end of file diff --git a/backend/utils/aggregationHelpers.js b/backend/utils/aggregationHelpers.js new file mode 100644 index 0000000000..fad0325fcf --- /dev/null +++ b/backend/utils/aggregationHelpers.js @@ -0,0 +1,740 @@ +// MongoDB aggregation pipeline helpers for dashboard operations + +/** + * Get species distribution aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering + * @param {number} limit - Number of top species to return (default: 10) + * @returns {Array} Aggregation pipeline + */ +export const getSpeciesDistributionPipeline = (matchQuery, limit = 10) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { + $group: { + _id: '$species', + count: { $sum: 1 } + } + }, + { $sort: { count: -1 } }, + { $limit: limit } + ]; +}; + +/** + * Get height statistics aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering + * @returns {Array} Aggregation pipeline + */ +export const getHeightStatsPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { $unwind: '$measurements' }, + { + $sort: { + 'treeId': 1, + 'measurements.measuredAt': -1 + } + }, + { + $group: { + _id: '$_id', + latestHeight: { $first: '$measurements.height' }, + latestMeasurement: { $first: '$measurements.measuredAt' } + } + }, + { + $group: { + _id: null, + avgHeight: { $avg: '$latestHeight' }, + minHeight: { $min: '$latestHeight' }, + maxHeight: { $max: '$latestHeight' }, + treesWithMeasurements: { $sum: 1 } + } + } + ]; +}; + +/** + * Get CO2 absorption statistics aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering + * @returns {Array} Aggregation pipeline + */ +export const getCO2StatsPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { $unwind: '$measurements' }, + { + $group: { + _id: null, + totalCO2: { $sum: '$measurements.co2Absorption' }, + avgCO2PerTree: { $avg: '$measurements.co2Absorption' }, + measurementCount: { $sum: 1 } + } + } + ]; +}; + +/** + * Get health status distribution aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering + * @returns {Array} Aggregation pipeline + */ +export const getHealthDistributionPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { $unwind: '$measurements' }, + { + $sort: { + 'treeId': 1, + 'measurements.measuredAt': -1 + } + }, + { + $group: { + _id: '$_id', + latestHealthStatus: { $first: '$measurements.healthStatus' } + } + }, + { + $group: { + _id: '$latestHealthStatus', + count: { $sum: 1 } + } + }, + { $sort: { count: -1 } } + ]; +}; + +/** + * Get forest statistics aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering + * @returns {Array} Aggregation pipeline + */ +export const getForestStatsPipeline = (matchQuery) => { + return [ + { $match: matchQuery }, + { + $group: { + _id: null, + totalArea: { $sum: '$area' }, + avgArea: { $avg: '$area' }, + minArea: { $min: '$area' }, + maxArea: { $max: '$area' } + } + } + ]; +}; + +/** + * Get latest height statistics aggregation pipeline (simplified for quick stats) + * @param {Object} matchQuery - Base match query for filtering + * @returns {Array} Aggregation pipeline + */ +export const getLatestHeightPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { $unwind: '$measurements' }, + { $sort: { 'measurements.measuredAt': -1 } }, + { + $group: { + _id: '$_id', + latestHeight: { $first: '$measurements.height' } + } + }, + { + $group: { + _id: null, + avgHeight: { $avg: '$latestHeight' } + } + } + ]; +}; + +/** + * Get total CO2 absorption aggregation pipeline (simplified for quick stats) + * @param {Object} matchQuery - Base match query for filtering + * @returns {Array} Aggregation pipeline + */ +export const getTotalCO2Pipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { $unwind: '$measurements' }, + { + $group: { + _id: null, + totalCO2: { $sum: '$measurements.co2Absorption' } + } + } + ]; +}; + +/** + * Get forest comparison aggregation pipeline + * @returns {Array} Aggregation pipeline + */ +export const getForestComparisonPipeline = () => { + return [ + { $match: { isActive: true } }, + { + $lookup: { + from: 'trees', + localField: '_id', + foreignField: 'forestId', + as: 'trees' + } + }, + { + $addFields: { + totalTrees: { $size: '$trees' }, + aliveTrees: { + $size: { + $filter: { + input: '$trees', + cond: { $eq: ['$$this.isAlive', true] } + } + } + } + } + }, + { + $addFields: { + survivalRate: { + $cond: { + if: { $gt: ['$totalTrees', 0] }, + then: { $multiply: [{ $divide: ['$aliveTrees', '$totalTrees'] }, 100] }, + else: 0 + } + }, + treeDensity: { + $cond: { + if: { $gt: ['$area', 0] }, + then: { $divide: ['$totalTrees', '$area'] }, + else: 0 + } + } + } + }, + { + $project: { + name: 1, + region: 1, + area: 1, + establishedDate: 1, + totalTrees: 1, + aliveTrees: 1, + survivalRate: { $round: ['$survivalRate', 2] }, + treeDensity: { $round: ['$treeDensity', 2] } + } + }, + { $sort: { totalTrees: -1 } } + ]; +}; + +/** + * Get forest analytics survival rate aggregation pipeline + * @param {Object} forestId - Forest ObjectId + * @returns {Array} Aggregation pipeline + */ +export const getForestSurvivalRatePipeline = (forestId) => { + return [ + { $match: { forestId } }, + { + $group: { + _id: { + year: { $year: '$plantedDate' }, + month: { $month: '$plantedDate' } + }, + totalPlanted: { $sum: 1 }, + surviving: { $sum: { $cond: ['$isAlive', 1, 0] } } + } + }, + { + $project: { + date: { + $dateFromParts: { + year: '$_id.year', + month: '$_id.month', + day: 1 + } + }, + totalPlanted: 1, + surviving: 1, + survivalRate: { + $multiply: [{ $divide: ['$surviving', '$totalPlanted'] }, 100] + } + } + }, + { $sort: { date: 1 } } + ]; +}; + +/** + * Get forest analytics height growth aggregation pipeline + * @param {Object} forestId - Forest ObjectId + * @param {Object} dateFilter - Date filter object for measurements + * @returns {Array} Aggregation pipeline + */ +export const getForestHeightGrowthPipeline = (forestId, dateFilter = {}) => { + const matchConditions = { forestId, isAlive: true, ...dateFilter }; + + return [ + { $match: matchConditions }, + { $unwind: '$measurements' }, + { + $group: { + _id: { + year: { $year: '$measurements.measuredAt' }, + month: { $month: '$measurements.measuredAt' } + }, + avgHeight: { $avg: '$measurements.height' } + } + }, + { + $project: { + date: { + $dateFromParts: { + year: '$_id.year', + month: '$_id.month', + day: 1 + } + }, + avgHeight: { $round: ['$avgHeight', 2] } + } + }, + { $sort: { date: 1 } } + ]; +}; + +/** + * Get forest analytics CO2 absorption aggregation pipeline + * @param {Object} forestId - Forest ObjectId + * @param {Object} dateFilter - Date filter object for measurements + * @returns {Array} Aggregation pipeline + */ +export const getForestCO2AbsorptionPipeline = (forestId, dateFilter = {}) => { + const matchConditions = { forestId, isAlive: true, ...dateFilter }; + + return [ + { $match: matchConditions }, + { $unwind: '$measurements' }, + { + $group: { + _id: { + year: { $year: '$measurements.measuredAt' }, + month: { $month: '$measurements.measuredAt' } + }, + totalCO2: { $sum: '$measurements.co2Absorption' } + } + }, + { + $project: { + date: { + $dateFromParts: { + year: '$_id.year', + month: '$_id.month', + day: 1 + } + }, + totalCO2: { $round: ['$totalCO2', 2] } + } + }, + { $sort: { date: 1 } } + ]; +}; + +/** + * INVESTOR-FOCUSED METRICS + */ + +/** + * Get portfolio value aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getPortfolioValuePipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $group: { + _id: null, + totalCurrentValue: { $sum: '$financials.currentValue' }, + totalAcquisitionCost: { $sum: '$financials.acquisitionCost' }, + forestCount: { $sum: 1 }, + avgCurrentValue: { $avg: '$financials.currentValue' }, + avgAcquisitionCost: { $avg: '$financials.acquisitionCost' } + } + } + ]; +}; + +/** + * Get ROI statistics aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getROIStatsPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $addFields: { + roiPercentage: { + $cond: { + if: { $and: [ + { $gt: ['$financials.acquisitionCost', 0] }, + { $ne: ['$financials.currentValue', null] } + ]}, + then: { + $multiply: [ + { $divide: [ + { $subtract: ['$financials.currentValue', '$financials.acquisitionCost'] }, + '$financials.acquisitionCost' + ]}, + 100 + ] + }, + else: 0 + } + } + } + }, + { + $group: { + _id: null, + avgROI: { $avg: '$roiPercentage' }, + minROI: { $min: '$roiPercentage' }, + maxROI: { $max: '$roiPercentage' }, + totalValueGain: { $sum: { $subtract: ['$financials.currentValue', '$financials.acquisitionCost'] } } + } + } + ]; +}; + +/** + * Get carbon credits aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getCarbonCreditsPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $group: { + _id: null, + totalIssued: { $sum: '$carbonMetrics.carbonCredits.issued' }, + totalSold: { $sum: '$carbonMetrics.carbonCredits.sold' }, + totalAvailable: { $sum: '$carbonMetrics.carbonCredits.available' }, + avgPricePerCredit: { $avg: '$carbonMetrics.carbonCredits.pricePerCredit' }, + totalCarbonStored: { $sum: '$carbonMetrics.totalCarbonStored' }, + totalAnnualSequestration: { $sum: '$carbonMetrics.annualSequestration' } + } + } + ]; +}; + +/** + * Get timber value aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering trees + * @returns {Array} Aggregation pipeline + */ +export const getTimberValuePipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { + $group: { + _id: null, + totalTimberValue: { $sum: '$economicValue.currentTimberValue' }, + totalCarbonCreditValue: { $sum: '$economicValue.carbonCreditValue' }, + avgTimberValuePerTree: { $avg: '$economicValue.currentTimberValue' }, + treeCount: { $sum: 1 } + } + } + ]; +}; + +/** + * Get maintenance budget utilization pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getMaintenanceBudgetPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $addFields: { + budgetUtilization: { + $cond: { + if: { $gt: ['$financials.maintenanceBudget.allocated', 0] }, + then: { + $multiply: [ + { $divide: ['$financials.maintenanceBudget.spent', '$financials.maintenanceBudget.allocated'] }, + 100 + ] + }, + else: 0 + } + } + } + }, + { + $group: { + _id: null, + totalAnnualBudget: { $sum: '$financials.maintenanceBudget.annual' }, + totalAllocated: { $sum: '$financials.maintenanceBudget.allocated' }, + totalSpent: { $sum: '$financials.maintenanceBudget.spent' }, + avgUtilization: { $avg: '$budgetUtilization' } + } + } + ]; +}; + +/** + * MANAGER-FOCUSED METRICS + */ + +/** + * Get biodiversity index aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getBiodiversityIndexPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $addFields: { + speciesCount: { $size: { $ifNull: ['$biodiversity.species', []] } }, + biodiversityScore: { $ifNull: ['$biodiversity.biodiversityIndex', 0] } + } + }, + { + $group: { + _id: null, + avgBiodiversityIndex: { $avg: '$biodiversityScore' }, + totalSpeciesCount: { $sum: '$speciesCount' }, + forestsWithData: { + $sum: { + $cond: [{ $gt: ['$biodiversityScore', 0] }, 1, 0] + } + }, + minBiodiversity: { $min: '$biodiversityScore' }, + maxBiodiversity: { $max: '$biodiversityScore' } + } + } + ]; +}; + +/** + * Get trees at risk aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering trees + * @returns {Array} Aggregation pipeline + */ +export const getTreesAtRiskPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isAlive: true } }, + { $unwind: '$measurements' }, + { + $sort: { + 'treeId': 1, + 'measurements.measuredAt': -1 + } + }, + { + $group: { + _id: '$_id', + latestHealthStatus: { $first: '$measurements.healthStatus' }, + treeId: { $first: '$treeId' }, + species: { $first: '$species' }, + forestId: { $first: '$forestId' } + } + }, + { + $facet: { + atRiskTrees: [ + { $match: { latestHealthStatus: { $in: ['poor', 'critical'] } } }, + { $count: "count" } + ], + criticalTrees: [ + { $match: { latestHealthStatus: 'critical' } }, + { $count: "count" } + ], + totalHealthyTrees: [ + { $match: { latestHealthStatus: { $in: ['excellent', 'good'] } } }, + { $count: "count" } + ], + riskBySpecies: [ + { $match: { latestHealthStatus: { $in: ['poor', 'critical'] } } }, + { + $group: { + _id: '$species', + count: { $sum: 1 } + } + }, + { $sort: { count: -1 } } + ] + } + }, + { + $project: { + treesAtRisk: { $arrayElemAt: ['$atRiskTrees.count', 0] }, + criticalTrees: { $arrayElemAt: ['$criticalTrees.count', 0] }, + healthyTrees: { $arrayElemAt: ['$totalHealthyTrees.count', 0] }, + riskBySpecies: 1 + } + } + ]; +}; + +/** + * Get fire risk aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getFireRiskPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $group: { + _id: '$environmental.fireRisk.current', + forestCount: { $sum: 1 }, + totalArea: { $sum: '$area' } + } + }, + { + $group: { + _id: null, + riskDistribution: { + $push: { + riskLevel: '$_id', + forestCount: '$forestCount', + totalArea: '$totalArea' + } + }, + highRiskForests: { + $sum: { + $cond: [ + { $in: ['$_id', ['high', 'extreme']] }, + '$forestCount', + 0 + ] + } + }, + highRiskArea: { + $sum: { + $cond: [ + { $in: ['$_id', ['high', 'extreme']] }, + '$totalArea', + 0 + ] + } + } + } + } + ]; +}; + +/** + * Get species diversity aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getSpeciesDiversityPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { $unwind: '$biodiversity.species' }, + { + $group: { + _id: '$biodiversity.species.name', + forestsFound: { $sum: 1 }, + totalCount: { $sum: '$biodiversity.species.count' }, + conservationStatuses: { $addToSet: '$biodiversity.species.conservationStatus' }, + categories: { $addToSet: '$biodiversity.species.category' } + } + }, + { + $group: { + _id: null, + uniqueSpecies: { $sum: 1 }, + speciesDetails: { + $push: { + name: '$_id', + forestsFound: '$forestsFound', + totalCount: '$totalCount', + conservationStatuses: '$conservationStatuses', + categories: '$categories' + } + }, + endangeredSpecies: { + $sum: { + $cond: [ + { $in: ['endangered', '$conservationStatuses'] }, + 1, + 0 + ] + } + } + } + } + ]; +}; + +/** + * Get soil health aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getSoilHealthPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $group: { + _id: null, + avgPH: { $avg: '$environmental.soilHealth.phLevel' }, + avgOrganicMatter: { $avg: '$environmental.soilHealth.organicMatter' }, + compactionDistribution: { + $push: '$environmental.soilHealth.compaction' + }, + erosionRiskDistribution: { + $push: '$environmental.soilHealth.erosionRisk' + }, + forestsWithSoilData: { + $sum: { + $cond: [ + { $ne: ['$environmental.soilHealth.phLevel', null] }, + 1, + 0 + ] + } + } + } + } + ]; +}; + +/** + * Get infrastructure condition aggregation pipeline + * @param {Object} matchQuery - Base match query for filtering forests + * @returns {Array} Aggregation pipeline + */ +export const getInfrastructureConditionPipeline = (matchQuery) => { + return [ + { $match: { ...matchQuery, isActive: true } }, + { + $project: { + roadsCondition: '$infrastructure.roads.condition', + facilitiesCondition: '$infrastructure.facilities.condition', + fencingCondition: '$infrastructure.fencing.condition', + totalRoads: { $size: { $ifNull: ['$infrastructure.roads', []] } }, + totalFacilities: { $size: { $ifNull: ['$infrastructure.facilities', []] } } + } + }, + { + $group: { + _id: null, + totalRoads: { $sum: '$totalRoads' }, + totalFacilities: { $sum: '$totalFacilities' }, + roadsCondition: { $push: '$roadsCondition' }, + facilitiesCondition: { $push: '$facilitiesCondition' }, + fencingCondition: { $push: '$fencingCondition' } + } + } + ]; +}; \ No newline at end of file diff --git a/backend/utils/dashboardUtils.js b/backend/utils/dashboardUtils.js new file mode 100644 index 0000000000..12997d9f37 --- /dev/null +++ b/backend/utils/dashboardUtils.js @@ -0,0 +1,265 @@ +// Dashboard utility functions for query building, aggregation, and calculations +import mongoose from 'mongoose'; + +/** + * Safely convert string ID to ObjectId + * @param {string} id - String ID to convert + * @returns {mongoose.Types.ObjectId|null} ObjectId or null if invalid + */ +const toObjectId = (id) => { + if (!id || typeof id !== 'string') return null; + try { + return new mongoose.Types.ObjectId(id.trim()); + } catch (error) { + console.warn('Invalid ObjectId format:', id); + return null; + } +}; + +/** + * Build query conditions for tree filtering + * @param {Object} filters - Filter parameters + * @param {string} filters.forestId - Forest ID filter (single) + * @param {string} filters.forestIds - Forest IDs filter (comma-separated) + * @param {string} filters.species - Species filter (regex) + * @param {string} filters.isAlive - Alive status filter + * @param {string} filters.startDate - Start date filter + * @param {string} filters.endDate - End date filter + * @returns {Object} Query conditions for trees + */ +export const buildTreeQuery = (filters = {}) => { + const { forestId, forestIds, species, isAlive, startDate, endDate } = filters; + const query = {}; + + // Handle both single forestId and multiple forestIds + if (forestIds) { + // Multiple forests selected (comma-separated string) + const forestIdArray = forestIds.split(',') + .map(id => toObjectId(id)) + .filter(id => id !== null); // Remove invalid IDs + + if (forestIdArray.length > 0) { + query.forestId = { $in: forestIdArray }; + } + } else if (forestId) { + // Single forest selected + const objectId = toObjectId(forestId); + if (objectId) { + query.forestId = objectId; + } + } + + if (species) { + query.species = new RegExp(species, 'i'); + } + + if (isAlive !== undefined) { + query.isAlive = isAlive === 'true'; + } + + if (startDate || endDate) { + query.plantedDate = {}; + if (startDate) query.plantedDate.$gte = new Date(startDate); + if (endDate) query.plantedDate.$lte = new Date(endDate); + } + + return query; +}; + +/** + * Build query conditions for forest filtering + * @param {Object} filters - Filter parameters + * @param {string} filters.forestId - Forest ID filter + * @param {string} filters.region - Region filter (regex) + * @param {string} filters.startDate - Start date filter for establishedDate + * @param {string} filters.endDate - End date filter for establishedDate + * @returns {Object} Query conditions for forests + */ +export const buildForestQuery = (filters = {}) => { + const { forestId, region, startDate, endDate } = filters; + const query = { isActive: true }; + + if (forestId) { + const objectId = toObjectId(forestId); + if (objectId) { + query._id = objectId; + } + } + + if (region) { + query.region = new RegExp(region, 'i'); + } + + if (startDate || endDate) { + query.establishedDate = {}; + if (startDate) query.establishedDate.$gte = new Date(startDate); + if (endDate) query.establishedDate.$lte = new Date(endDate); + } + + return query; +}; + +/** + * Build pagination options for database queries + * @param {Object} params - Pagination parameters + * @param {number} params.page - Page number (default: 1) + * @param {number} params.limit - Items per page (default: 10) + * @param {string} params.sortBy - Field to sort by (default: 'name') + * @param {string} params.sortOrder - Sort direction: 'asc' or 'desc' (default: 'asc') + * @returns {Object} Pagination options with skip, limit, and sort + */ +export const buildPaginationOptions = (params = {}) => { + const { + page = 1, + limit = 10, + sortBy = 'name', + sortOrder = 'asc' + } = params; + + const skip = (page - 1) * limit; + const sortDirection = sortOrder === 'desc' ? -1 : 1; + + return { + skip, + limit: parseInt(limit), + sort: { [sortBy]: sortDirection }, + pagination: { + currentPage: parseInt(page), + limit: parseInt(limit) + } + }; +}; + +/** + * Build pagination response metadata + * @param {number} totalCount - Total number of items + * @param {number} page - Current page number + * @param {number} limit - Items per page + * @returns {Object} Pagination metadata + */ +export const buildPaginationResponse = (totalCount, page, limit) => { + return { + currentPage: parseInt(page), + totalPages: Math.ceil(totalCount / limit), + totalCount, + hasNextPage: page * limit < totalCount, + hasPrevPage: page > 1 + }; +}; + +/** + * Round number to 2 decimal places + * @param {number} value - Number to round + * @returns {number} Rounded number + */ +export const roundToTwo = (value) => { + return Math.round(value * 100) / 100; +}; + +/** + * Calculate survival rate percentage + * @param {number} alive - Number of alive trees + * @param {number} total - Total number of trees + * @returns {number} Survival rate as percentage (0-100) + */ +export const calculateSurvivalRate = (alive, total) => { + return total > 0 ? (alive / total) * 100 : 0; +}; + +/** + * Calculate tree density (trees per hectare) + * @param {number} trees - Number of trees + * @param {number} area - Area in hectares + * @returns {number} Tree density + */ +export const calculateTreeDensity = (trees, area) => { + return area > 0 ? trees / area : 0; +}; + +/** + * Get date filter for recent activity (last N days) + * @param {number} days - Number of days back + * @returns {Date} Date object for filtering + */ +export const getRecentDateFilter = (days = 30) => { + const date = new Date(); + date.setDate(date.getDate() - days); + return date; +}; + +/** + * Standard error response for dashboard endpoints + * @param {Object} res - Express response object + * @param {Error} error - Error object + * @param {string} message - User-friendly error message + * @param {number} statusCode - HTTP status code (default: 500) + */ +export const handleDashboardError = (res, error, message, statusCode = 500) => { + console.error(`Dashboard error: ${message}`, error); + res.status(statusCode).json({ + success: false, + message, + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); +}; + +/** + * Standard error response for controller endpoints + * @param {Object} res - Express response object + * @param {Error} error - Error object + * @param {string} context - Context of the error (e.g., 'Get forests error') + * @param {number} statusCode - HTTP status code (default: 500) + */ +export const handleControllerError = (res, error, context, statusCode = 500) => { + console.error(`${context}:`, error); + res.status(statusCode).json({ + success: false, + message: 'Internal server error', + error: process.env.NODE_ENV === 'development' ? error.message : undefined + }); +}; + +/** + * Standard success response helper + * @param {Object} res - Express response object + * @param {Object|Array} data - Response data + * @param {string} message - Success message (optional) + * @param {number} statusCode - HTTP status code (default: 200) + */ +export const sendSuccessResponse = (res, data, message = null, statusCode = 200) => { + const response = { + success: true, + data + }; + + if (message) { + response.message = message; + } + + res.status(statusCode).json(response); +}; + +/** + * Standard not found response helper + * @param {Object} res - Express response object + * @param {string} resource - Name of the resource that wasn't found + */ +export const sendNotFoundResponse = (res, resource = 'Resource') => { + res.status(404).json({ + success: false, + message: `${resource} not found` + }); +}; + +/** + * Standard validation error response helper + * @param {Object} res - Express response object + * @param {Array} errors - Validation errors array + */ +export const sendValidationErrorResponse = (res, errors) => { + res.status(400).json({ + success: false, + message: 'Validation failed', + errors + }); +}; \ No newline at end of file diff --git a/backend/utils/dataFormatters.js b/backend/utils/dataFormatters.js new file mode 100644 index 0000000000..5b6ba80295 --- /dev/null +++ b/backend/utils/dataFormatters.js @@ -0,0 +1,323 @@ +// Data formatting utilities for dashboard responses +import { roundToTwo, calculateSurvivalRate, calculateTreeDensity } from './dashboardUtils.js'; + +/** + * Format overview statistics data + * @param {Object} data - Raw overview data + * @param {number} data.totalTrees - Total number of trees + * @param {number} data.aliveTrees - Number of alive trees + * @param {number} data.totalForests - Total number of forests + * @param {number} data.totalUsers - Total number of users + * @param {number} data.recentActivity - Recent activity count + * @returns {Object} Formatted overview data + */ +export const formatOverviewData = (data) => { + const { totalTrees, aliveTrees, totalForests, totalUsers, recentActivity } = data; + const deadTrees = totalTrees - aliveTrees; + const survivalRate = calculateSurvivalRate(aliveTrees, totalTrees); + + return { + totalTrees, + aliveTrees, + deadTrees, + survivalRate: roundToTwo(survivalRate), + totalForests, + totalUsers, + recentActivity + }; +}; + +/** + * Format height statistics data + * @param {Array} heightStats - Raw height statistics from aggregation + * @returns {Object} Formatted height data + */ +export const formatHeightData = (heightStats) => { + if (!heightStats || heightStats.length === 0) { + return { + average: 0, + minimum: 0, + maximum: 0, + treesWithMeasurements: 0 + }; + } + + const stats = heightStats[0]; + return { + average: roundToTwo(stats.avgHeight || 0), + minimum: roundToTwo(stats.minHeight || 0), + maximum: roundToTwo(stats.maxHeight || 0), + treesWithMeasurements: stats.treesWithMeasurements || 0 + }; +}; + +/** + * Format CO2 absorption statistics data + * @param {Array} co2Stats - Raw CO2 statistics from aggregation + * @returns {Object} Formatted CO2 data + */ +export const formatCO2Data = (co2Stats) => { + if (!co2Stats || co2Stats.length === 0) { + return { + totalAbsorption: 0, + averagePerTree: 0, + totalMeasurements: 0 + }; + } + + const stats = co2Stats[0]; + return { + totalAbsorption: roundToTwo(stats.totalCO2 || 0), + averagePerTree: roundToTwo(stats.avgCO2PerTree || 0), + totalMeasurements: stats.measurementCount || 0 + }; +}; + +/** + * Format forest statistics data + * @param {Array} forestStats - Raw forest statistics from aggregation + * @param {number} totalTrees - Total number of trees for density calculation + * @returns {Object} Formatted forest data + */ +export const formatForestData = (forestStats, totalTrees) => { + if (!forestStats || forestStats.length === 0) { + return { + totalArea: 0, + averageArea: 0, + treeDensity: 0 + }; + } + + const stats = forestStats[0]; + const totalArea = stats.totalArea || 0; + const treeDensity = calculateTreeDensity(totalTrees, totalArea); + + return { + totalArea: roundToTwo(totalArea), + averageArea: roundToTwo(stats.avgArea || 0), + treeDensity: roundToTwo(treeDensity) + }; +}; + +/** + * Format complete dashboard response + * @param {Object} rawData - All raw data from database queries + * @param {Object} filters - Applied filters + * @returns {Object} Complete formatted dashboard response + */ +export const formatDashboardResponse = (rawData, filters) => { + const { + totalTrees, + aliveTrees, + totalForests, + totalUsers, + recentActivity, + speciesDistribution, + heightStats, + co2Stats, + healthDistribution, + forestStats, + // New metrics data + portfolioValue, + roiStats, + carbonCredits, + timberValue, + maintenanceBudget, + biodiversityIndex, + treesAtRisk, + fireRisk, + speciesDiversity, + soilHealth, + infrastructureCondition + } = rawData; + + return { + success: true, + data: { + overview: formatOverviewData({ + totalTrees, + aliveTrees, + totalForests, + totalUsers, + recentActivity + }), + height: formatHeightData(heightStats), + co2: formatCO2Data(co2Stats), + forest: formatForestData(forestStats, totalTrees), + distributions: { + species: speciesDistribution, + health: healthDistribution + }, + // Enhanced metrics for investors and managers + investor: formatInvestorMetrics({ + portfolioValue, + roiStats, + carbonCredits, + timberValue, + maintenanceBudget + }), + manager: formatManagerMetrics({ + biodiversityIndex, + treesAtRisk, + fireRisk, + speciesDiversity, + soilHealth, + infrastructureCondition + }), + filters, + lastUpdated: new Date().toISOString() + } + }; +}; + +/** + * Format quick stats response + * @param {Object} rawData - Raw quick stats data + * @returns {Object} Formatted quick stats response + */ +export const formatQuickStatsResponse = (rawData) => { + const { + totalTrees, + aliveTrees, + totalForests, + avgHeight, + totalCO2 + } = rawData; + + const survivalRate = calculateSurvivalRate(aliveTrees, totalTrees); + + return { + success: true, + data: { + totalTrees, + aliveTrees, + survivalRate: roundToTwo(survivalRate), + averageHeight: roundToTwo(avgHeight || 0), + totalCO2Absorption: roundToTwo(totalCO2 || 0), + totalForests + } + }; +}; + +/** + * Format forest comparison response + * @param {Array} forestStats - Raw forest comparison data + * @returns {Object} Formatted forest comparison response + */ +export const formatForestComparisonResponse = (forestStats) => { + return { + success: true, + data: { + forests: forestStats, + totalForests: forestStats.length + } + }; +}; + +/** + * ENHANCED FORMATTERS FOR NEW METRICS + */ + +/** + * Format investor metrics data + * @param {Object} data - Raw investor metrics data + * @returns {Object} Formatted investor metrics + */ +export const formatInvestorMetrics = (data) => { + const { portfolioValue, roiStats, carbonCredits, timberValue, maintenanceBudget } = data; + + // Safely extract values with proper fallbacks + const portfolioData = portfolioValue && portfolioValue.length > 0 ? portfolioValue[0] : {}; + const roiData = roiStats && roiStats.length > 0 ? roiStats[0] : {}; + const carbonData = carbonCredits && carbonCredits.length > 0 ? carbonCredits[0] : {}; + const timberData = timberValue && timberValue.length > 0 ? timberValue[0] : {}; + const budgetData = maintenanceBudget && maintenanceBudget.length > 0 ? maintenanceBudget[0] : {}; + + return { + portfolio: { + totalCurrentValue: roundToTwo(portfolioData.totalCurrentValue || 0), + totalAcquisitionCost: roundToTwo(portfolioData.totalAcquisitionCost || 0), + forestCount: portfolioData.forestCount || 0, + averageValue: roundToTwo(portfolioData.avgCurrentValue || 0) + }, + roi: { + averageROI: roundToTwo(roiData.avgROI || 0), + totalValueGain: roundToTwo(roiData.totalValueGain || 0), + minROI: roundToTwo(roiData.minROI || 0), + maxROI: roundToTwo(roiData.maxROI || 0) + }, + carbonCredits: { + totalAvailable: carbonData.totalAvailable || 0, + totalSold: carbonData.totalSold || 0, + averagePrice: roundToTwo(carbonData.avgPricePerCredit || 0), + totalCarbonStored: roundToTwo(carbonData.totalCarbonStored || 0), + annualSequestration: roundToTwo(carbonData.totalAnnualSequestration || 0) + }, + timber: { + totalValue: roundToTwo(timberData.totalTimberValue || 0), + carbonCreditValue: roundToTwo(timberData.totalCarbonCreditValue || 0), + averageValuePerTree: roundToTwo(timberData.avgTimberValuePerTree || 0) + }, + maintenance: { + totalBudget: roundToTwo(budgetData.totalAnnualBudget || 0), + totalSpent: roundToTwo(budgetData.totalSpent || 0), + utilization: roundToTwo(budgetData.avgUtilization || 0) + } + }; +}; + +/** + * Format manager metrics data + * @param {Object} data - Raw manager metrics data + * @returns {Object} Formatted manager metrics + */ +export const formatManagerMetrics = (data) => { + const { biodiversityIndex, treesAtRisk, fireRisk, speciesDiversity, soilHealth, infrastructureCondition } = data; + + // Safely extract values with proper fallbacks + const biodiversityData = biodiversityIndex && biodiversityIndex.length > 0 ? biodiversityIndex[0] : {}; + const riskData = treesAtRisk && treesAtRisk.length > 0 ? treesAtRisk[0] : {}; + const fireData = fireRisk && fireRisk.length > 0 ? fireRisk[0] : {}; + const speciesData = speciesDiversity && speciesDiversity.length > 0 ? speciesDiversity[0] : {}; + const soilData = soilHealth && soilHealth.length > 0 ? soilHealth[0] : {}; + const infrastructureData = infrastructureCondition && infrastructureCondition.length > 0 ? infrastructureCondition[0] : {}; + + return { + biodiversity: { + averageIndex: roundToTwo(biodiversityData.avgBiodiversityIndex || 0), + totalSpecies: biodiversityData.totalSpeciesCount || 0, + forestsWithData: biodiversityData.forestsWithData || 0, + range: { + min: roundToTwo(biodiversityData.minBiodiversity || 0), + max: roundToTwo(biodiversityData.maxBiodiversity || 0) + } + }, + treesAtRisk: { + total: riskData.treesAtRisk || 0, + critical: riskData.criticalTrees || 0, + healthy: riskData.healthyTrees || 0, + riskBySpecies: riskData.riskBySpecies || [] + }, + fireRisk: { + highRiskForests: fireData.highRiskForests || 0, + highRiskArea: roundToTwo(fireData.highRiskArea || 0), + distribution: fireData.riskDistribution || [] + }, + speciesDiversity: { + uniqueSpecies: speciesData.uniqueSpecies || 0, + endangeredSpecies: speciesData.endangeredSpecies || 0, + speciesDetails: speciesData.speciesDetails || [] + }, + soilHealth: { + averagePH: roundToTwo(soilData.avgPH || 0), + averageOrganicMatter: roundToTwo(soilData.avgOrganicMatter || 0), + forestsWithData: soilData.forestsWithSoilData || 0 + }, + infrastructure: { + totalRoads: infrastructureData.totalRoads || 0, + totalFacilities: infrastructureData.totalFacilities || 0, + roadConditions: infrastructureData.roadsCondition || [], + facilityConditions: infrastructureData.facilitiesCondition || [] + } + }; +}; \ No newline at end of file diff --git a/backend/utils/exportHelpers.js b/backend/utils/exportHelpers.js new file mode 100644 index 0000000000..433419888b --- /dev/null +++ b/backend/utils/exportHelpers.js @@ -0,0 +1,373 @@ +// Export utility functions for data transformation and formatting +import { buildTreeQuery, roundToTwo } from './dashboardUtils.js'; + +/** + * Build query conditions for tree exports with additional export-specific filters + * @param {Object} queryParams - Query parameters from request + * @param {string} queryParams.forestId - Forest ID filter + * @param {string} queryParams.species - Species filter + * @param {string} queryParams.isAlive - Alive status filter + * @param {string} queryParams.startDate - Start date filter + * @param {string} queryParams.endDate - End date filter + * @returns {Object} Query conditions for tree export + */ +export const buildExportTreeQuery = (queryParams) => { + const { isAlive, ...baseFilters } = queryParams; + + const query = buildTreeQuery(baseFilters); + + // Add export-specific isAlive filter + if (isAlive !== undefined) { + query.isAlive = isAlive === 'true'; + } + + return query; +}; + +/** + * Transform tree data into export row format + * @param {Object} tree - Tree document from database + * @param {Object} options - Export options with field selections + * @returns {Object} Formatted tree row data + */ +export const transformTreeToExportRow = (tree, options = {}) => { + const row = {}; + + // Basic Information - always included + if (options.basicInfo !== 'false') { + row['Tree ID'] = tree.treeId; + row['Forest Name'] = tree.forestId?.name || 'Unknown'; + row['Region'] = tree.forestId?.region || 'Unknown'; + row['Species'] = tree.species; + row['Planted Date'] = tree.plantedDate ? new Date(tree.plantedDate).toISOString().split('T')[0] : ''; + row['Is Alive'] = tree.isAlive ? 'Yes' : 'No'; + row['Age (Days)'] = tree.plantedDate ? Math.floor((new Date() - new Date(tree.plantedDate)) / (1000 * 60 * 60 * 24)) : 0; + + if (!tree.isAlive) { + row['Death Date'] = tree.deathDate ? new Date(tree.deathDate).toISOString().split('T')[0] : ''; + row['Death Cause'] = tree.deathCause || ''; + } + } + + // Location + if (options.location === 'true') { + row['Longitude'] = tree.location?.coordinates?.[0] || ''; + row['Latitude'] = tree.location?.coordinates?.[1] || ''; + } + + // Health Status & Latest Measurements + if (options.health === 'true' || options.measurements === 'true') { + if (tree.measurements && tree.measurements.length > 0) { + const latestMeasurement = tree.measurements + .sort((a, b) => new Date(b.measuredAt) - new Date(a.measuredAt))[0]; + + if (options.health === 'true') { + row['Current Health Status'] = latestMeasurement.healthStatus || ''; + } + + if (options.measurements === 'true') { + row['Current Height (m)'] = latestMeasurement.height || ''; + row['Current Diameter (cm)'] = latestMeasurement.diameter || ''; + row['Current CO2 Absorption (kg)'] = latestMeasurement.co2Absorption || ''; + row['Latest Measurement Date'] = latestMeasurement.measuredAt ? new Date(latestMeasurement.measuredAt).toISOString().split('T')[0] : ''; + row['Measurement Notes'] = latestMeasurement.notes || ''; + } + } + } + + // Images + if (options.images === 'true' && tree.images && tree.images.length > 0) { + row['Image URLs'] = tree.images.map(img => img.url).join('; '); + row['Image Count'] = tree.images.length; + } + + // Genetics + if (options.genetics === 'true' && tree.genetics) { + row['Seed Source'] = tree.genetics.seedSource || ''; + row['Cultivar'] = tree.genetics.cultivar || ''; + row['Parent Tree ID'] = tree.genetics.parentTreeId || ''; + row['Genetic Diversity'] = tree.genetics.geneticDiversity || ''; + row['Provenance Region'] = tree.genetics.provenanceRegion || ''; + } + + // Growth Model + if (options.growthModel === 'true' && tree.growthModel) { + row['Expected Height at 15 Years (m)'] = tree.growthModel.expectedHeightAt15Years || ''; + row['Expected Diameter at 15 Years (cm)'] = tree.growthModel.expectedDiameterAt15Years || ''; + row['Growth Rate'] = tree.growthModel.growthRate || ''; + row['Site Index'] = tree.growthModel.siteIndex || ''; + row['Maturity Age (years)'] = tree.growthModel.maturityAge || ''; + row['Max Height (m)'] = tree.growthModel.maxHeight || ''; + row['Max Diameter (cm)'] = tree.growthModel.maxDiameter || ''; + } + + // Economic Value + if (options.economicValue === 'true' && tree.economicValue) { + row['Current Timber Value'] = tree.economicValue.currentTimberValue || ''; + row['Carbon Credit Value'] = tree.economicValue.carbonCreditValue || ''; + row['Last Valuation Date'] = tree.economicValue.lastValuation ? new Date(tree.economicValue.lastValuation).toISOString().split('T')[0] : ''; + row['Valuation Method'] = tree.economicValue.valuationMethod || ''; + row['Market Price per m³'] = tree.economicValue.marketPricePerCubicMeter || ''; + } + + // Canopy + if (options.canopy === 'true' && tree.canopy) { + row['Canopy Diameter (m)'] = tree.canopy.diameter || ''; + row['Canopy Area (m²)'] = tree.canopy.area || ''; + row['Canopy Density'] = tree.canopy.density || ''; + row['Canopy Condition'] = tree.canopy.condition || ''; + row['Leaf Area Index'] = tree.canopy.leafArea || ''; + if (tree.canopy.seasonalChanges) { + row['Spring Bud Break'] = tree.canopy.seasonalChanges.springBudBreak ? new Date(tree.canopy.seasonalChanges.springBudBreak).toISOString().split('T')[0] : ''; + row['Fall Color Change'] = tree.canopy.seasonalChanges.fallColorChange ? new Date(tree.canopy.seasonalChanges.fallColorChange).toISOString().split('T')[0] : ''; + row['Leaf Drop'] = tree.canopy.seasonalChanges.leafDrop ? new Date(tree.canopy.seasonalChanges.leafDrop).toISOString().split('T')[0] : ''; + } + } + + // Ecological Benefits + if (options.ecologicalBenefits === 'true' && tree.ecologicalBenefits) { + // Stormwater + if (tree.ecologicalBenefits.stormwaterIntercepted) { + row['Stormwater Intercepted (gallons)'] = tree.ecologicalBenefits.stormwaterIntercepted.gallons || ''; + row['Stormwater Value'] = tree.ecologicalBenefits.stormwaterIntercepted.value || ''; + } + // CO2 + if (tree.ecologicalBenefits.co2Sequestered) { + row['CO2 Sequestered (lbs)'] = tree.ecologicalBenefits.co2Sequestered.pounds || ''; + row['CO2 Sequestered Value'] = tree.ecologicalBenefits.co2Sequestered.value || ''; + } + if (tree.ecologicalBenefits.co2Stored) { + row['CO2 Stored (lbs)'] = tree.ecologicalBenefits.co2Stored.pounds || ''; + row['CO2 Stored Value'] = tree.ecologicalBenefits.co2Stored.value || ''; + } + // Air quality + if (tree.ecologicalBenefits.airPollutantsRemoved) { + row['Air Pollutants Removed (lbs)'] = tree.ecologicalBenefits.airPollutantsRemoved.pounds || ''; + row['Air Pollutants Value'] = tree.ecologicalBenefits.airPollutantsRemoved.value || ''; + } + // Soil + if (tree.ecologicalBenefits.soilStabilization) { + row['Root Area (m²)'] = tree.ecologicalBenefits.soilStabilization.rootArea || ''; + row['Erosion Prevention (tons/year)'] = tree.ecologicalBenefits.soilStabilization.erosionPrevention || ''; + } + // Wildlife + if (tree.ecologicalBenefits.wildlifeHabitat) { + row['Nesting Sites'] = tree.ecologicalBenefits.wildlifeHabitat.nestingSites || ''; + row['Food Production (kg/year)'] = tree.ecologicalBenefits.wildlifeHabitat.foodProduction || ''; + row['Biodiversity Support'] = tree.ecologicalBenefits.wildlifeHabitat.biodiversitySupport || ''; + } + } + + // Environmental Factors + if (options.environmentalFactors === 'true' && tree.environmentalFactors) { + if (tree.environmentalFactors.microclimate) { + row['Avg Temperature'] = tree.environmentalFactors.microclimate.avgTemperature || ''; + row['Humidity'] = tree.environmentalFactors.microclimate.humidity || ''; + row['Wind Exposure'] = tree.environmentalFactors.microclimate.windExposure || ''; + } + if (tree.environmentalFactors.siteConditions) { + row['Slope'] = tree.environmentalFactors.siteConditions.slope || ''; + row['Aspect'] = tree.environmentalFactors.siteConditions.aspect || ''; + row['Drainage'] = tree.environmentalFactors.siteConditions.drainage || ''; + row['Competition Index'] = tree.environmentalFactors.siteConditions.competitionIndex || ''; + } + row['Forest Position'] = tree.environmentalFactors.forestPosition || ''; + } + + // Maintenance + if (options.maintenance === 'true' && tree.maintenance) { + // Latest fertilization + if (tree.maintenance.fertilization && tree.maintenance.fertilization.length > 0) { + const latestFert = tree.maintenance.fertilization + .sort((a, b) => new Date(b.date) - new Date(a.date))[0]; + row['Last Fertilization Date'] = latestFert.date ? new Date(latestFert.date).toISOString().split('T')[0] : ''; + row['Fertilization Type'] = latestFert.type || ''; + row['NPK Ratio'] = latestFert.npkRatio || ''; + } + // Latest pest control + if (tree.maintenance.pestControl && tree.maintenance.pestControl.length > 0) { + const latestPest = tree.maintenance.pestControl + .sort((a, b) => new Date(b.date) - new Date(a.date))[0]; + row['Last Pest Control Date'] = latestPest.date ? new Date(latestPest.date).toISOString().split('T')[0] : ''; + row['Pest Type'] = latestPest.pestType || ''; + row['Treatment'] = latestPest.treatment || ''; + } + // Latest damage report + if (tree.maintenance.damageReports && tree.maintenance.damageReports.length > 0) { + const latestDamage = tree.maintenance.damageReports + .sort((a, b) => new Date(b.date) - new Date(a.date))[0]; + row['Last Damage Date'] = latestDamage.date ? new Date(latestDamage.date).toISOString().split('T')[0] : ''; + row['Damage Type'] = latestDamage.type || ''; + row['Damage Severity'] = latestDamage.severity || ''; + } + // Pruning + if (tree.maintenance.pruning) { + row['Last Pruned'] = tree.maintenance.pruning.lastPruned ? new Date(tree.maintenance.pruning.lastPruned).toISOString().split('T')[0] : ''; + row['Next Pruning Scheduled'] = tree.maintenance.pruning.nextScheduled ? new Date(tree.maintenance.pruning.nextScheduled).toISOString().split('T')[0] : ''; + } + } + + // Metadata + if (options.metadata === 'true') { + if (tree.metadata) { + row['Soil Condition'] = tree.metadata.soilCondition || ''; + row['Sunlight Exposure'] = tree.metadata.sunlightExposure || ''; + row['Water Access'] = tree.metadata.waterAccess || ''; + row['Seedling Source'] = tree.metadata.seedlingSource || ''; + row['Planting Method'] = tree.metadata.plantingMethod || ''; + row['Initial Spacing (m)'] = tree.metadata.initialSpacing || ''; + row['Management Objective'] = tree.metadata.managementObjective || ''; + } + row['Created At'] = tree.createdAt ? new Date(tree.createdAt).toISOString() : ''; + row['Updated At'] = tree.updatedAt ? new Date(tree.updatedAt).toISOString() : ''; + } + + return row; +}; + +/** + * Transform tree measurement data into export row format + * @param {Object} tree - Tree document from database + * @param {Object} measurement - Measurement data + * @param {number} measurementIndex - Index of measurement + * @returns {Object} Formatted measurement row data + */ +export const transformMeasurementToExportRow = (tree, measurement, measurementIndex) => { + return { + 'Tree ID': tree.treeId, + 'Forest Name': tree.forestId?.name || 'Unknown', + 'Species': tree.species, + 'Measurement #': measurementIndex + 1, + 'Height (m)': measurement.height, + 'Diameter (cm)': measurement.diameter || '', + 'CO2 Absorption (kg)': measurement.co2Absorption || '', + 'Health Status': measurement.healthStatus, + 'Measurement Date': measurement.measuredAt.toISOString().split('T')[0], + 'Notes': measurement.notes || '' + }; +}; + +/** + * Process trees data for CSV export + * @param {Array} trees - Array of tree documents + * @param {Object} options - Processing options with field selections + * @returns {Array} Array of formatted export rows + */ +export const processTreesForExport = (trees, options = {}) => { + const exportData = []; + + trees.forEach(tree => { + // Transform each tree with all field options + const row = transformTreeToExportRow(tree, options); + exportData.push(row); + }); + + return exportData; +}; + +/** + * Generate CSV content from data array + * @param {Array} data - Array of objects to convert to CSV + * @returns {string} CSV formatted string + */ +export const generateCSVContent = (data) => { + if (!data || data.length === 0) { + return ''; + } + + const headers = Object.keys(data[0]); + const rows = data.map(row => + headers.map(header => { + const value = row[header]; + // Escape values containing commas with quotes + return typeof value === 'string' && value.includes(',') ? `"${value}"` : value; + }).join(',') + ); + + return [headers.join(','), ...rows].join('\n'); +}; + +/** + * Generate export filename with timestamp + * @param {string} baseFilename - Base filename without extension + * @param {string} extension - File extension (e.g., 'csv', 'xlsx') + * @returns {string} Generated filename with timestamp + */ +export const generateExportFilename = (baseFilename, extension) => { + const timestamp = new Date().toISOString().split('T')[0]; + return `${baseFilename}_${timestamp}.${extension}`; +}; + +/** + * Set CSV response headers + * @param {Object} res - Express response object + * @param {string} filename - Filename for download + * @param {string} content - CSV content for content-length header + */ +export const setCSVResponseHeaders = (res, filename, content) => { + res.setHeader('Content-Type', 'text/csv'); + res.setHeader('Content-Disposition', `attachment; filename="${filename}"`); + res.setHeader('Content-Length', Buffer.byteLength(content)); +}; + +/** + * Set XLSX response headers + * @param {Object} res - Express response object + * @param {string} filename - Filename for download + * @param {Buffer} buffer - XLSX buffer for content-length header + */ +export const setXLSXResponseHeaders = (res, filename, buffer) => { + res.setHeader('Content-Type', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'); + res.setHeader('Content-Disposition', `attachment; filename="${filename}"`); + res.setHeader('Content-Length', buffer.length); +}; + +/** + * Calculate forest analytics statistics + * @param {Object} forest - Forest document + * @param {number} totalTrees - Total trees in forest + * @param {number} aliveTrees - Alive trees in forest + * @param {number} avgHeight - Average height of trees + * @param {number} totalCO2 - Total CO2 absorption + * @returns {Object} Formatted forest analytics data + */ +export const calculateForestAnalytics = (forest, totalTrees, aliveTrees, avgHeight, totalCO2) => { + const deadTrees = totalTrees - aliveTrees; + const survivalRate = totalTrees > 0 ? (aliveTrees / totalTrees) * 100 : 0; + const treesPerHectare = totalTrees > 0 ? totalTrees / forest.area : 0; + + return { + 'Forest Name': forest.name, + 'Region': forest.region, + 'Area (hectares)': forest.area, + 'Established Date': forest.establishedDate.toISOString().split('T')[0], + 'Total Trees': totalTrees, + 'Alive Trees': aliveTrees, + 'Dead Trees': deadTrees, + 'Survival Rate (%)': roundToTwo(survivalRate), + 'Average Height (m)': roundToTwo(avgHeight), + 'Total CO2 Absorption (kg)': roundToTwo(totalCO2), + 'Trees per Hectare': roundToTwo(treesPerHectare) + }; +}; + +/** + * Generate export statistics for XLSX files + * @param {Array} trees - Array of tree data + * @returns {Object} Statistics object + */ +export const generateExportStatistics = (trees) => { + const aliveTrees = trees.filter(t => t.isAlive).length; + const deadTrees = trees.length - aliveTrees; + const survivalRate = trees.length > 0 ? (aliveTrees / trees.length) * 100 : 0; + + return { + 'Total Trees': trees.length, + 'Alive Trees': aliveTrees, + 'Dead Trees': deadTrees, + 'Survival Rate (%)': roundToTwo(survivalRate), + 'Export Date': new Date().toISOString().split('T')[0], + 'Unique Forests': [...new Set(trees.map(t => t.forestId?.name).filter(Boolean))].length, + 'Unique Species': [...new Set(trees.map(t => t.species))].length + }; +}; \ No newline at end of file diff --git a/backend/utils/jwt.js b/backend/utils/jwt.js new file mode 100644 index 0000000000..c19df4d95e --- /dev/null +++ b/backend/utils/jwt.js @@ -0,0 +1,35 @@ +import jwt from 'jsonwebtoken'; + +const JWT_SECRET = process.env.JWT_SECRET || 'your-super-secret-jwt-key-change-in-production'; +const JWT_EXPIRES_IN = process.env.JWT_EXPIRES_IN || '7d'; + +// Generate JWT token +export const generateToken = (userId) => { + return jwt.sign( + { userId }, + JWT_SECRET, + { expiresIn: JWT_EXPIRES_IN } + ); +}; + +// Verify JWT token +export const verifyToken = (token) => { + try { + return jwt.verify(token, JWT_SECRET); + } catch (error) { + throw new Error('Invalid or expired token'); + } +}; + +// Extract token from request headers +export const extractTokenFromHeader = (authHeader) => { + if (!authHeader) { + throw new Error('Authorization header is missing'); + } + + if (!authHeader.startsWith('Bearer ')) { + throw new Error('Authorization header must start with Bearer'); + } + + return authHeader.substring(7); +}; \ No newline at end of file diff --git a/backend/utils/simulationEngine.js b/backend/utils/simulationEngine.js new file mode 100644 index 0000000000..58363bbab1 --- /dev/null +++ b/backend/utils/simulationEngine.js @@ -0,0 +1,484 @@ +/** + * Mathematical Simulation Engine for Dashboard Metrics + * + * This utility generates realistic financial and ecological metrics based on actual tree data. + * All calculations are deterministic but include realistic variations based on: + * - Tree species characteristics + * - Age and growth patterns + * - Seasonal variations + * - Market conditions + * - Environmental factors + */ + +import mongoose from 'mongoose'; +import { Tree, Forest } from '../models/index.js'; + +/** + * Species-specific multipliers for various metrics + * Based on Swedish forestry data and market conditions + */ +const SPECIES_FACTORS = { + 'pine': { + timberValue: 1.0, + co2Absorption: 1.2, + maintenanceCost: 0.8, + growthRate: 1.0, + marketMultiplier: 1.0, + maturityAge: 80, + peakGrowthAge: 15 + }, + 'spruce': { + timberValue: 1.1, + co2Absorption: 1.4, + maintenanceCost: 0.9, + growthRate: 1.2, + marketMultiplier: 1.05, + maturityAge: 70, + peakGrowthAge: 12 + }, + 'birch': { + timberValue: 0.7, + co2Absorption: 0.8, + maintenanceCost: 0.6, + growthRate: 1.5, + marketMultiplier: 0.8, + maturityAge: 60, + peakGrowthAge: 10 + }, + 'oak': { + timberValue: 2.0, + co2Absorption: 1.8, + maintenanceCost: 1.2, + growthRate: 0.6, + marketMultiplier: 1.8, + maturityAge: 120, + peakGrowthAge: 25 + }, + 'fir': { + timberValue: 0.95, + co2Absorption: 1.3, + maintenanceCost: 0.85, + growthRate: 1.1, + marketMultiplier: 0.95, + maturityAge: 75, + peakGrowthAge: 14 + } +}; + +/** + * Market price data for Sweden (SEK) + */ +const MARKET_PRICES = { + timber: { + basePricePerCubicMeter: 650, // Average Swedish timber price + volatilityFactor: 0.15, // 15% price variation + seasonalAdjustment: { + 1: 0.95, 2: 0.92, 3: 0.98, 4: 1.05, // Winter low, spring pickup + 5: 1.08, 6: 1.12, 7: 1.10, 8: 1.06, // Summer peak construction season + 9: 1.03, 10: 1.00, 11: 0.97, 12: 0.93 // Autumn decline + } + }, + carbonCredits: { + basePricePerTon: 85, // EUR per ton CO2, converted to SEK ~900 + volatilityFactor: 0.25, // Higher volatility in carbon markets + trendMultiplier: 1.08 // 8% annual growth trend + }, + maintenanceRates: { + baseRatePerHectare: 1200, // SEK per hectare annually + complexityMultiplier: { + low: 0.8, + medium: 1.0, + high: 1.3, + extreme: 1.8 + } + } +}; + +/** + * Calculate deterministic pseudo-random value based on seed + * @param {string|number} seed - Seed value for consistency + * @param {number} min - Minimum value + * @param {number} max - Maximum value + * @returns {number} Deterministic random value + */ +const seededRandom = (seed, min = 0, max = 1) => { + const seedNum = typeof seed === 'string' ? + seed.split('').reduce((a, b) => { a = ((a << 5) - a) + b.charCodeAt(0); return a & a; }, 0) : + seed; + const x = Math.sin(seedNum) * 10000; + const random = x - Math.floor(x); + return min + random * (max - min); +}; + +/** + * Get species factor with fallback to pine + * @param {string} species - Tree species + * @returns {Object} Species factors + */ +const getSpeciesFactors = (species) => { + const normalizedSpecies = species.toLowerCase().trim(); + return SPECIES_FACTORS[normalizedSpecies] || SPECIES_FACTORS['pine']; +}; + +/** + * Calculate tree age in years + * @param {Date} plantedDate - When the tree was planted + * @returns {number} Age in years + */ +const calculateAge = (plantedDate) => { + const now = new Date(); + const planted = new Date(plantedDate); + return Math.max(0, (now - planted) / (1000 * 60 * 60 * 24 * 365.25)); +}; + +/** + * Simulate timber value for a single tree + * @param {Object} tree - Tree document + * @param {Date} currentDate - Current date for market conditions + * @returns {number} Estimated timber value in SEK + */ +const simulateTreeTimberValue = (tree, currentDate = new Date()) => { + const age = calculateAge(tree.plantedDate); + const species = getSpeciesFactors(tree.species); + + // Get latest measurement or estimate + const latestMeasurement = tree.measurements && tree.measurements.length > 0 ? + tree.measurements[tree.measurements.length - 1] : null; + + // Estimate height and diameter if no measurements + const height = latestMeasurement?.height || Math.max(0.5, age * 0.8 * species.growthRate); + const diameter = latestMeasurement?.diameter || height * 0.08; // Rough diameter estimate + + // Volume calculation (simplified cone formula) + const volume = Math.PI * Math.pow(diameter / 200, 2) * height * 0.4; // cubic meters + + // Market price with seasonal adjustment + const month = currentDate.getMonth() + 1; + const seasonalPrice = MARKET_PRICES.timber.basePricePerCubicMeter * + MARKET_PRICES.timber.seasonalAdjustment[month]; + + // Add species-specific multiplier and market volatility + const treeId = tree._id.toString(); + const volatility = seededRandom(treeId + currentDate.getFullYear(), + 1 - MARKET_PRICES.timber.volatilityFactor, + 1 + MARKET_PRICES.timber.volatilityFactor); + + const finalPrice = seasonalPrice * species.timberValue * species.marketMultiplier * volatility; + + return Math.round(volume * finalPrice); +}; + +/** + * Simulate CO2 absorption and carbon credit value for a tree + * @param {Object} tree - Tree document + * @param {Date} currentDate - Current date for market conditions + * @returns {Object} CO2 data with absorption and credit value + */ +const simulateTreeCO2Value = (tree, currentDate = new Date()) => { + const age = calculateAge(tree.plantedDate); + const species = getSpeciesFactors(tree.species); + + // CO2 absorption increases with age up to maturity, then stabilizes + const maturityFactor = Math.min(1, age / (species.maturityAge * 0.6)); + const peakGrowthFactor = age <= species.peakGrowthAge ? + (age / species.peakGrowthAge) : + Math.max(0.7, 1 - (age - species.peakGrowthAge) / species.maturityAge); + + // Base CO2 absorption (kg per year) + const baseCO2 = 25 * species.co2Absorption * maturityFactor * peakGrowthFactor; + + // Add some variation based on tree health + const healthMultiplier = tree.isAlive ? 1.0 : 0; + const treeVariation = seededRandom(tree._id.toString(), 0.8, 1.2); + + const annualCO2Absorption = baseCO2 * healthMultiplier * treeVariation; + const lifetimeCO2Storage = annualCO2Absorption * Math.min(age, species.maturityAge * 0.8); + + // Calculate carbon credit value + const currentYear = currentDate.getFullYear(); + const yearsFromBase = currentYear - 2020; + const trendAdjustedPrice = MARKET_PRICES.carbonCredits.basePricePerTon * + Math.pow(MARKET_PRICES.carbonCredits.trendMultiplier, yearsFromBase); + + const priceVolatility = seededRandom(tree._id.toString() + currentYear, + 1 - MARKET_PRICES.carbonCredits.volatilityFactor, + 1 + MARKET_PRICES.carbonCredits.volatilityFactor); + + const creditValue = (lifetimeCO2Storage / 1000) * trendAdjustedPrice * priceVolatility * 10.5; // EUR to SEK + + return { + annualAbsorption: Math.round(annualCO2Absorption * 100) / 100, + lifetimeStorage: Math.round(lifetimeCO2Storage * 100) / 100, + creditValue: Math.round(creditValue), + marketPrice: Math.round(trendAdjustedPrice * 10.5) // Per ton in SEK + }; +}; + +/** + * Simulate maintenance costs for a forest + * @param {Object} forest - Forest document + * @param {number} treeCount - Number of trees in forest + * @param {Date} currentDate - Current date for calculations + * @returns {Object} Maintenance cost breakdown + */ +const simulateMaintenanceCosts = (forest, treeCount, currentDate = new Date()) => { + const area = forest.area || (treeCount * 0.01); // Estimate 100 trees per hectare + + // Determine complexity based on forest characteristics + const complexityFactors = []; + if (forest.terrain === 'mountainous') complexityFactors.push('high'); + if (forest.region && forest.region.includes('north')) complexityFactors.push('medium'); + if (treeCount > 10000) complexityFactors.push('high'); + if (area > 100) complexityFactors.push('medium'); + + const complexityLevel = complexityFactors.includes('high') ? 'high' : + complexityFactors.includes('medium') ? 'medium' : 'low'; + + const baseRate = MARKET_PRICES.maintenanceRates.baseRatePerHectare; + const complexityMultiplier = MARKET_PRICES.maintenanceRates.complexityMultiplier[complexityLevel]; + + // Add seasonal variation (more maintenance in spring/summer) + const month = currentDate.getMonth() + 1; + const seasonalMultiplier = month >= 4 && month <= 9 ? 1.2 : 0.8; + + const annualCost = area * baseRate * complexityMultiplier * seasonalMultiplier; + const forestVariation = seededRandom(forest._id.toString(), 0.9, 1.1); + + return { + annualBudget: Math.round(annualCost * forestVariation), + monthlyBudget: Math.round(annualCost * forestVariation / 12), + complexityLevel, + area: Math.round(area * 100) / 100, + ratePerHectare: Math.round(baseRate * complexityMultiplier) + }; +}; + +/** + * Simulate portfolio-wide investment metrics + * @param {Array} trees - Array of tree documents + * @param {Array} forests - Array of forest documents + * @param {Object} filters - Date and forest filters + * @returns {Object} Investment metrics + */ +export const simulateInvestorMetrics = async (trees, forests, filters = {}) => { + const currentDate = new Date(); + + // Calculate timber values + const timberValues = trees.map(tree => simulateTreeTimberValue(tree, currentDate)); + const totalTimberValue = timberValues.reduce((sum, value) => sum + value, 0); + const avgTimberValue = trees.length > 0 ? totalTimberValue / trees.length : 0; + + // Calculate CO2 values + const co2Data = trees.map(tree => simulateTreeCO2Value(tree, currentDate)); + const totalCO2Credits = co2Data.reduce((sum, data) => sum + data.creditValue, 0); + const totalCO2Absorption = co2Data.reduce((sum, data) => sum + data.annualAbsorption, 0); + + // Calculate maintenance costs + const maintenanceCosts = forests.map(forest => { + const forestTrees = trees.filter(tree => tree.forestId.toString() === forest._id.toString()); + return simulateMaintenanceCosts(forest, forestTrees.length, currentDate); + }); + const totalMaintenanceBudget = maintenanceCosts.reduce((sum, cost) => sum + cost.annualBudget, 0); + + // Estimate initial investment (planting costs + land acquisition + infrastructure) + const avgPlantingCostPerTree = 65; // SEK per tree (planting + sapling + maintenance) + const landAndInfrastructureCost = forests.length * 150000; // 150k SEK per forest + const estimatedInitialInvestment = (trees.length * avgPlantingCostPerTree) + landAndInfrastructureCost; + + // Enhanced portfolio value calculation (timber appreciates over time) + const enhancedTimberValue = totalTimberValue * 2.8; // Market appreciation + const currentPortfolioValue = enhancedTimberValue + totalCO2Credits + (forests.length * 500000); // Add land value + + // Calculate realistic ROI (Swedish forestry typically 6-12% annually, compound over years) + const baseROI = estimatedInitialInvestment > 0 ? + ((currentPortfolioValue - estimatedInitialInvestment) / estimatedInitialInvestment) * 100 : 0; + const roi = Math.max(baseROI, 45); // Ensure minimum realistic ROI for mature forests + + return { + portfolio: { + totalCurrentValue: Math.round(currentPortfolioValue), + forestCount: forests.length, + treeCount: trees.length, + averageValue: Math.round(currentPortfolioValue / Math.max(1, forests.length)), + totalAcquisitionCost: Math.round(estimatedInitialInvestment) + }, + timber: { + totalValue: Math.round(enhancedTimberValue), + averageValuePerTree: Math.round(enhancedTimberValue / Math.max(1, trees.length)) + }, + carbonCredits: { + totalAvailable: Math.round(totalCO2Absorption), + totalSold: Math.round(totalCO2Absorption * 0.3), // Assume 30% sold + averagePrice: Math.round(co2Data.length > 0 ? co2Data[0].marketPrice : 0), + totalRevenue: Math.round(totalCO2Credits * 0.3) + }, + roi: { + averageROI: Math.round(roi * 100) / 100, + minROI: Math.round((roi - 5) * 100) / 100, // Estimated range + maxROI: Math.round((roi + 8) * 100) / 100 + }, + maintenance: { + totalBudget: Math.round(totalMaintenanceBudget), + totalSpent: Math.round(totalMaintenanceBudget * seededRandom(currentDate.getTime(), 0.6, 0.9)), + utilization: Math.round(seededRandom(currentDate.getTime(), 60, 90) * 100) / 100 + } + }; +}; + +/** + * Simulate ecological and environmental metrics + * @param {Array} trees - Array of tree documents + * @param {Array} forests - Array of forest documents + * @param {Object} filters - Date and forest filters + * @returns {Object} Ecological metrics + */ +export const simulateEcologicalMetrics = async (trees, forests, filters = {}) => { + const currentDate = new Date(); + const aliveTrees = trees.filter(tree => tree.isAlive); + + // Enhanced biodiversity index calculation + const speciesDistribution = trees.reduce((acc, tree) => { + const normalizedSpecies = tree.species.toLowerCase(); + acc[normalizedSpecies] = (acc[normalizedSpecies] || 0) + 1; + return acc; + }, {}); + + // Ensure minimum biodiversity for realistic Swedish forests + const minSpecies = ['scots pine', 'norway spruce', 'silver birch']; + minSpecies.forEach(species => { + if (!speciesDistribution[species]) { + speciesDistribution[species] = Math.max(1, Math.floor(trees.length * 0.15)); // 15% minimum per major species + } + }); + + const speciesCount = Math.max(Object.keys(speciesDistribution).length, 3); + const shannonIndex = Object.values(speciesDistribution) + .map(count => { + const p = count / trees.length; + return p > 0 ? -p * Math.log(p) : 0; + }) + .reduce((sum, value) => sum + value, 0); + + // Environmental benefits calculation + const totalForestArea = forests.reduce((sum, forest) => sum + (forest.area || 0), 0); + const co2Data = aliveTrees.map(tree => simulateTreeCO2Value(tree, currentDate)); + + const environmentalBenefits = { + carbonSequestration: { + annualTons: Math.round(co2Data.reduce((sum, data) => sum + data.annualAbsorption, 0) / 1000), + lifetimeStorage: Math.round(co2Data.reduce((sum, data) => sum + data.lifetimeStorage, 0) / 1000) + }, + oxygenProduction: { + annualTons: Math.round(co2Data.reduce((sum, data) => sum + data.annualAbsorption, 0) * 0.73 / 1000) // O2/CO2 ratio + }, + soilProtection: { + erosionPrevention: Math.round(totalForestArea * 2.5), // tons per hectare + waterFiltration: Math.round(totalForestArea * 15000) // liters per hectare per year + } + }; + + // Risk assessment + const treesAtRisk = aliveTrees.filter(tree => { + const age = calculateAge(tree.plantedDate); + const species = getSpeciesFactors(tree.species); + return age > species.maturityAge * 0.9 || + (tree.measurements && tree.measurements.length > 0 && + tree.measurements[tree.measurements.length - 1].healthStatus === 'poor'); + }); + + return { + biodiversity: { + speciesCount, + shannonIndex: Math.round(shannonIndex * 1000) / 1000, + dominantSpecies: Object.entries(speciesDistribution) + .sort(([,a], [,b]) => b - a) + .slice(0, 3) + .map(([species, count]) => ({ species, count, percentage: Math.round((count / trees.length) * 10000) / 100 })) + }, + environmental: environmentalBenefits, + riskAssessment: { + treesAtRisk: Math.max(treesAtRisk.length, Math.floor(aliveTrees.length * 0.05)), // Minimum 5% at risk is realistic + riskPercentage: Math.max(Math.round((treesAtRisk.length / aliveTrees.length) * 10000) / 100, 5.0), + mainRiskFactors: ['bark_beetle', 'storm_damage', 'disease', 'drought_stress'] + }, + forestHealth: { + overallScore: Math.round((aliveTrees.length / trees.length) * 100), + aliveTrees: aliveTrees.length, + totalTrees: trees.length + } + }; +}; + +/** + * Main simulation function that combines all metrics + * @param {Object} filters - Filter parameters (forestIds, dateRange, etc.) + * @returns {Object} Complete simulation data + */ +export const generateDashboardSimulation = async (filters = {}) => { + try { + // Build query from filters + const treeQuery = {}; + const forestQuery = { isActive: true }; + + // Handle forest filtering + if (filters.forestIds) { + const forestIdArray = filters.forestIds.split(',') + .map(id => id.trim()) + .filter(id => id.length === 24); // Valid ObjectId length + + if (forestIdArray.length > 0) { + const objectIds = forestIdArray.map(id => new mongoose.Types.ObjectId(id)); + treeQuery.forestId = { $in: objectIds }; + forestQuery._id = { $in: objectIds }; + } + } else if (filters.forestId) { + const objectId = new mongoose.Types.ObjectId(filters.forestId); + treeQuery.forestId = objectId; + forestQuery._id = objectId; + } + + // Handle date filtering + if (filters.startDate || filters.endDate) { + treeQuery.plantedDate = {}; + if (filters.startDate) treeQuery.plantedDate.$gte = new Date(filters.startDate); + if (filters.endDate) treeQuery.plantedDate.$lte = new Date(filters.endDate); + } + + // Handle other filters + if (filters.species) { + treeQuery.species = new RegExp(filters.species, 'i'); + } + + if (filters.isAlive !== undefined) { + treeQuery.isAlive = filters.isAlive === 'true'; + } + + // Fetch data + const [trees, forests] = await Promise.all([ + Tree.find(treeQuery).lean(), + Forest.find(forestQuery).lean() + ]); + + // Generate simulations + const [investorMetrics, ecologicalMetrics] = await Promise.all([ + simulateInvestorMetrics(trees, forests, filters), + simulateEcologicalMetrics(trees, forests, filters) + ]); + + return { + investor: investorMetrics, + ecological: ecologicalMetrics, + summary: { + totalTrees: trees.length, + totalForests: forests.length, + aliveTrees: trees.filter(t => t.isAlive).length, + filters, + lastUpdated: new Date().toISOString() + } + }; + + } catch (error) { + console.error('Simulation generation error:', error); + throw new Error('Failed to generate dashboard simulation'); + } +}; \ No newline at end of file diff --git a/backend/utils/treeHelpers.js b/backend/utils/treeHelpers.js new file mode 100644 index 0000000000..9664162944 --- /dev/null +++ b/backend/utils/treeHelpers.js @@ -0,0 +1,156 @@ +// Tree-specific utility functions for controllers and services + +import { validationResult } from 'express-validator'; +import { Forest } from '../models/index.js'; +import { + handleControllerError, + sendSuccessResponse, + sendNotFoundResponse, + sendValidationErrorResponse, + buildPaginationResponse +} from './dashboardUtils.js'; + +/** + * Handle validation errors for tree operations + * @param {Object} req - Express request object + * @param {Object} res - Express response object + * @returns {boolean} True if validation failed, false if passed + */ +export const handleValidationErrors = (req, res) => { + const errors = validationResult(req); + if (!errors.isEmpty()) { + sendValidationErrorResponse(res, errors.array()); + return true; + } + return false; +}; + +/** + * Verify forest exists and is active + * @param {string} forestId - Forest ID to verify + * @returns {Promise} Forest object if exists and active, null otherwise + */ +export const verifyForestExists = async (forestId) => { + const forest = await Forest.findById(forestId); + return (forest && forest.isActive) ? forest : null; +}; + +/** + * Transform tree data for mapping display + * @param {Array} trees - Array of tree documents + * @returns {Array} Array of transformed tree markers + */ +export const transformTreesForMapping = (trees) => { + return trees.map(tree => ({ + id: tree._id, + treeId: tree.treeId, + coordinates: tree.location.coordinates, + species: tree.species, + currentHeight: tree.currentHeight, + currentHealthStatus: tree.currentHealthStatus, + measurementCount: tree.measurements.length + })); +}; + +/** + * Create measurement data object for tree + * @param {Object} measurementBody - Request body with measurement data + * @param {string} userId - ID of user taking measurement + * @returns {Object} Formatted measurement object + */ +export const createMeasurementData = (measurementBody, userId) => { + return { + ...measurementBody, + measuredBy: userId, + measuredAt: new Date() + }; +}; + +/** + * Handle tree operation errors consistently + * @param {Object} res - Express response object + * @param {Error} error - Error object + * @param {string} operation - Operation that failed (e.g., 'Get trees') + */ +export const handleTreeError = (res, error, operation) => { + handleControllerError(res, error, `${operation} error`); +}; + +/** + * Send tree response with pagination + * @param {Object} res - Express response object + * @param {Array} trees - Tree data + * @param {number} totalCount - Total count of trees + * @param {number} page - Current page + * @param {number} limit - Items per page + * @param {string} message - Optional success message + */ +export const sendTreeListResponse = (res, trees, totalCount, page, limit, message = null) => { + const data = { + trees, + pagination: buildPaginationResponse(totalCount, page, limit) + }; + + sendSuccessResponse(res, data, message); +}; + +/** + * Send single tree response + * @param {Object} res - Express response object + * @param {Object} tree - Tree data + * @param {string} message - Optional success message + * @param {number} statusCode - HTTP status code + */ +export const sendTreeResponse = (res, tree, message = null, statusCode = 200) => { + sendSuccessResponse(res, { tree }, message, statusCode); +}; + +/** + * Send tree mapping response + * @param {Object} res - Express response object + * @param {Object} forest - Forest data + * @param {Array} trees - Transformed tree markers + */ +export const sendTreeMappingResponse = (res, forest, trees) => { + const data = { + forest: { + id: forest._id, + name: forest.name, + region: forest.region + }, + trees, + totalCount: trees.length + }; + + sendSuccessResponse(res, data); +}; + +/** + * Send measurement response + * @param {Object} res - Express response object + * @param {Object} tree - Updated tree with measurements + * @param {string} message - Success message + */ +export const sendMeasurementResponse = (res, tree, message = 'Measurement added successfully') => { + const data = { + tree, + latestMeasurement: tree.measurements[tree.measurements.length - 1] + }; + + sendSuccessResponse(res, data, message, 201); +}; + +/** + * Send tree measurements history response + * @param {Object} res - Express response object + * @param {string} treeId - Tree ID + * @param {Array} measurements - Measurements array + */ +export const sendMeasurementsHistoryResponse = (res, treeId, measurements) => { + const data = { + treeId, + measurements + }; + + sendSuccessResponse(res, data); +}; \ No newline at end of file diff --git a/frontend/.eslintrc.json b/frontend/.eslintrc.json new file mode 100644 index 0000000000..8fda746b04 --- /dev/null +++ b/frontend/.eslintrc.json @@ -0,0 +1,30 @@ +{ + "env": { + "browser": true, + "es2021": true, + "node": true + }, + "extends": [ + "eslint:recommended", + "plugin:react/recommended", + "plugin:react-hooks/recommended" + ], + "parserOptions": { + "ecmaVersion": "latest", + "sourceType": "module" + }, + "settings": { + "react": { + "version": "detect" + } + }, + "rules": { + "react/react-in-jsx-scope": "off", + "react/prop-types": "off", + "no-unused-vars": "warn" + }, + "ignorePatterns": [ + "dist/**/*", + "node_modules/**/*" + ] +} diff --git a/frontend/.prettierrc b/frontend/.prettierrc new file mode 100644 index 0000000000..bd8d004fc4 --- /dev/null +++ b/frontend/.prettierrc @@ -0,0 +1,8 @@ +{ + "semi": true, + "trailingComma": "es5", + "singleQuote": true, + "printWidth": 80, + "tabWidth": 2, + "useTabs": false +} \ No newline at end of file diff --git a/frontend/config/README.md b/frontend/config/README.md new file mode 100644 index 0000000000..1cc429383f --- /dev/null +++ b/frontend/config/README.md @@ -0,0 +1,28 @@ +# Configuration Files + +This directory contains all configuration files for the frontend application. + +## Files Overview + +| File | Purpose | Documentation | +| -------------------- | ----------------------------------------------------------- | ------------------------------------------------------------------- | +| `.eslintrc.js` | ESLint configuration for code quality and style enforcement | [ESLint Docs](https://eslint.org/docs/user-guide/configuring/) | +| `.prettierrc` | Prettier configuration for code formatting | [Prettier Docs](https://prettier.io/docs/en/configuration.html) | +| `tailwind.config.js` | Tailwind CSS configuration and design tokens | [Tailwind Docs](https://tailwindcss.com/docs/configuration) | +| `postcss.config.js` | PostCSS configuration for CSS processing | [PostCSS Docs](https://postcss.org/) | +| `vitest.config.js` | Vitest configuration for testing | [Vitest Docs](https://vitest.dev/config/) | +| `lighthouserc.json` | Lighthouse CI configuration for performance monitoring | [Lighthouse CI Docs](https://github.com/GoogleChrome/lighthouse-ci) | + +## Usage + +Most tools automatically detect their config files in this directory through explicit references in: + +- `package.json` scripts +- `vite.config.js` +- GitHub Actions workflow + +## Notes + +- Paths in config files are relative to the project root, not the config directory +- Changes to these files may require restarting development servers +- All configs follow the project's ESLint and Prettier rules diff --git a/frontend/config/lighthouserc.json b/frontend/config/lighthouserc.json new file mode 100644 index 0000000000..be5ff202d6 --- /dev/null +++ b/frontend/config/lighthouserc.json @@ -0,0 +1,20 @@ +{ + "ci": { + "collect": { + "startServerCommand": "npm run preview", + "url": ["http://localhost:4173"], + "numberOfRuns": 3 + }, + "assert": { + "assertions": { + "categories:performance": ["error", {"minScore": 0.9}], + "categories:accessibility": ["error", {"minScore": 0.9}], + "categories:best-practices": ["error", {"minScore": 0.9}], + "categories:seo": ["error", {"minScore": 0.9}] + } + }, + "upload": { + "target": "temporary-public-storage" + } + } +} \ No newline at end of file diff --git a/frontend/config/postcss.config.js b/frontend/config/postcss.config.js new file mode 100644 index 0000000000..760d963261 --- /dev/null +++ b/frontend/config/postcss.config.js @@ -0,0 +1,9 @@ +import tailwindcss from 'tailwindcss' +import autoprefixer from 'autoprefixer' + +export default { + plugins: [ + tailwindcss({ config: './tailwind.config.js' }), + autoprefixer, + ], +} diff --git a/frontend/config/vitest.config.js b/frontend/config/vitest.config.js new file mode 100644 index 0000000000..0cb7e064a6 --- /dev/null +++ b/frontend/config/vitest.config.js @@ -0,0 +1,22 @@ +import { defineConfig } from 'vitest/config' +import react from '@vitejs/plugin-react' +import path from 'path' + +export default defineConfig({ + plugins: [react()], + test: { + globals: true, + environment: 'jsdom', + setupFiles: './src/test/setup.js', + }, + resolve: { + alias: { + '@': path.resolve(__dirname, '../src'), + '@components': path.resolve(__dirname, '../src/components'), + '@pages': path.resolve(__dirname, '../src/pages'), + '@hooks': path.resolve(__dirname, '../src/hooks'), + '@utils': path.resolve(__dirname, '../src/utils'), + '@assets': path.resolve(__dirname, '../src/assets'), + }, + }, +}) \ No newline at end of file diff --git a/frontend/dist/assets/ChartComponents-BKyCVLLP.js b/frontend/dist/assets/ChartComponents-BKyCVLLP.js new file mode 100644 index 0000000000..c70482e549 --- /dev/null +++ b/frontend/dist/assets/ChartComponents-BKyCVLLP.js @@ -0,0 +1,57 @@ +import{j as n,d as r}from"./index-tEqMizXb.js";const j=r.div` + background: white; + border-radius: 0.75rem; + padding: 1.5rem; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; + height: 450px; + display: flex; + flex-direction: column; + overflow: hidden; +`,w=r.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; +`,C=r.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`,c=r.div` + background: white; + border: 1px solid #e5e7eb; + border-radius: 0.5rem; + padding: 0.75rem; + box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.1); +`,l=r.div` + text-align: center; + margin-bottom: 1rem; +`,m=r.div` + font-size: 2.5rem; + font-weight: 700; + color: #10b981; + line-height: 1; +`,g=r.div` + font-size: 0.875rem; + color: #6b7280; + margin-top: 0.25rem; +`,x=r.div` + display: flex; + justify-content: center; + gap: 1rem; + margin-top: 1rem; + flex-wrap: wrap; +`,p=r.div` + display: flex; + align-items: center; +`,h=r.div` + width: 0.75rem; + height: 0.75rem; + border-radius: 50%; + margin-right: 0.5rem; + background-color: ${t=>t.color}; +`,b=r.span` + font-size: 0.875rem; + color: #374151; +`,f=(t,e,o)=>e?e(t):`${t}${o}`,v=({active:t,payload:e,label:o,valueFormatter:i,unit:s=""})=>t&&e&&e.length?n.jsxs(c,{children:[n.jsx("p",{className:"font-medium",children:o}),e.map((a,d)=>n.jsx("p",{className:"text-sm text-gray-600",children:f(a.value,i,s)},d))]}):null,y=({value:t,label:e})=>n.jsxs(l,{children:[n.jsxs(m,{children:[t,"%"]}),n.jsx(g,{children:e})]}),L=({data:t})=>n.jsx(x,{children:t.map((e,o)=>n.jsxs(p,{children:[n.jsx(h,{color:e.color}),n.jsxs(b,{children:[e.name," (",e.value,"%)"]})]},o))});export{j as C,w as a,C as b,y as c,v as d,L as e}; diff --git a/frontend/dist/assets/DarkModeToggle-ZfYhBz0e.js b/frontend/dist/assets/DarkModeToggle-ZfYhBz0e.js new file mode 100644 index 0000000000..2d83391681 --- /dev/null +++ b/frontend/dist/assets/DarkModeToggle-ZfYhBz0e.js @@ -0,0 +1 @@ +import{j as r,m as i,n as a}from"./index-tEqMizXb.js";import{I as l}from"./IconButton-DmJ5sToe.js";const d=({className:t="",...o})=>r.jsx("svg",{className:t,fill:"none",stroke:"currentColor",viewBox:"0 0 24 24","aria-hidden":"true",...o,children:r.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 3v1m0 16v1m9-9h-1M4 12H3m15.364 6.364l-.707-.707M6.343 6.343l-.707-.707m12.728 0l-.707.707M6.343 17.657l-.707.707M16 12a4 4 0 11-8 0 4 4 0 018 0z"})}),c=({className:t="",...o})=>r.jsx("svg",{className:t,fill:"none",stroke:"currentColor",viewBox:"0 0 24 24","aria-hidden":"true",...o,children:r.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M20.354 15.354A9 9 0 018.646 3.646 9.003 9.003 0 0012 21a9.003 9.003 0 008.354-5.646z"})}),g=({className:t="",size:o="md"})=>{const{isDarkMode:e,toggleDarkMode:s}=i(),n={sm:"w-4 h-4",md:"w-5 h-5",lg:"w-6 h-6"};return r.jsxs(l,{onClick:s,className:a("relative bg-gray-100 hover:bg-gray-200 dark:bg-gray-800 dark:hover:bg-gray-700 text-gray-600 dark:text-gray-300",t),size:o,"aria-label":e?"Switch to light mode":"Switch to dark mode",title:e?"Switch to light mode":"Switch to dark mode",children:[r.jsx(d,{className:a(n[o],"transition-all duration-200",e?"rotate-90 scale-0":"rotate-0 scale-100")}),r.jsx(c,{className:a(n[o],"absolute transition-all duration-200",e?"rotate-0 scale-100":"-rotate-90 scale-0")})]})};export{g as D}; diff --git a/frontend/dist/assets/EcologicalDashboardPage-B0Gl5Ex5.js b/frontend/dist/assets/EcologicalDashboardPage-B0Gl5Ex5.js new file mode 100644 index 0000000000..647d0a83fe --- /dev/null +++ b/frontend/dist/assets/EcologicalDashboardPage-B0Gl5Ex5.js @@ -0,0 +1 @@ +import{r as v,j as e,b as J,L as Y}from"./index-tEqMizXb.js";import{C,a as I,b as L}from"./ChartComponents-BKyCVLLP.js";import{a2 as Z,a3 as _,R as q,a as G,X as V,Y as O,T as X,b as Q,B as ee,d as M,L as ae,e as te,E as se}from"./EnhancedStatCard-CVgqadrW.js";import{A as re,a as R,u as ne}from"./useDashboardStats-3Mt6zhuV.js";import{G as ie}from"./GlobalFilters-B5C4gkva.js";import{u as oe,D as de,a as le}from"./useSidebarState-CJJaSDFr.js";import"./vendor-BtP0CW_r.js";import"./DarkModeToggle-ZfYhBz0e.js";import"./IconButton-DmJ5sToe.js";var ce=["axis"],he=v.forwardRef((S,h)=>v.createElement(Z,{chartName:"ComposedChart",defaultTooltipEventType:"axis",validateTooltipEventTypes:ce,tooltipPayloadSearcher:_,categoricalChartProps:S,ref:h}));const $=({filters:S={},chartType:h="line"})=>{const[k,u]=v.useState([]),[W,x]=v.useState(!0),[F,E]=v.useState(null);v.useEffect(()=>{(async()=>{var s,c,i;try{x(!0),E(null);const r=(await te.getStats(S)).data,l=(s=r==null?void 0:r.manager)==null?void 0:s.biodiversity,o=(c=r==null?void 0:r.manager)==null?void 0:c.speciesDiversity,xe=(i=r==null?void 0:r.manager)==null?void 0:i.soilHealth;if(h==="area"){const A=new Date().getFullYear(),j=[];for(let d=-6;d<=0;d++){const b=A+d,a=(6+d)/6,n=(o==null?void 0:o.uniqueSpecies)||25,w=(o==null?void 0:o.endangeredSpecies)||3,y=Math.round(n*.15*a*(.8+Math.random()*.4)),m=Math.round(n*.35*a*(.9+Math.random()*.2)),f=Math.round(n*.3*a*(1+Math.random()*.3)),g=Math.round(n*.2*a*(.85+Math.random()*.3)),N=Math.max(0,Math.round(w*(1.2-a*.4))),D=Math.round(n*.1*a),H=y+m+f+g-N-D;j.push({year:b,yearLabel:b.toString(),mammals:y,birds:m,insects:f,plants:g,endangered:N,vulnerable:D,stable:Math.max(0,H),totalSpecies:y+m+f+g,biodiversityIndex:Math.round(((l==null?void 0:l.averageIndex)||5)*a*10)/10,endemicSpecies:Math.round(n*.05*a),newSpeciesDiscovered:Math.floor(Math.random()*3)+(d===0?1:0)})}u(j)}else if(h==="composed"){const A=new Date().getFullYear(),j=[];for(let d=-5;d<=0;d++){const b=A+d,a=(5+d)/5,n=Math.round((((l==null?void 0:l.averageIndex)||2.5)*a+(Math.random()-.5)*.3)*10)/10,w=Math.round((.8*a+(Math.random()-.5)*.1)*100)/100,y=Math.round(((o==null?void 0:o.uniqueSpecies)||20)*a*(.9+Math.random()*.2)),m=Math.round((.7+a*.2+(Math.random()-.5)*.1)*100)/100,f=Math.round(60+a*30+Math.random()*10),g=Math.round(50+a*40+(Math.random()-.5)*10),N=Math.max(0,Math.round(8-a*5+(Math.random()-.5)*2));j.push({year:b,yearLabel:b.toString(),shannonIndex:n,simpsonIndex:Math.round(w*100),speciesRichness:y,evenness:Math.round(m*100),connectivity:f,habitatQuality:g,invasiveSpecies:N,overallBiodiversityScore:Math.round((n*10+y/2+f+g)/4)})}u(j)}else{const A=new Date().getFullYear(),j=[];for(let d=-7;d<=0;d++){const b=A+d,a=(7+d)/7,n=Math.round(((o==null?void 0:o.uniqueSpecies)||25)*a*(.9+Math.random()*.2)),w=Math.max(0,Math.round(((o==null?void 0:o.endangeredSpecies)||5)*(1.3-a*.5))),y=Math.round(((l==null?void 0:l.averageIndex)||3)*a*10+Math.random()*5)/10,m=Math.round(n*(.85+a*.1)),f=n-m,g=Math.round(100*a*(.8+Math.random()*.4)),N=Math.round(80*a*(.9+Math.random()*.2)),D=Math.round(120*a*(.85+Math.random()*.3)),H=Math.round(50+a*40+(Math.random()-.5)*10),U=Math.round((.65+a*.2+(Math.random()-.5)*.05)*100);j.push({year:b,yearLabel:b.toString(),totalSpecies:n,endangeredSpecies:w,biodiversityIndex:y,nativeSpecies:m,nonNativeSpecies:f,pollinatorPopulation:g,predatorPopulation:N,herbivorePopulation:D,connectivityIndex:H,geneticDiversity:U,nativeRatio:Math.round(m/n*100),populationHealthIndex:Math.round((g+N+D)/3),conservationEffectiveness:Math.round((n-w)/n*100)})}u(j)}}catch(p){console.error("Error fetching biodiversity trends data:",p),E(p.message||"Failed to load biodiversity data")}finally{x(!1)}})()},[S,h]);const T=t=>t?t.toLocaleString():"0",K=t=>t?t.toFixed(1):"0.0",P=t=>t?`${Math.round(t)}%`:"0%",B=({active:t,payload:s,label:c})=>t&&s&&s.length?(s[0].payload,e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white mb-2",children:c}),e.jsx("div",{className:"space-y-1",children:s.map((i,p)=>e.jsxs("p",{className:"text-sm",style:{color:i.color},children:[i.name,": ",i.name.includes("Index")||i.name.includes("index")?K(i.value):i.name.includes("%")||i.name.includes("Ratio")||i.name.includes("Effectiveness")?P(i.value):T(i.value)]},p))})]})):null,z=({active:t,payload:s,label:c})=>{if(t&&s&&s.length){const i=s[0].payload,p=s.reduce((r,l)=>r+l.value,0);return e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white mb-2",children:c}),e.jsxs("div",{className:"space-y-1",children:[e.jsxs("p",{className:"text-sm font-medium text-green-600 dark:text-green-400",children:["Total Species: ",T(p)]}),s.map((r,l)=>e.jsxs("p",{className:"text-sm",style:{color:r.color},children:[r.name,": ",T(r.value)," (",(r.value/p*100).toFixed(0),"%)"]},l)),e.jsx("div",{className:"border-t pt-1 mt-1",children:e.jsxs("p",{className:"text-xs text-gray-500 dark:text-gray-400",children:["Biodiversity Index: ",K(i.biodiversityIndex)]})})]})]})}return null};return W?e.jsxs(C,{children:[e.jsx(I,{children:e.jsx(L,{children:"Biodiversity Trends"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx(J,{size:"32px",text:"Loading biodiversity data..."})})]}):F?e.jsxs(C,{children:[e.jsx(I,{children:e.jsx(L,{children:"Biodiversity Trends"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsxs("div",{className:"text-center",children:[e.jsx("p",{className:"text-red-600 mb-2",children:"Error loading chart data"}),e.jsx("p",{className:"text-sm text-gray-500",children:F})]})})]}):!k||k.length===0?e.jsxs(C,{children:[e.jsx(I,{children:e.jsx(L,{children:"Biodiversity Trends"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx("p",{className:"text-gray-500",children:"No biodiversity data available"})})]}):h==="area"?e.jsxs(C,{children:[e.jsxs(I,{children:[e.jsx(L,{children:"Species Composition Trends"}),e.jsx("p",{className:"text-sm text-gray-500 dark:text-gray-400",children:"Evolution of species diversity by taxonomic groups"})]}),e.jsx(q,{width:"100%",height:300,children:e.jsxs(re,{data:k,margin:{top:10,right:30,left:0,bottom:0},children:[e.jsx(G,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(V,{dataKey:"yearLabel",className:"text-xs",tick:{fontSize:12}}),e.jsx(O,{className:"text-xs",tick:{fontSize:12}}),e.jsx(X,{content:e.jsx(z,{})}),e.jsx(Q,{}),e.jsx(R,{type:"monotone",dataKey:"plants",stackId:"1",stroke:"#10b981",fill:"#10b981",name:"Plants",fillOpacity:.8}),e.jsx(R,{type:"monotone",dataKey:"insects",stackId:"1",stroke:"#f59e0b",fill:"#f59e0b",name:"Insects",fillOpacity:.8}),e.jsx(R,{type:"monotone",dataKey:"birds",stackId:"1",stroke:"#3b82f6",fill:"#3b82f6",name:"Birds",fillOpacity:.8}),e.jsx(R,{type:"monotone",dataKey:"mammals",stackId:"1",stroke:"#8b5cf6",fill:"#8b5cf6",name:"Mammals",fillOpacity:.8})]})})]}):h==="composed"?e.jsxs(C,{children:[e.jsxs(I,{children:[e.jsx(L,{children:"Biodiversity Metrics Dashboard"}),e.jsx("p",{className:"text-sm text-gray-500 dark:text-gray-400",children:"Comprehensive biodiversity health indicators"})]}),e.jsx(q,{width:"100%",height:300,children:e.jsxs(he,{data:k,margin:{top:5,right:30,left:20,bottom:5},children:[e.jsx(G,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(V,{dataKey:"yearLabel",className:"text-xs",tick:{fontSize:12}}),e.jsx(O,{yAxisId:"left",className:"text-xs",tick:{fontSize:12}}),e.jsx(O,{yAxisId:"right",orientation:"right",className:"text-xs",tick:{fontSize:12}}),e.jsx(X,{content:e.jsx(B,{})}),e.jsx(Q,{}),e.jsx(ee,{yAxisId:"left",dataKey:"speciesRichness",fill:"#e5e7eb",name:"Species Richness",opacity:.6}),e.jsx(M,{yAxisId:"right",type:"monotone",dataKey:"shannonIndex",stroke:"#10b981",strokeWidth:3,name:"Shannon Index",dot:{r:4,fill:"#10b981"}}),e.jsx(M,{yAxisId:"right",type:"monotone",dataKey:"connectivity",stroke:"#3b82f6",strokeWidth:2,name:"Habitat Connectivity",dot:{r:3,fill:"#3b82f6"}}),e.jsx(M,{yAxisId:"left",type:"monotone",dataKey:"invasiveSpecies",stroke:"#ef4444",strokeWidth:2,name:"Invasive Species",dot:{r:3,fill:"#ef4444"},strokeDasharray:"5 5"})]})})]}):e.jsxs(C,{children:[e.jsxs(I,{children:[e.jsx(L,{children:"Biodiversity Trends Over Time"}),e.jsx("p",{className:"text-sm text-gray-500 dark:text-gray-400",children:"Long-term trends in species diversity and conservation status"})]}),e.jsx(q,{width:"100%",height:300,children:e.jsxs(ae,{data:k,margin:{top:5,right:30,left:20,bottom:5},children:[e.jsx(G,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(V,{dataKey:"yearLabel",className:"text-xs",tick:{fontSize:12}}),e.jsx(O,{className:"text-xs",tick:{fontSize:12}}),e.jsx(X,{content:e.jsx(B,{})}),e.jsx(Q,{}),e.jsx(M,{type:"monotone",dataKey:"totalSpecies",stroke:"#10b981",strokeWidth:3,name:"Total Species",dot:{r:4,fill:"#10b981"}}),e.jsx(M,{type:"monotone",dataKey:"nativeSpecies",stroke:"#3b82f6",strokeWidth:2,name:"Native Species",dot:{r:3,fill:"#3b82f6"}}),e.jsx(M,{type:"monotone",dataKey:"endangeredSpecies",stroke:"#ef4444",strokeWidth:2,name:"Endangered Species",dot:{r:3,fill:"#ef4444"},strokeDasharray:"5 5"}),e.jsx(M,{type:"monotone",dataKey:"biodiversityIndex",stroke:"#8b5cf6",strokeWidth:2,name:"Biodiversity Index",dot:{r:3,fill:"#8b5cf6"}})]})})]})},ke=()=>{var B,z,t,s;const{sidebarOpen:S,toggleSidebar:h,closeSidebar:k}=oe(),[u,W]=v.useState({}),{stats:x,loading:F,error:E,refresh:T}=ne(u),K=v.useCallback(c=>{W(c)},[]),P=c=>c.toLocaleString();return e.jsxs("div",{className:"min-h-screen bg-gray-50 dark:bg-gray-900 flex flex-col",children:[e.jsx(de,{onToggleSidebar:h}),e.jsx(le,{isOpen:S,onClose:k}),e.jsx("main",{className:"flex-1 p-4 md:p-6 lg:p-8 lg:ml-64",children:e.jsxs("div",{className:"max-w-7xl mx-auto",children:[e.jsx("div",{className:"mb-8",children:e.jsx("div",{className:"flex justify-between items-center mb-4",children:e.jsxs("div",{children:[e.jsx("nav",{className:"flex mb-2","aria-label":"Breadcrumb",children:e.jsxs("ol",{className:"flex items-center space-x-4",children:[e.jsx("li",{children:e.jsx(Y,{to:"/dashboard",className:"text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-300",children:"Overview"})}),e.jsx("li",{children:e.jsx("svg",{className:"flex-shrink-0 h-4 w-4 text-gray-400",fill:"currentColor",viewBox:"0 0 20 20","aria-hidden":"true",children:e.jsx("path",{d:"M5.555 17.776l8-16 .894.448-8 16-.894-.448z"})})}),e.jsx("li",{children:e.jsx("span",{className:"text-gray-900 dark:text-white font-medium",children:"Ecological Dashboard"})})]})}),e.jsx("h2",{className:"text-3xl font-bold text-gray-900 dark:text-white mb-2",children:"Ecological Dashboard"}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300",children:"Environmental impact, biodiversity tracking, and conservation metrics for sustainable forest management."})]})})}),e.jsx(ie,{onFiltersChange:K}),F?e.jsx("div",{className:"flex justify-center items-center py-12",children:e.jsx(J,{text:"Loading ecological data..."})}):E?e.jsxs("div",{className:"bg-red-50 border border-red-200 rounded-lg p-4 mb-8",children:[e.jsxs("p",{className:"text-red-600",children:["Error loading ecological statistics: ",E]}),e.jsx("button",{onClick:T,className:"mt-2 text-red-600 hover:text-red-800 underline",children:"Try again"})]}):e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Forest Health Metrics"}),e.jsx("div",{className:"grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6",children:e.jsx(se,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M19.428 15.428a2 2 0 00-1.022-.547l-2.387-.477a6 6 0 00-3.86.517l-.318.158a6 6 0 01-3.86.517L6.05 15.21a2 2 0 00-1.806.547M8 4h8l-1 1v5.172a2 2 0 00.586 1.414l5 5c1.26 1.26.367 3.414-1.415 3.414H4.828c-1.782 0-2.674-2.154-1.414-3.414l5-5A2 2 0 009 10.172V5L8 4z"})}),title:"Number of Species",value:P(((z=(B=x==null?void 0:x.manager)==null?void 0:B.biodiversity)==null?void 0:z.totalSpecies)||((s=(t=x==null?void 0:x.ecological)==null?void 0:t.biodiversity)==null?void 0:s.speciesCount)||10),subtitle:"Different tree species in the system",color:"purple"})})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Biodiversity Analysis"}),e.jsxs("div",{className:"grid grid-cols-1 lg:grid-cols-2 gap-6",children:[e.jsx("div",{className:"col-span-1",children:e.jsx($,{filters:u,chartType:"line"})}),e.jsx("div",{className:"col-span-1",children:e.jsx($,{filters:u,chartType:"area"})}),e.jsx("div",{className:"col-span-1 lg:col-span-2",children:e.jsx($,{filters:u,chartType:"composed"})})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Related Analysis"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-6",children:[e.jsx(Y,{to:"/dashboard",className:"block",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow-md hover:shadow-lg transition-shadow duration-200 p-6 border border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600",children:[e.jsxs("div",{className:"flex items-center mb-3",children:[e.jsx("div",{className:"p-2 bg-gray-100 dark:bg-gray-700 rounded-lg mr-4",children:e.jsx("svg",{className:"h-5 w-5 text-gray-600 dark:text-gray-400",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"})})}),e.jsx("h4",{className:"text-lg font-medium text-gray-900 dark:text-white",children:"Overview Dashboard"})]}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300 text-sm",children:"Return to the main overview for quick insights across all metrics."})]})}),e.jsx(Y,{to:"/dashboard/financial",className:"block",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow-md hover:shadow-lg transition-shadow duration-200 p-6 border border-gray-200 dark:border-gray-700 hover:border-blue-300 dark:hover:border-blue-600",children:[e.jsxs("div",{className:"flex items-center mb-3",children:[e.jsx("div",{className:"p-2 bg-blue-100 dark:bg-blue-900 rounded-lg mr-4",children:e.jsx("svg",{className:"h-5 w-5 text-blue-600 dark:text-blue-400",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M13 7h8m0 0v8m0-8l-8 8-4-4-6 6"})})}),e.jsx("h4",{className:"text-lg font-medium text-gray-900 dark:text-white",children:"Financial Dashboard"})]}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300 text-sm",children:"Explore the financial implications of your conservation efforts and environmental performance."})]})})]})]})]})})]})};export{ke as EcologicalDashboardPage}; diff --git a/frontend/dist/assets/EnhancedStatCard-CVgqadrW.js b/frontend/dist/assets/EnhancedStatCard-CVgqadrW.js new file mode 100644 index 0000000000..4d916ae140 --- /dev/null +++ b/frontend/dist/assets/EnhancedStatCard-CVgqadrW.js @@ -0,0 +1,69 @@ +var eg=Object.defineProperty;var tg=(e,t,r)=>t in e?eg(e,t,{enumerable:!0,configurable:!0,writable:!0,value:r}):e[t]=r;var Ji=(e,t,r)=>tg(e,typeof t!="symbol"?t+"":t,r);import{A as rg,r as y,c as H,e as hh,j as be}from"./index-tEqMizXb.js";import{g as At,r as Cu}from"./vendor-BtP0CW_r.js";const Yr=new rg,EM={getStats:(e={})=>Yr.get("/dashboard/stats",e),getEnhancedStats:(e={})=>Yr.get("/dashboard/enhanced-stats",e)},_M={getSurvivalRate:(e={})=>Yr.get("/charts/survival-rate",e),getHeightGrowth:(e={})=>Yr.get("/charts/height-growth",e),getCO2Absorption:(e={})=>Yr.get("/charts/co2-absorption",e)};var Qi={},ea={},pc;function ng(){return pc||(pc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return r==="__proto__"}e.isUnsafeProperty=t}(ea)),ea}var ta={},mc;function vh(){return mc||(mc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){switch(typeof r){case"number":case"symbol":return!1;case"string":return r.includes(".")||r.includes("[")||r.includes("]")}}e.isDeepKey=t}(ta)),ta}var ra={},yc;function ph(){return yc||(yc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){var n;return typeof r=="string"||typeof r=="symbol"?r:Object.is((n=r==null?void 0:r.valueOf)==null?void 0:n.call(r),-0)?"-0":String(r)}e.toKey=t}(ra)),ra}var na={},gc;function ku(){return gc||(gc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){const n=[],i=r.length;if(i===0)return n;let a=0,o="",u="",c=!1;for(r.charCodeAt(0)===46&&(n.push(""),a++);ae===0?0:e>0?1:-1,qe=e=>typeof e=="number"&&e!=+e,Gt=e=>typeof e=="string"&&e.indexOf("%")===e.length-1,k=e=>(typeof e=="number"||e instanceof Number)&&!qe(e),ft=e=>k(e)||typeof e=="string",cg=0,xr=e=>{var t=++cg;return"".concat(e||"").concat(t)},ut=function(t,r){var n=arguments.length>2&&arguments[2]!==void 0?arguments[2]:0,i=arguments.length>3&&arguments[3]!==void 0?arguments[3]:!1;if(!k(t)&&typeof t!="string")return n;var a;if(Gt(t)){if(r==null)return n;var o=t.indexOf("%");a=r*parseFloat(t.slice(0,o))/100}else a=+t;return qe(a)&&(a=n),i&&r!=null&&a>r&&(a=r),a},mh=e=>{if(!Array.isArray(e))return!1;for(var t=e.length,r={},n=0;nk(e)&&k(t)?r=>e+r*(t-e):()=>t;function jM(e,t,r){return k(e)&&k(t)?e+r*(t-e):t}function yh(e,t,r){if(!(!e||!e.length))return e.find(n=>n&&(typeof t=="function"?t(n):nr(n,t))===r)}var TM=e=>{if(!e||!e.length)return null;for(var t=e.length,r=0,n=0,i=0,a=0,o=1/0,u=-1/0,c=0,l=0,s=0;se===null||typeof e>"u",cn=e=>X(e)?e:"".concat(e.charAt(0).toUpperCase()).concat(e.slice(1)),lg=["viewBox","children"],fg=["aria-activedescendant","aria-atomic","aria-autocomplete","aria-busy","aria-checked","aria-colcount","aria-colindex","aria-colspan","aria-controls","aria-current","aria-describedby","aria-details","aria-disabled","aria-errormessage","aria-expanded","aria-flowto","aria-haspopup","aria-hidden","aria-invalid","aria-keyshortcuts","aria-label","aria-labelledby","aria-level","aria-live","aria-modal","aria-multiline","aria-multiselectable","aria-orientation","aria-owns","aria-placeholder","aria-posinset","aria-pressed","aria-readonly","aria-relevant","aria-required","aria-roledescription","aria-rowcount","aria-rowindex","aria-rowspan","aria-selected","aria-setsize","aria-sort","aria-valuemax","aria-valuemin","aria-valuenow","aria-valuetext","className","color","height","id","lang","max","media","method","min","name","style","target","width","role","tabIndex","accentHeight","accumulate","additive","alignmentBaseline","allowReorder","alphabetic","amplitude","arabicForm","ascent","attributeName","attributeType","autoReverse","azimuth","baseFrequency","baselineShift","baseProfile","bbox","begin","bias","by","calcMode","capHeight","clip","clipPath","clipPathUnits","clipRule","colorInterpolation","colorInterpolationFilters","colorProfile","colorRendering","contentScriptType","contentStyleType","cursor","cx","cy","d","decelerate","descent","diffuseConstant","direction","display","divisor","dominantBaseline","dur","dx","dy","edgeMode","elevation","enableBackground","end","exponent","externalResourcesRequired","fill","fillOpacity","fillRule","filter","filterRes","filterUnits","floodColor","floodOpacity","focusable","fontFamily","fontSize","fontSizeAdjust","fontStretch","fontStyle","fontVariant","fontWeight","format","from","fx","fy","g1","g2","glyphName","glyphOrientationHorizontal","glyphOrientationVertical","glyphRef","gradientTransform","gradientUnits","hanging","horizAdvX","horizOriginX","href","ideographic","imageRendering","in2","in","intercept","k1","k2","k3","k4","k","kernelMatrix","kernelUnitLength","kerning","keyPoints","keySplines","keyTimes","lengthAdjust","letterSpacing","lightingColor","limitingConeAngle","local","markerEnd","markerHeight","markerMid","markerStart","markerUnits","markerWidth","mask","maskContentUnits","maskUnits","mathematical","mode","numOctaves","offset","opacity","operator","order","orient","orientation","origin","overflow","overlinePosition","overlineThickness","paintOrder","panose1","pathLength","patternContentUnits","patternTransform","patternUnits","pointerEvents","pointsAtX","pointsAtY","pointsAtZ","preserveAlpha","preserveAspectRatio","primitiveUnits","r","radius","refX","refY","renderingIntent","repeatCount","repeatDur","requiredExtensions","requiredFeatures","restart","result","rotate","rx","ry","seed","shapeRendering","slope","spacing","specularConstant","specularExponent","speed","spreadMethod","startOffset","stdDeviation","stemh","stemv","stitchTiles","stopColor","stopOpacity","strikethroughPosition","strikethroughThickness","string","stroke","strokeDasharray","strokeDashoffset","strokeLinecap","strokeLinejoin","strokeMiterlimit","strokeOpacity","strokeWidth","surfaceScale","systemLanguage","tableValues","targetX","targetY","textAnchor","textDecoration","textLength","textRendering","to","transform","u1","u2","underlinePosition","underlineThickness","unicode","unicodeBidi","unicodeRange","unitsPerEm","vAlphabetic","values","vectorEffect","version","vertAdvY","vertOriginX","vertOriginY","vHanging","vIdeographic","viewTarget","visibility","vMathematical","widths","wordSpacing","writingMode","x1","x2","x","xChannelSelector","xHeight","xlinkActuate","xlinkArcrole","xlinkHref","xlinkRole","xlinkShow","xlinkTitle","xlinkType","xmlBase","xmlLang","xmlns","xmlnsXlink","xmlSpace","y1","y2","y","yChannelSelector","z","zoomAndPan","ref","key","angle"],Pc=["points","pathLength"],oa={svg:lg,polygon:Pc,polyline:Pc},Iu=["dangerouslySetInnerHTML","onCopy","onCopyCapture","onCut","onCutCapture","onPaste","onPasteCapture","onCompositionEnd","onCompositionEndCapture","onCompositionStart","onCompositionStartCapture","onCompositionUpdate","onCompositionUpdateCapture","onFocus","onFocusCapture","onBlur","onBlurCapture","onChange","onChangeCapture","onBeforeInput","onBeforeInputCapture","onInput","onInputCapture","onReset","onResetCapture","onSubmit","onSubmitCapture","onInvalid","onInvalidCapture","onLoad","onLoadCapture","onError","onErrorCapture","onKeyDown","onKeyDownCapture","onKeyPress","onKeyPressCapture","onKeyUp","onKeyUpCapture","onAbort","onAbortCapture","onCanPlay","onCanPlayCapture","onCanPlayThrough","onCanPlayThroughCapture","onDurationChange","onDurationChangeCapture","onEmptied","onEmptiedCapture","onEncrypted","onEncryptedCapture","onEnded","onEndedCapture","onLoadedData","onLoadedDataCapture","onLoadedMetadata","onLoadedMetadataCapture","onLoadStart","onLoadStartCapture","onPause","onPauseCapture","onPlay","onPlayCapture","onPlaying","onPlayingCapture","onProgress","onProgressCapture","onRateChange","onRateChangeCapture","onSeeked","onSeekedCapture","onSeeking","onSeekingCapture","onStalled","onStalledCapture","onSuspend","onSuspendCapture","onTimeUpdate","onTimeUpdateCapture","onVolumeChange","onVolumeChangeCapture","onWaiting","onWaitingCapture","onAuxClick","onAuxClickCapture","onClick","onClickCapture","onContextMenu","onContextMenuCapture","onDoubleClick","onDoubleClickCapture","onDrag","onDragCapture","onDragEnd","onDragEndCapture","onDragEnter","onDragEnterCapture","onDragExit","onDragExitCapture","onDragLeave","onDragLeaveCapture","onDragOver","onDragOverCapture","onDragStart","onDragStartCapture","onDrop","onDropCapture","onMouseDown","onMouseDownCapture","onMouseEnter","onMouseLeave","onMouseMove","onMouseMoveCapture","onMouseOut","onMouseOutCapture","onMouseOver","onMouseOverCapture","onMouseUp","onMouseUpCapture","onSelect","onSelectCapture","onTouchCancel","onTouchCancelCapture","onTouchEnd","onTouchEndCapture","onTouchMove","onTouchMoveCapture","onTouchStart","onTouchStartCapture","onPointerDown","onPointerDownCapture","onPointerMove","onPointerMoveCapture","onPointerUp","onPointerUpCapture","onPointerCancel","onPointerCancelCapture","onPointerEnter","onPointerEnterCapture","onPointerLeave","onPointerLeaveCapture","onPointerOver","onPointerOverCapture","onPointerOut","onPointerOutCapture","onGotPointerCapture","onGotPointerCaptureCapture","onLostPointerCapture","onLostPointerCaptureCapture","onScroll","onScrollCapture","onWheel","onWheelCapture","onAnimationStart","onAnimationStartCapture","onAnimationEnd","onAnimationEndCapture","onAnimationIteration","onAnimationIterationCapture","onTransitionEnd","onTransitionEndCapture"],Du=(e,t)=>{if(!e||typeof e=="function"||typeof e=="boolean")return null;var r=e;if(y.isValidElement(e)&&(r=e.props),typeof r!="object"&&typeof r!="function")return null;var n={};return Object.keys(r).forEach(i=>{Iu.includes(i)&&(n[i]=a=>r[i](r,a))}),n},dg=(e,t,r)=>n=>(e(t,r,n),null),mi=(e,t,r)=>{if(e===null||typeof e!="object"&&typeof e!="function")return null;var n=null;return Object.keys(e).forEach(i=>{var a=e[i];Iu.includes(i)&&typeof a=="function"&&(n||(n={}),n[i]=dg(a,t,r))}),n},Ac=e=>typeof e=="string"?e:e?e.displayName||e.name||"Component":"",Sc=null,ua=null,gh=e=>{if(e===Sc&&Array.isArray(ua))return ua;var t=[];return y.Children.forEach(e,r=>{X(r)||(sg.isFragment(r)?t=t.concat(gh(r.props.children)):t.push(r))}),ua=t,Sc=e,t};function Nu(e,t){var r=[],n=[];return Array.isArray(t)?n=t.map(i=>Ac(i)):n=[Ac(t)],gh(e).forEach(i=>{var a=nr(i,"type.displayName")||nr(i,"type.name");n.indexOf(a)!==-1&&r.push(i)}),r}var bh=e=>e&&typeof e=="object"&&"clipDot"in e?!!e.clipDot:!0,hg=(e,t,r,n)=>{var i,a=(i=n&&(oa==null?void 0:oa[n]))!==null&&i!==void 0?i:[];return t.startsWith("data-")||typeof e!="function"&&(n&&a.includes(t)||fg.includes(t))||r&&Iu.includes(t)},W=(e,t,r)=>{if(!e||typeof e=="function"||typeof e=="boolean")return null;var n=e;if(y.isValidElement(e)&&(n=e.props),typeof n!="object"&&typeof n!="function")return null;var i={};return Object.keys(n).forEach(a=>{var o;hg((o=n)===null||o===void 0?void 0:o[a],a,t,r)&&(i[a]=n[a])}),i},vg=["children","width","height","viewBox","className","style","title","desc"];function No(){return No=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var{children:r,width:n,height:i,viewBox:a,className:o,style:u,title:c,desc:l}=e,s=pg(e,vg),f=a||{width:n,height:i,x:0,y:0},d=H("recharts-surface",o);return y.createElement("svg",No({},W(s,!0,"svg"),{className:d,width:n,height:i,style:u,viewBox:"".concat(f.x," ").concat(f.y," ").concat(f.width," ").concat(f.height),ref:t}),y.createElement("title",null,c),y.createElement("desc",null,l),r)}),yg=["children","className"];function $o(){return $o=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var{children:r,className:n}=e,i=gg(e,yg),a=H("recharts-layer",n);return y.createElement("g",$o({className:a},W(i,!0),{ref:t}),r)}),xh=y.createContext(null),xg=()=>y.useContext(xh);function J(e){return function(){return e}}const wh=Math.cos,Nn=Math.sin,st=Math.sqrt,$n=Math.PI,yi=2*$n,Ro=Math.PI,Lo=2*Ro,Ut=1e-6,wg=Lo-Ut;function Oh(e){this._+=e[0];for(let t=1,r=e.length;t=0))throw new Error(`invalid digits: ${e}`);if(t>15)return Oh;const r=10**t;return function(n){this._+=n[0];for(let i=1,a=n.length;iUt)if(!(Math.abs(f*c-l*s)>Ut)||!a)this._append`L${this._x1=t},${this._y1=r}`;else{let h=n-o,p=i-u,v=c*c+l*l,m=h*h+p*p,g=Math.sqrt(v),b=Math.sqrt(d),x=a*Math.tan((Ro-Math.acos((v+d-m)/(2*g*b)))/2),O=x/b,w=x/g;Math.abs(O-1)>Ut&&this._append`L${t+O*s},${r+O*f}`,this._append`A${a},${a},0,0,${+(f*h>s*p)},${this._x1=t+w*c},${this._y1=r+w*l}`}}arc(t,r,n,i,a,o){if(t=+t,r=+r,n=+n,o=!!o,n<0)throw new Error(`negative radius: ${n}`);let u=n*Math.cos(i),c=n*Math.sin(i),l=t+u,s=r+c,f=1^o,d=o?i-a:a-i;this._x1===null?this._append`M${l},${s}`:(Math.abs(this._x1-l)>Ut||Math.abs(this._y1-s)>Ut)&&this._append`L${l},${s}`,n&&(d<0&&(d=d%Lo+Lo),d>wg?this._append`A${n},${n},0,1,${f},${t-u},${r-c}A${n},${n},0,1,${f},${this._x1=l},${this._y1=s}`:d>Ut&&this._append`A${n},${n},0,${+(d>=Ro)},${f},${this._x1=t+n*Math.cos(a)},${this._y1=r+n*Math.sin(a)}`)}rect(t,r,n,i){this._append`M${this._x0=this._x1=+t},${this._y0=this._y1=+r}h${n=+n}v${+i}h${-n}Z`}toString(){return this._}}function Ru(e){let t=3;return e.digits=function(r){if(!arguments.length)return t;if(r==null)t=null;else{const n=Math.floor(r);if(!(n>=0))throw new RangeError(`invalid digits: ${r}`);t=n}return e},()=>new Pg(t)}function Lu(e){return typeof e=="object"&&"length"in e?e:Array.from(e)}function Ph(e){this._context=e}Ph.prototype={areaStart:function(){this._line=0},areaEnd:function(){this._line=NaN},lineStart:function(){this._point=0},lineEnd:function(){(this._line||this._line!==0&&this._point===1)&&this._context.closePath(),this._line=1-this._line},point:function(e,t){switch(e=+e,t=+t,this._point){case 0:this._point=1,this._line?this._context.lineTo(e,t):this._context.moveTo(e,t);break;case 1:this._point=2;default:this._context.lineTo(e,t);break}}};function gi(e){return new Ph(e)}function Ah(e){return e[0]}function Sh(e){return e[1]}function Eh(e,t){var r=J(!0),n=null,i=gi,a=null,o=Ru(u);e=typeof e=="function"?e:e===void 0?Ah:J(e),t=typeof t=="function"?t:t===void 0?Sh:J(t);function u(c){var l,s=(c=Lu(c)).length,f,d=!1,h;for(n==null&&(a=i(h=o())),l=0;l<=s;++l)!(l=h;--p)u.point(x[p],O[p]);u.lineEnd(),u.areaEnd()}g&&(x[d]=+e(m,d,f),O[d]=+t(m,d,f),u.point(n?+n(m,d,f):x[d],r?+r(m,d,f):O[d]))}if(b)return u=null,b+""||null}function s(){return Eh().defined(i).curve(o).context(a)}return l.x=function(f){return arguments.length?(e=typeof f=="function"?f:J(+f),n=null,l):e},l.x0=function(f){return arguments.length?(e=typeof f=="function"?f:J(+f),l):e},l.x1=function(f){return arguments.length?(n=f==null?null:typeof f=="function"?f:J(+f),l):n},l.y=function(f){return arguments.length?(t=typeof f=="function"?f:J(+f),r=null,l):t},l.y0=function(f){return arguments.length?(t=typeof f=="function"?f:J(+f),l):t},l.y1=function(f){return arguments.length?(r=f==null?null:typeof f=="function"?f:J(+f),l):r},l.lineX0=l.lineY0=function(){return s().x(e).y(t)},l.lineY1=function(){return s().x(e).y(r)},l.lineX1=function(){return s().x(n).y(t)},l.defined=function(f){return arguments.length?(i=typeof f=="function"?f:J(!!f),l):i},l.curve=function(f){return arguments.length?(o=f,a!=null&&(u=o(a)),l):o},l.context=function(f){return arguments.length?(f==null?a=u=null:u=o(a=f),l):a},l}class _h{constructor(t,r){this._context=t,this._x=r}areaStart(){this._line=0}areaEnd(){this._line=NaN}lineStart(){this._point=0}lineEnd(){(this._line||this._line!==0&&this._point===1)&&this._context.closePath(),this._line=1-this._line}point(t,r){switch(t=+t,r=+r,this._point){case 0:{this._point=1,this._line?this._context.lineTo(t,r):this._context.moveTo(t,r);break}case 1:this._point=2;default:{this._x?this._context.bezierCurveTo(this._x0=(this._x0+t)/2,this._y0,this._x0,r,t,r):this._context.bezierCurveTo(this._x0,this._y0=(this._y0+r)/2,t,this._y0,t,r);break}}this._x0=t,this._y0=r}}function Ag(e){return new _h(e,!0)}function Sg(e){return new _h(e,!1)}const Bu={draw(e,t){const r=st(t/$n);e.moveTo(r,0),e.arc(0,0,r,0,yi)}},Eg={draw(e,t){const r=st(t/5)/2;e.moveTo(-3*r,-r),e.lineTo(-r,-r),e.lineTo(-r,-3*r),e.lineTo(r,-3*r),e.lineTo(r,-r),e.lineTo(3*r,-r),e.lineTo(3*r,r),e.lineTo(r,r),e.lineTo(r,3*r),e.lineTo(-r,3*r),e.lineTo(-r,r),e.lineTo(-3*r,r),e.closePath()}},jh=st(1/3),_g=jh*2,jg={draw(e,t){const r=st(t/_g),n=r*jh;e.moveTo(0,-r),e.lineTo(n,0),e.lineTo(0,r),e.lineTo(-n,0),e.closePath()}},Tg={draw(e,t){const r=st(t),n=-r/2;e.rect(n,n,r,r)}},Cg=.8908130915292852,Th=Nn($n/10)/Nn(7*$n/10),kg=Nn(yi/10)*Th,Mg=-wh(yi/10)*Th,Ig={draw(e,t){const r=st(t*Cg),n=kg*r,i=Mg*r;e.moveTo(0,-r),e.lineTo(n,i);for(let a=1;a<5;++a){const o=yi*a/5,u=wh(o),c=Nn(o);e.lineTo(c*r,-u*r),e.lineTo(u*n-c*i,c*n+u*i)}e.closePath()}},sa=st(3),Dg={draw(e,t){const r=-st(t/(sa*3));e.moveTo(0,r*2),e.lineTo(-sa*r,-r),e.lineTo(sa*r,-r),e.closePath()}},Ye=-.5,Ge=st(3)/2,Bo=1/st(12),Ng=(Bo/2+1)*3,$g={draw(e,t){const r=st(t/Ng),n=r/2,i=r*Bo,a=n,o=r*Bo+r,u=-a,c=o;e.moveTo(n,i),e.lineTo(a,o),e.lineTo(u,c),e.lineTo(Ye*n-Ge*i,Ge*n+Ye*i),e.lineTo(Ye*a-Ge*o,Ge*a+Ye*o),e.lineTo(Ye*u-Ge*c,Ge*u+Ye*c),e.lineTo(Ye*n+Ge*i,Ye*i-Ge*n),e.lineTo(Ye*a+Ge*o,Ye*o-Ge*a),e.lineTo(Ye*u+Ge*c,Ye*c-Ge*u),e.closePath()}};function Rg(e,t){let r=null,n=Ru(i);e=typeof e=="function"?e:J(e||Bu),t=typeof t=="function"?t:J(t===void 0?64:+t);function i(){let a;if(r||(r=a=n()),e.apply(this,arguments).draw(r,+t.apply(this,arguments)),a)return r=null,a+""||null}return i.type=function(a){return arguments.length?(e=typeof a=="function"?a:J(a),i):e},i.size=function(a){return arguments.length?(t=typeof a=="function"?a:J(+a),i):t},i.context=function(a){return arguments.length?(r=a??null,i):r},i}function Rn(){}function Ln(e,t,r){e._context.bezierCurveTo((2*e._x0+e._x1)/3,(2*e._y0+e._y1)/3,(e._x0+2*e._x1)/3,(e._y0+2*e._y1)/3,(e._x0+4*e._x1+t)/6,(e._y0+4*e._y1+r)/6)}function Ch(e){this._context=e}Ch.prototype={areaStart:function(){this._line=0},areaEnd:function(){this._line=NaN},lineStart:function(){this._x0=this._x1=this._y0=this._y1=NaN,this._point=0},lineEnd:function(){switch(this._point){case 3:Ln(this,this._x1,this._y1);case 2:this._context.lineTo(this._x1,this._y1);break}(this._line||this._line!==0&&this._point===1)&&this._context.closePath(),this._line=1-this._line},point:function(e,t){switch(e=+e,t=+t,this._point){case 0:this._point=1,this._line?this._context.lineTo(e,t):this._context.moveTo(e,t);break;case 1:this._point=2;break;case 2:this._point=3,this._context.lineTo((5*this._x0+this._x1)/6,(5*this._y0+this._y1)/6);default:Ln(this,e,t);break}this._x0=this._x1,this._x1=e,this._y0=this._y1,this._y1=t}};function Lg(e){return new Ch(e)}function kh(e){this._context=e}kh.prototype={areaStart:Rn,areaEnd:Rn,lineStart:function(){this._x0=this._x1=this._x2=this._x3=this._x4=this._y0=this._y1=this._y2=this._y3=this._y4=NaN,this._point=0},lineEnd:function(){switch(this._point){case 1:{this._context.moveTo(this._x2,this._y2),this._context.closePath();break}case 2:{this._context.moveTo((this._x2+2*this._x3)/3,(this._y2+2*this._y3)/3),this._context.lineTo((this._x3+2*this._x2)/3,(this._y3+2*this._y2)/3),this._context.closePath();break}case 3:{this.point(this._x2,this._y2),this.point(this._x3,this._y3),this.point(this._x4,this._y4);break}}},point:function(e,t){switch(e=+e,t=+t,this._point){case 0:this._point=1,this._x2=e,this._y2=t;break;case 1:this._point=2,this._x3=e,this._y3=t;break;case 2:this._point=3,this._x4=e,this._y4=t,this._context.moveTo((this._x0+4*this._x1+e)/6,(this._y0+4*this._y1+t)/6);break;default:Ln(this,e,t);break}this._x0=this._x1,this._x1=e,this._y0=this._y1,this._y1=t}};function Bg(e){return new kh(e)}function Mh(e){this._context=e}Mh.prototype={areaStart:function(){this._line=0},areaEnd:function(){this._line=NaN},lineStart:function(){this._x0=this._x1=this._y0=this._y1=NaN,this._point=0},lineEnd:function(){(this._line||this._line!==0&&this._point===3)&&this._context.closePath(),this._line=1-this._line},point:function(e,t){switch(e=+e,t=+t,this._point){case 0:this._point=1;break;case 1:this._point=2;break;case 2:this._point=3;var r=(this._x0+4*this._x1+e)/6,n=(this._y0+4*this._y1+t)/6;this._line?this._context.lineTo(r,n):this._context.moveTo(r,n);break;case 3:this._point=4;default:Ln(this,e,t);break}this._x0=this._x1,this._x1=e,this._y0=this._y1,this._y1=t}};function qg(e){return new Mh(e)}function Ih(e){this._context=e}Ih.prototype={areaStart:Rn,areaEnd:Rn,lineStart:function(){this._point=0},lineEnd:function(){this._point&&this._context.closePath()},point:function(e,t){e=+e,t=+t,this._point?this._context.lineTo(e,t):(this._point=1,this._context.moveTo(e,t))}};function zg(e){return new Ih(e)}function Ec(e){return e<0?-1:1}function _c(e,t,r){var n=e._x1-e._x0,i=t-e._x1,a=(e._y1-e._y0)/(n||i<0&&-0),o=(r-e._y1)/(i||n<0&&-0),u=(a*i+o*n)/(n+i);return(Ec(a)+Ec(o))*Math.min(Math.abs(a),Math.abs(o),.5*Math.abs(u))||0}function jc(e,t){var r=e._x1-e._x0;return r?(3*(e._y1-e._y0)/r-t)/2:t}function ca(e,t,r){var n=e._x0,i=e._y0,a=e._x1,o=e._y1,u=(a-n)/3;e._context.bezierCurveTo(n+u,i+u*t,a-u,o-u*r,a,o)}function Bn(e){this._context=e}Bn.prototype={areaStart:function(){this._line=0},areaEnd:function(){this._line=NaN},lineStart:function(){this._x0=this._x1=this._y0=this._y1=this._t0=NaN,this._point=0},lineEnd:function(){switch(this._point){case 2:this._context.lineTo(this._x1,this._y1);break;case 3:ca(this,this._t0,jc(this,this._t0));break}(this._line||this._line!==0&&this._point===1)&&this._context.closePath(),this._line=1-this._line},point:function(e,t){var r=NaN;if(e=+e,t=+t,!(e===this._x1&&t===this._y1)){switch(this._point){case 0:this._point=1,this._line?this._context.lineTo(e,t):this._context.moveTo(e,t);break;case 1:this._point=2;break;case 2:this._point=3,ca(this,jc(this,r=_c(this,e,t)),r);break;default:ca(this,this._t0,r=_c(this,e,t));break}this._x0=this._x1,this._x1=e,this._y0=this._y1,this._y1=t,this._t0=r}}};function Dh(e){this._context=new Nh(e)}(Dh.prototype=Object.create(Bn.prototype)).point=function(e,t){Bn.prototype.point.call(this,t,e)};function Nh(e){this._context=e}Nh.prototype={moveTo:function(e,t){this._context.moveTo(t,e)},closePath:function(){this._context.closePath()},lineTo:function(e,t){this._context.lineTo(t,e)},bezierCurveTo:function(e,t,r,n,i,a){this._context.bezierCurveTo(t,e,n,r,a,i)}};function Kg(e){return new Bn(e)}function Wg(e){return new Dh(e)}function $h(e){this._context=e}$h.prototype={areaStart:function(){this._line=0},areaEnd:function(){this._line=NaN},lineStart:function(){this._x=[],this._y=[]},lineEnd:function(){var e=this._x,t=this._y,r=e.length;if(r)if(this._line?this._context.lineTo(e[0],t[0]):this._context.moveTo(e[0],t[0]),r===2)this._context.lineTo(e[1],t[1]);else for(var n=Tc(e),i=Tc(t),a=0,o=1;o=0;--t)i[t]=(o[t]-i[t+1])/a[t];for(a[r-1]=(e[r]+i[r-1])/2,t=0;t=0&&(this._t=1-this._t,this._line=1-this._line)},point:function(e,t){switch(e=+e,t=+t,this._point){case 0:this._point=1,this._line?this._context.lineTo(e,t):this._context.moveTo(e,t);break;case 1:this._point=2;default:{if(this._t<=0)this._context.lineTo(this._x,t),this._context.lineTo(e,t);else{var r=this._x*(1-this._t)+e*this._t;this._context.lineTo(r,this._y),this._context.lineTo(r,t)}break}}this._x=e,this._y=t}};function Ug(e){return new bi(e,.5)}function Hg(e){return new bi(e,0)}function Yg(e){return new bi(e,1)}function wr(e,t){if((o=e.length)>1)for(var r=1,n,i,a=e[t[0]],o,u=a.length;r=0;)r[t]=t;return r}function Gg(e,t){return e[t]}function Vg(e){const t=[];return t.key=e,t}function Xg(){var e=J([]),t=qo,r=wr,n=Gg;function i(a){var o=Array.from(e.apply(this,arguments),Vg),u,c=o.length,l=-1,s;for(const f of a)for(u=0,++l;u0){for(var r,n,i=0,a=e[0].length,o;i0){for(var r=0,n=e[t[0]],i,a=n.length;r0)||!((a=(i=e[t[0]]).length)>0))){for(var r=0,n=1,i,a,o;n{var t="symbol".concat(cn(e));return Rh[t]||Bu},s0=(e,t,r)=>{if(t==="area")return e;switch(r){case"cross":return 5*e*e/9;case"diamond":return .5*e*e/Math.sqrt(3);case"square":return e*e;case"star":{var n=18*o0;return 1.25*e*e*(Math.tan(n)-Math.tan(n*2)*Math.tan(n)**2)}case"triangle":return Math.sqrt(3)*e*e/4;case"wye":return(21-10*Math.sqrt(3))*e*e/8;default:return Math.PI*e*e/4}},c0=(e,t)=>{Rh["symbol".concat(cn(e))]=t},qu=e=>{var{type:t="circle",size:r=64,sizeType:n="area"}=e,i=i0(e,e0),a=kc(kc({},i),{},{type:t,size:r,sizeType:n}),o=()=>{var f=u0(t),d=Rg().type(f).size(s0(r,n,t));return d()},{className:u,cx:c,cy:l}=a,s=W(a,!0);return c===+c&&l===+l&&r===+r?y.createElement("path",zo({},s,{className:H("recharts-symbols",u),transform:"translate(".concat(c,", ").concat(l,")"),d:o()})):null};qu.registerSymbol=c0;function Ko(){return Ko=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var d=s.formatter||i,h=H({"recharts-legend-item":!0,["legend-item-".concat(f)]:!0,inactive:s.inactive});if(s.type==="none")return null;var p=s.inactive?a:s.color,v=d?d(s.value,s,f):s.value;return y.createElement("li",Ko({className:h,style:c,key:"legend-item-".concat(f)},mi(this.props,s,f)),y.createElement($u,{width:r,height:r,viewBox:u,style:l,"aria-label":"".concat(v," legend icon")},this.renderIcon(s,o)),y.createElement("span",{className:"recharts-legend-item-text",style:{color:p}},v))})}render(){var{payload:t,layout:r,align:n}=this.props;if(!t||!t.length)return null;var i={padding:0,margin:0,textAlign:r==="horizontal"?n:"left"};return y.createElement("ul",{className:"recharts-default-legend",style:i},this.renderItems())}}zu(Ku,"displayName","Legend");zu(Ku,"defaultProps",{align:"center",iconSize:14,inactiveColor:"#ccc",layout:"horizontal",verticalAlign:"middle"});var la={},fa={},Ic;function h0(){return Ic||(Ic=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r,n){const i=new Map;for(let a=0;a=0}e.isLength=t}(pa)),pa}var $c;function Wu(){return $c||($c=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=v0();function r(n){return n!=null&&typeof n!="function"&&t.isLength(n.length)}e.isArrayLike=r}(va)),va}var ma={},Rc;function p0(){return Rc||(Rc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return typeof r=="object"&&r!==null}e.isObjectLike=t}(ma)),ma}var Lc;function m0(){return Lc||(Lc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Wu(),r=p0();function n(i){return r.isObjectLike(i)&&t.isArrayLike(i)}e.isArrayLikeObject=n}(ha)),ha}var ya={},ga={},Bc;function y0(){return Bc||(Bc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Mu();function r(n){return function(i){return t.get(i,n)}}e.property=r}(ga)),ga}var ba={},xa={},wa={},Oa={},qc;function Bh(){return qc||(qc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return r!==null&&(typeof r=="object"||typeof r=="function")}e.isObject=t}(Oa)),Oa}var Pa={},zc;function qh(){return zc||(zc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return r==null||typeof r!="object"&&typeof r!="function"}e.isPrimitive=t}(Pa)),Pa}var Aa={},Kc;function Fu(){return Kc||(Kc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r,n){return r===n||Number.isNaN(r)&&Number.isNaN(n)}e.eq=t}(Aa)),Aa}var Wc;function g0(){return Wc||(Wc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Uu(),r=Bh(),n=qh(),i=Fu();function a(f,d,h){return typeof h!="function"?t.isMatch(f,d):o(f,d,function p(v,m,g,b,x,O){const w=h(v,m,g,b,x,O);return w!==void 0?!!w:o(v,m,p,O)},new Map)}function o(f,d,h,p){if(d===f)return!0;switch(typeof d){case"object":return u(f,d,h,p);case"function":return Object.keys(d).length>0?o(f,{...d},h,p):i.eq(f,d);default:return r.isObject(f)?typeof d=="string"?d==="":!0:i.eq(f,d)}}function u(f,d,h,p){if(d==null)return!0;if(Array.isArray(d))return l(f,d,h,p);if(d instanceof Map)return c(f,d,h,p);if(d instanceof Set)return s(f,d,h,p);const v=Object.keys(d);if(f==null)return v.length===0;if(v.length===0)return!0;if(p&&p.has(d))return p.get(d)===f;p&&p.set(d,f);try{for(let m=0;m{})}e.isMatch=r}(xa)),xa}var Sa={},Ea={},_a={},Uc;function zh(){return Uc||(Uc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return Object.getOwnPropertySymbols(r).filter(n=>Object.prototype.propertyIsEnumerable.call(r,n))}e.getSymbols=t}(_a)),_a}var ja={},Hc;function Hu(){return Hc||(Hc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return r==null?r===void 0?"[object Undefined]":"[object Null]":Object.prototype.toString.call(r)}e.getTag=t}(ja)),ja}var Ta={},Yc;function Yu(){return Yc||(Yc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t="[object RegExp]",r="[object String]",n="[object Number]",i="[object Boolean]",a="[object Arguments]",o="[object Symbol]",u="[object Date]",c="[object Map]",l="[object Set]",s="[object Array]",f="[object Function]",d="[object ArrayBuffer]",h="[object Object]",p="[object Error]",v="[object DataView]",m="[object Uint8Array]",g="[object Uint8ClampedArray]",b="[object Uint16Array]",x="[object Uint32Array]",O="[object BigUint64Array]",w="[object Int8Array]",P="[object Int16Array]",A="[object Int32Array]",E="[object BigInt64Array]",j="[object Float32Array]",N="[object Float64Array]";e.argumentsTag=a,e.arrayBufferTag=d,e.arrayTag=s,e.bigInt64ArrayTag=E,e.bigUint64ArrayTag=O,e.booleanTag=i,e.dataViewTag=v,e.dateTag=u,e.errorTag=p,e.float32ArrayTag=j,e.float64ArrayTag=N,e.functionTag=f,e.int16ArrayTag=P,e.int32ArrayTag=A,e.int8ArrayTag=w,e.mapTag=c,e.numberTag=n,e.objectTag=h,e.regexpTag=t,e.setTag=l,e.stringTag=r,e.symbolTag=o,e.uint16ArrayTag=b,e.uint32ArrayTag=x,e.uint8ArrayTag=m,e.uint8ClampedArrayTag=g}(Ta)),Ta}var Ca={},Gc;function b0(){return Gc||(Gc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return ArrayBuffer.isView(r)&&!(r instanceof DataView)}e.isTypedArray=t}(Ca)),Ca}var Vc;function Kh(){return Vc||(Vc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=zh(),r=Hu(),n=Yu(),i=qh(),a=b0();function o(s,f){return u(s,void 0,s,new Map,f)}function u(s,f,d,h=new Map,p=void 0){const v=p==null?void 0:p(s,f,d,h);if(v!=null)return v;if(i.isPrimitive(s))return s;if(h.has(s))return h.get(s);if(Array.isArray(s)){const m=new Array(s.length);h.set(s,m);for(let g=0;gt.isMatch(a,i)}e.matches=n}(ba)),ba}var ka={},Ma={},Ia={},Jc;function O0(){return Jc||(Jc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Kh(),r=Yu();function n(i,a){return t.cloneDeepWith(i,(o,u,c,l)=>{const s=a==null?void 0:a(o,u,c,l);if(s!=null)return s;if(typeof i=="object")switch(Object.prototype.toString.call(i)){case r.numberTag:case r.stringTag:case r.booleanTag:{const f=new i.constructor(i==null?void 0:i.valueOf());return t.copyProperties(f,i),f}case r.argumentsTag:{const f={};return t.copyProperties(f,i),f.length=i.length,f[Symbol.iterator]=i[Symbol.iterator],f}default:return}})}e.cloneDeepWith=n}(Ia)),Ia}var Qc;function P0(){return Qc||(Qc=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=O0();function r(n){return t.cloneDeepWith(n)}e.cloneDeep=r}(Ma)),Ma}var Da={},Na={},el;function Wh(){return el||(el=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=/^(?:0|[1-9]\d*)$/;function r(n,i=Number.MAX_SAFE_INTEGER){switch(typeof n){case"number":return Number.isInteger(n)&&n>=0&&n"u"||typeof window.document>"u"||typeof window.document.createElement>"u"?l:u;return za.useSyncExternalStore=e.useSyncExternalStore!==void 0?e.useSyncExternalStore:s,za}var cl;function M0(){return cl||(cl=1,qa.exports=k0()),qa.exports}/** + * @license React + * use-sync-external-store-shim/with-selector.production.js + * + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */var ll;function I0(){if(ll)return Ba;ll=1;var e=Cu(),t=M0();function r(l,s){return l===s&&(l!==0||1/l===1/s)||l!==l&&s!==s}var n=typeof Object.is=="function"?Object.is:r,i=t.useSyncExternalStore,a=e.useRef,o=e.useEffect,u=e.useMemo,c=e.useDebugValue;return Ba.useSyncExternalStoreWithSelector=function(l,s,f,d,h){var p=a(null);if(p.current===null){var v={hasValue:!1,value:null};p.current=v}else v=p.current;p=u(function(){function g(P){if(!b){if(b=!0,x=P,P=d(P),h!==void 0&&v.hasValue){var A=v.value;if(h(A,P))return O=A}return O=P}if(A=O,n(x,P))return A;var E=d(P);return h!==void 0&&h(A,E)?(x=P,A):(x=P,O=E)}var b=!1,x,O,w=f===void 0?null:f;return[function(){return g(s())},w===null?void 0:function(){return g(w())}]},[s,f,d,h]);var m=i(l,p[0],p[1]);return o(function(){v.hasValue=!0,v.value=m},[m]),c(m),m},Ba}var fl;function D0(){return fl||(fl=1,La.exports=I0()),La.exports}var N0=D0(),Gu=y.createContext(null),$0=e=>e,re=()=>{var e=y.useContext(Gu);return e?e.store.dispatch:$0},Dn=()=>{},R0=()=>Dn,L0=(e,t)=>e===t;function D(e){var t=y.useContext(Gu);return N0.useSyncExternalStoreWithSelector(t?t.subscription.addNestedSub:R0,t?t.store.getState:Dn,t?t.store.getState:Dn,t?e:Dn,L0)}function B0(e,t=`expected a function, instead received ${typeof e}`){if(typeof e!="function")throw new TypeError(t)}function q0(e,t=`expected an object, instead received ${typeof e}`){if(typeof e!="object")throw new TypeError(t)}function z0(e,t="expected all items to be functions, instead received the following types: "){if(!e.every(r=>typeof r=="function")){const r=e.map(n=>typeof n=="function"?`function ${n.name||"unnamed"}()`:typeof n).join(", ");throw new TypeError(`${t}[${r}]`)}}var dl=e=>Array.isArray(e)?e:[e];function K0(e){const t=Array.isArray(e[0])?e[0]:e;return z0(t,"createSelector expects all input-selectors to be functions, but received the following types: "),t}function W0(e,t){const r=[],{length:n}=e;for(let i=0;i{r=wn(),o.resetResultsCount()},o.resultsCount=()=>a,o.resetResultsCount=()=>{a=0},o}function Y0(e,...t){const r=typeof e=="function"?{memoize:e,memoizeOptions:t}:e,n=(...i)=>{let a=0,o=0,u,c={},l=i.pop();typeof l=="object"&&(c=l,l=i.pop()),B0(l,`createSelector expects an output function after the inputs, but received: [${typeof l}]`);const s={...r,...c},{memoize:f,memoizeOptions:d=[],argsMemoize:h=Uh,argsMemoizeOptions:p=[]}=s,v=dl(d),m=dl(p),g=K0(i),b=f(function(){return a++,l.apply(null,arguments)},...v),x=h(function(){o++;const w=W0(g,arguments);return u=b.apply(null,w),u},...m);return Object.assign(x,{resultFunc:l,memoizedResultFunc:b,dependencies:g,dependencyRecomputations:()=>o,resetDependencyRecomputations:()=>{o=0},lastResult:()=>u,recomputations:()=>a,resetRecomputations:()=>{a=0},memoize:f,argsMemoize:h})};return Object.assign(n,{withTypes:()=>n}),n}var S=Y0(Uh),G0=Object.assign((e,t=S)=>{q0(e,`createStructuredSelector expects first argument to be an object where each property is a selector, instead received a ${typeof e}`);const r=Object.keys(e),n=r.map(a=>e[a]);return t(n,(...a)=>a.reduce((o,u,c)=>(o[r[c]]=u,o),{}))},{withTypes:()=>G0}),Ka={},Wa={},Fa={},vl;function V0(){return vl||(vl=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(n){return typeof n=="symbol"?1:n===null?2:n===void 0?3:n!==n?4:0}const r=(n,i,a)=>{if(n!==i){const o=t(n),u=t(i);if(o===u&&o===0){if(ni)return a==="desc"?-1:1}return a==="desc"?u-o:o-u}return 0};e.compareValues=r}(Fa)),Fa}var Ua={},Ha={},pl;function Hh(){return pl||(pl=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return typeof r=="symbol"||r instanceof Symbol}e.isSymbol=t}(Ha)),Ha}var ml;function X0(){return ml||(ml=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Hh(),r=/\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/,n=/^\w*$/;function i(a,o){return Array.isArray(a)?!1:typeof a=="number"||typeof a=="boolean"||a==null||t.isSymbol(a)?!0:typeof a=="string"&&(n.test(a)||!r.test(a))||o!=null&&Object.hasOwn(o,a)}e.isKey=i}(Ua)),Ua}var yl;function Z0(){return yl||(yl=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=V0(),r=X0(),n=ku();function i(a,o,u,c){if(a==null)return[];u=c?void 0:u,Array.isArray(a)||(a=Object.values(a)),Array.isArray(o)||(o=o==null?[null]:[o]),o.length===0&&(o=[null]),Array.isArray(u)||(u=u==null?[]:[u]),u=u.map(h=>String(h));const l=(h,p)=>{let v=h;for(let m=0;mp==null||h==null?p:typeof h=="object"&&"key"in h?Object.hasOwn(p,h.key)?p[h.key]:l(p,h.path):typeof h=="function"?h(p):Array.isArray(h)?l(p,h):typeof p=="object"?p[h]:p,f=o.map(h=>(Array.isArray(h)&&h.length===1&&(h=h[0]),h==null||typeof h=="function"||Array.isArray(h)||r.isKey(h)?h:{key:h,path:n.toPath(h)}));return a.map(h=>({original:h,criteria:f.map(p=>s(p,h))})).slice().sort((h,p)=>{for(let v=0;vh.original)}e.orderBy=i}(Wa)),Wa}var Ya={},gl;function J0(){return gl||(gl=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r,n=1){const i=[],a=Math.floor(n),o=(u,c)=>{for(let l=0;l1&&n.isIterateeCall(a,o[0],o[1])?o=[]:u>2&&n.isIterateeCall(o[0],o[1],o[2])&&(o=[o[0]]),t.orderBy(a,r.flatten(o),["asc"])}e.sortBy=i}(Ka)),Ka}var Va,wl;function eb(){return wl||(wl=1,Va=Q0().sortBy),Va}var tb=eb();const xi=At(tb);var Gh=e=>e.legend.settings,rb=e=>e.legend.size,nb=e=>e.legend.payload,ib=S([nb,Gh],(e,t)=>{var{itemSorter:r}=t,n=e.flat(1);return r?xi(n,r):n});function ab(){return D(ib)}var On=1;function Vh(){var e=arguments.length>0&&arguments[0]!==void 0?arguments[0]:[],[t,r]=y.useState({height:0,left:0,top:0,width:0}),n=y.useCallback(i=>{if(i!=null){var a=i.getBoundingClientRect(),o={height:a.height,left:a.left,top:a.top,width:a.width};(Math.abs(o.height-t.height)>On||Math.abs(o.left-t.left)>On||Math.abs(o.top-t.top)>On||Math.abs(o.width-t.width)>On)&&r({height:o.height,left:o.left,top:o.top,width:o.width})}},[t.width,t.height,t.top,t.left,...e]);return[t,n]}function pe(e){return`Minified Redux error #${e}; visit https://redux.js.org/Errors?code=${e} for the full message or use the non-minified dev environment for full errors. `}var ob=typeof Symbol=="function"&&Symbol.observable||"@@observable",Ol=ob,Xa=()=>Math.random().toString(36).substring(7).split("").join("."),ub={INIT:`@@redux/INIT${Xa()}`,REPLACE:`@@redux/REPLACE${Xa()}`,PROBE_UNKNOWN_ACTION:()=>`@@redux/PROBE_UNKNOWN_ACTION${Xa()}`},qn=ub;function Vu(e){if(typeof e!="object"||e===null)return!1;let t=e;for(;Object.getPrototypeOf(t)!==null;)t=Object.getPrototypeOf(t);return Object.getPrototypeOf(e)===t||Object.getPrototypeOf(e)===null}function Xh(e,t,r){if(typeof e!="function")throw new Error(pe(2));if(typeof t=="function"&&typeof r=="function"||typeof r=="function"&&typeof arguments[3]=="function")throw new Error(pe(0));if(typeof t=="function"&&typeof r>"u"&&(r=t,t=void 0),typeof r<"u"){if(typeof r!="function")throw new Error(pe(1));return r(Xh)(e,t)}let n=e,i=t,a=new Map,o=a,u=0,c=!1;function l(){o===a&&(o=new Map,a.forEach((m,g)=>{o.set(g,m)}))}function s(){if(c)throw new Error(pe(3));return i}function f(m){if(typeof m!="function")throw new Error(pe(4));if(c)throw new Error(pe(5));let g=!0;l();const b=u++;return o.set(b,m),function(){if(g){if(c)throw new Error(pe(6));g=!1,l(),o.delete(b),a=null}}}function d(m){if(!Vu(m))throw new Error(pe(7));if(typeof m.type>"u")throw new Error(pe(8));if(typeof m.type!="string")throw new Error(pe(17));if(c)throw new Error(pe(9));try{c=!0,i=n(i,m)}finally{c=!1}return(a=o).forEach(b=>{b()}),m}function h(m){if(typeof m!="function")throw new Error(pe(10));n=m,d({type:qn.REPLACE})}function p(){const m=f;return{subscribe(g){if(typeof g!="object"||g===null)throw new Error(pe(11));function b(){const O=g;O.next&&O.next(s())}return b(),{unsubscribe:m(b)}},[Ol](){return this}}}return d({type:qn.INIT}),{dispatch:d,subscribe:f,getState:s,replaceReducer:h,[Ol]:p}}function sb(e){Object.keys(e).forEach(t=>{const r=e[t];if(typeof r(void 0,{type:qn.INIT})>"u")throw new Error(pe(12));if(typeof r(void 0,{type:qn.PROBE_UNKNOWN_ACTION()})>"u")throw new Error(pe(13))})}function Zh(e){const t=Object.keys(e),r={};for(let a=0;a"u")throw u&&u.type,new Error(pe(14));l[f]=p,c=c||p!==h}return c=c||n.length!==Object.keys(o).length,c?l:o}}function zn(...e){return e.length===0?t=>t:e.length===1?e[0]:e.reduce((t,r)=>(...n)=>t(r(...n)))}function cb(...e){return t=>(r,n)=>{const i=t(r,n);let a=()=>{throw new Error(pe(15))};const o={getState:i.getState,dispatch:(c,...l)=>a(c,...l)},u=e.map(c=>c(o));return a=zn(...u)(i.dispatch),{...i,dispatch:a}}}function Jh(e){return Vu(e)&&"type"in e&&typeof e.type=="string"}var Qh=Symbol.for("immer-nothing"),Pl=Symbol.for("immer-draftable"),Ke=Symbol.for("immer-state");function at(e,...t){throw new Error(`[Immer] minified error nr: ${e}. Full error at: https://bit.ly/3cXEKWf`)}var Or=Object.getPrototypeOf;function ir(e){return!!e&&!!e[Ke]}function xt(e){var t;return e?ev(e)||Array.isArray(e)||!!e[Pl]||!!((t=e.constructor)!=null&&t[Pl])||Oi(e)||Pi(e):!1}var lb=Object.prototype.constructor.toString();function ev(e){if(!e||typeof e!="object")return!1;const t=Or(e);if(t===null)return!0;const r=Object.hasOwnProperty.call(t,"constructor")&&t.constructor;return r===Object?!0:typeof r=="function"&&Function.toString.call(r)===lb}function Kn(e,t){wi(e)===0?Reflect.ownKeys(e).forEach(r=>{t(r,e[r],e)}):e.forEach((r,n)=>t(n,r,e))}function wi(e){const t=e[Ke];return t?t.type_:Array.isArray(e)?1:Oi(e)?2:Pi(e)?3:0}function Wo(e,t){return wi(e)===2?e.has(t):Object.prototype.hasOwnProperty.call(e,t)}function tv(e,t,r){const n=wi(e);n===2?e.set(t,r):n===3?e.add(r):e[t]=r}function fb(e,t){return e===t?e!==0||1/e===1/t:e!==e&&t!==t}function Oi(e){return e instanceof Map}function Pi(e){return e instanceof Set}function Ht(e){return e.copy_||e.base_}function Fo(e,t){if(Oi(e))return new Map(e);if(Pi(e))return new Set(e);if(Array.isArray(e))return Array.prototype.slice.call(e);const r=ev(e);if(t===!0||t==="class_only"&&!r){const n=Object.getOwnPropertyDescriptors(e);delete n[Ke];let i=Reflect.ownKeys(n);for(let a=0;a1&&(e.set=e.add=e.clear=e.delete=db),Object.freeze(e),t&&Object.entries(e).forEach(([r,n])=>Xu(n,!0))),e}function db(){at(2)}function Ai(e){return Object.isFrozen(e)}var hb={};function ar(e){const t=hb[e];return t||at(0,e),t}var Xr;function rv(){return Xr}function vb(e,t){return{drafts_:[],parent_:e,immer_:t,canAutoFreeze_:!0,unfinalizedDrafts_:0}}function Al(e,t){t&&(ar("Patches"),e.patches_=[],e.inversePatches_=[],e.patchListener_=t)}function Uo(e){Ho(e),e.drafts_.forEach(pb),e.drafts_=null}function Ho(e){e===Xr&&(Xr=e.parent_)}function Sl(e){return Xr=vb(Xr,e)}function pb(e){const t=e[Ke];t.type_===0||t.type_===1?t.revoke_():t.revoked_=!0}function El(e,t){t.unfinalizedDrafts_=t.drafts_.length;const r=t.drafts_[0];return e!==void 0&&e!==r?(r[Ke].modified_&&(Uo(t),at(4)),xt(e)&&(e=Wn(t,e),t.parent_||Fn(t,e)),t.patches_&&ar("Patches").generateReplacementPatches_(r[Ke].base_,e,t.patches_,t.inversePatches_)):e=Wn(t,r,[]),Uo(t),t.patches_&&t.patchListener_(t.patches_,t.inversePatches_),e!==Qh?e:void 0}function Wn(e,t,r){if(Ai(t))return t;const n=t[Ke];if(!n)return Kn(t,(i,a)=>_l(e,n,t,i,a,r)),t;if(n.scope_!==e)return t;if(!n.modified_)return Fn(e,n.base_,!0),n.base_;if(!n.finalized_){n.finalized_=!0,n.scope_.unfinalizedDrafts_--;const i=n.copy_;let a=i,o=!1;n.type_===3&&(a=new Set(i),i.clear(),o=!0),Kn(a,(u,c)=>_l(e,n,i,u,c,r,o)),Fn(e,i,!1),r&&e.patches_&&ar("Patches").generatePatches_(n,r,e.patches_,e.inversePatches_)}return n.copy_}function _l(e,t,r,n,i,a,o){if(ir(i)){const u=a&&t&&t.type_!==3&&!Wo(t.assigned_,n)?a.concat(n):void 0,c=Wn(e,i,u);if(tv(r,n,c),ir(c))e.canAutoFreeze_=!1;else return}else o&&r.add(i);if(xt(i)&&!Ai(i)){if(!e.immer_.autoFreeze_&&e.unfinalizedDrafts_<1)return;Wn(e,i),(!t||!t.scope_.parent_)&&typeof n!="symbol"&&Object.prototype.propertyIsEnumerable.call(r,n)&&Fn(e,i)}}function Fn(e,t,r=!1){!e.parent_&&e.immer_.autoFreeze_&&e.canAutoFreeze_&&Xu(t,r)}function mb(e,t){const r=Array.isArray(e),n={type_:r?1:0,scope_:t?t.scope_:rv(),modified_:!1,finalized_:!1,assigned_:{},parent_:t,base_:e,draft_:null,copy_:null,revoke_:null,isManual_:!1};let i=n,a=Zu;r&&(i=[n],a=Zr);const{revoke:o,proxy:u}=Proxy.revocable(i,a);return n.draft_=u,n.revoke_=o,u}var Zu={get(e,t){if(t===Ke)return e;const r=Ht(e);if(!Wo(r,t))return yb(e,r,t);const n=r[t];return e.finalized_||!xt(n)?n:n===Za(e.base_,t)?(Ja(e),e.copy_[t]=Go(n,e)):n},has(e,t){return t in Ht(e)},ownKeys(e){return Reflect.ownKeys(Ht(e))},set(e,t,r){const n=nv(Ht(e),t);if(n!=null&&n.set)return n.set.call(e.draft_,r),!0;if(!e.modified_){const i=Za(Ht(e),t),a=i==null?void 0:i[Ke];if(a&&a.base_===r)return e.copy_[t]=r,e.assigned_[t]=!1,!0;if(fb(r,i)&&(r!==void 0||Wo(e.base_,t)))return!0;Ja(e),Yo(e)}return e.copy_[t]===r&&(r!==void 0||t in e.copy_)||Number.isNaN(r)&&Number.isNaN(e.copy_[t])||(e.copy_[t]=r,e.assigned_[t]=!0),!0},deleteProperty(e,t){return Za(e.base_,t)!==void 0||t in e.base_?(e.assigned_[t]=!1,Ja(e),Yo(e)):delete e.assigned_[t],e.copy_&&delete e.copy_[t],!0},getOwnPropertyDescriptor(e,t){const r=Ht(e),n=Reflect.getOwnPropertyDescriptor(r,t);return n&&{writable:!0,configurable:e.type_!==1||t!=="length",enumerable:n.enumerable,value:r[t]}},defineProperty(){at(11)},getPrototypeOf(e){return Or(e.base_)},setPrototypeOf(){at(12)}},Zr={};Kn(Zu,(e,t)=>{Zr[e]=function(){return arguments[0]=arguments[0][0],t.apply(this,arguments)}});Zr.deleteProperty=function(e,t){return Zr.set.call(this,e,t,void 0)};Zr.set=function(e,t,r){return Zu.set.call(this,e[0],t,r,e[0])};function Za(e,t){const r=e[Ke];return(r?Ht(r):e)[t]}function yb(e,t,r){var i;const n=nv(t,r);return n?"value"in n?n.value:(i=n.get)==null?void 0:i.call(e.draft_):void 0}function nv(e,t){if(!(t in e))return;let r=Or(e);for(;r;){const n=Object.getOwnPropertyDescriptor(r,t);if(n)return n;r=Or(r)}}function Yo(e){e.modified_||(e.modified_=!0,e.parent_&&Yo(e.parent_))}function Ja(e){e.copy_||(e.copy_=Fo(e.base_,e.scope_.immer_.useStrictShallowCopy_))}var gb=class{constructor(e){this.autoFreeze_=!0,this.useStrictShallowCopy_=!1,this.produce=(t,r,n)=>{if(typeof t=="function"&&typeof r!="function"){const a=r;r=t;const o=this;return function(c=a,...l){return o.produce(c,s=>r.call(this,s,...l))}}typeof r!="function"&&at(6),n!==void 0&&typeof n!="function"&&at(7);let i;if(xt(t)){const a=Sl(this),o=Go(t,void 0);let u=!0;try{i=r(o),u=!1}finally{u?Uo(a):Ho(a)}return Al(a,n),El(i,a)}else if(!t||typeof t!="object"){if(i=r(t),i===void 0&&(i=t),i===Qh&&(i=void 0),this.autoFreeze_&&Xu(i,!0),n){const a=[],o=[];ar("Patches").generateReplacementPatches_(t,i,a,o),n(a,o)}return i}else at(1,t)},this.produceWithPatches=(t,r)=>{if(typeof t=="function")return(o,...u)=>this.produceWithPatches(o,c=>t(c,...u));let n,i;return[this.produce(t,r,(o,u)=>{n=o,i=u}),n,i]},typeof(e==null?void 0:e.autoFreeze)=="boolean"&&this.setAutoFreeze(e.autoFreeze),typeof(e==null?void 0:e.useStrictShallowCopy)=="boolean"&&this.setUseStrictShallowCopy(e.useStrictShallowCopy)}createDraft(e){xt(e)||at(8),ir(e)&&(e=gt(e));const t=Sl(this),r=Go(e,void 0);return r[Ke].isManual_=!0,Ho(t),r}finishDraft(e,t){const r=e&&e[Ke];(!r||!r.isManual_)&&at(9);const{scope_:n}=r;return Al(n,t),El(void 0,n)}setAutoFreeze(e){this.autoFreeze_=e}setUseStrictShallowCopy(e){this.useStrictShallowCopy_=e}applyPatches(e,t){let r;for(r=t.length-1;r>=0;r--){const i=t[r];if(i.path.length===0&&i.op==="replace"){e=i.value;break}}r>-1&&(t=t.slice(r+1));const n=ar("Patches").applyPatches_;return ir(e)?n(e,t):this.produce(e,i=>n(i,t))}};function Go(e,t){const r=Oi(e)?ar("MapSet").proxyMap_(e,t):Pi(e)?ar("MapSet").proxySet_(e,t):mb(e,t);return(t?t.scope_:rv()).drafts_.push(r),r}function gt(e){return ir(e)||at(10,e),iv(e)}function iv(e){if(!xt(e)||Ai(e))return e;const t=e[Ke];let r;if(t){if(!t.modified_)return t.base_;t.finalized_=!0,r=Fo(e,t.scope_.immer_.useStrictShallowCopy_)}else r=Fo(e,!0);return Kn(r,(n,i)=>{tv(r,n,iv(i))}),t&&(t.finalized_=!1),r}var We=new gb,av=We.produce;We.produceWithPatches.bind(We);We.setAutoFreeze.bind(We);We.setUseStrictShallowCopy.bind(We);We.applyPatches.bind(We);We.createDraft.bind(We);We.finishDraft.bind(We);function ov(e){return({dispatch:r,getState:n})=>i=>a=>typeof a=="function"?a(r,n,e):i(a)}var bb=ov(),xb=ov,wb=typeof window<"u"&&window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__?window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__:function(){if(arguments.length!==0)return typeof arguments[0]=="object"?zn:zn.apply(null,arguments)};function Qe(e,t){function r(...n){if(t){let i=t(...n);if(!i)throw new Error(Be(0));return{type:e,payload:i.payload,..."meta"in i&&{meta:i.meta},..."error"in i&&{error:i.error}}}return{type:e,payload:n[0]}}return r.toString=()=>`${e}`,r.type=e,r.match=n=>Jh(n)&&n.type===e,r}var uv=class Ur extends Array{constructor(...t){super(...t),Object.setPrototypeOf(this,Ur.prototype)}static get[Symbol.species](){return Ur}concat(...t){return super.concat.apply(this,t)}prepend(...t){return t.length===1&&Array.isArray(t[0])?new Ur(...t[0].concat(this)):new Ur(...t.concat(this))}};function jl(e){return xt(e)?av(e,()=>{}):e}function Pn(e,t,r){return e.has(t)?e.get(t):e.set(t,r(t)).get(t)}function Ob(e){return typeof e=="boolean"}var Pb=()=>function(t){const{thunk:r=!0,immutableCheck:n=!0,serializableCheck:i=!0,actionCreatorCheck:a=!0}=t??{};let o=new uv;return r&&(Ob(r)?o.push(bb):o.push(xb(r.extraArgument))),o},Ab="RTK_autoBatch",Tl=e=>t=>{setTimeout(t,e)},Sb=(e={type:"raf"})=>t=>(...r)=>{const n=t(...r);let i=!0,a=!1,o=!1;const u=new Set,c=e.type==="tick"?queueMicrotask:e.type==="raf"?typeof window<"u"&&window.requestAnimationFrame?window.requestAnimationFrame:Tl(10):e.type==="callback"?e.queueNotification:Tl(e.timeout),l=()=>{o=!1,a&&(a=!1,u.forEach(s=>s()))};return Object.assign({},n,{subscribe(s){const f=()=>i&&s(),d=n.subscribe(f);return u.add(s),()=>{d(),u.delete(s)}},dispatch(s){var f;try{return i=!((f=s==null?void 0:s.meta)!=null&&f[Ab]),a=!i,a&&(o||(o=!0,c(l))),n.dispatch(s)}finally{i=!0}}})},Eb=e=>function(r){const{autoBatch:n=!0}=r??{};let i=new uv(e);return n&&i.push(Sb(typeof n=="object"?n:void 0)),i};function _b(e){const t=Pb(),{reducer:r=void 0,middleware:n,devTools:i=!0,preloadedState:a=void 0,enhancers:o=void 0}=e||{};let u;if(typeof r=="function")u=r;else if(Vu(r))u=Zh(r);else throw new Error(Be(1));let c;typeof n=="function"?c=n(t):c=t();let l=zn;i&&(l=wb({trace:!1,...typeof i=="object"&&i}));const s=cb(...c),f=Eb(s);let d=typeof o=="function"?o(f):f();const h=l(...d);return Xh(u,a,h)}function sv(e){const t={},r=[];let n;const i={addCase(a,o){const u=typeof a=="string"?a:a.type;if(!u)throw new Error(Be(28));if(u in t)throw new Error(Be(29));return t[u]=o,i},addMatcher(a,o){return r.push({matcher:a,reducer:o}),i},addDefaultCase(a){return n=a,i}};return e(i),[t,r,n]}function jb(e){return typeof e=="function"}function Tb(e,t){let[r,n,i]=sv(t),a;if(jb(e))a=()=>jl(e());else{const u=jl(e);a=()=>u}function o(u=a(),c){let l=[r[c.type],...n.filter(({matcher:s})=>s(c)).map(({reducer:s})=>s)];return l.filter(s=>!!s).length===0&&(l=[i]),l.reduce((s,f)=>{if(f)if(ir(s)){const h=f(s,c);return h===void 0?s:h}else{if(xt(s))return av(s,d=>f(d,c));{const d=f(s,c);if(d===void 0){if(s===null)return s;throw Error("A case reducer on a non-draftable value must not return undefined")}return d}}return s},u)}return o.getInitialState=a,o}var Cb="ModuleSymbhasOwnPr-0123456789ABCDEFGHNRVfgctiUvz_KqYTJkLxpZXIjQW",kb=(e=21)=>{let t="",r=e;for(;r--;)t+=Cb[Math.random()*64|0];return t},Mb=Symbol.for("rtk-slice-createasyncthunk");function Ib(e,t){return`${e}/${t}`}function Db({creators:e}={}){var r;const t=(r=e==null?void 0:e.asyncThunk)==null?void 0:r[Mb];return function(i){const{name:a,reducerPath:o=a}=i;if(!a)throw new Error(Be(11));const u=(typeof i.reducers=="function"?i.reducers($b()):i.reducers)||{},c=Object.keys(u),l={sliceCaseReducersByName:{},sliceCaseReducersByType:{},actionCreators:{},sliceMatchers:[]},s={addCase(O,w){const P=typeof O=="string"?O:O.type;if(!P)throw new Error(Be(12));if(P in l.sliceCaseReducersByType)throw new Error(Be(13));return l.sliceCaseReducersByType[P]=w,s},addMatcher(O,w){return l.sliceMatchers.push({matcher:O,reducer:w}),s},exposeAction(O,w){return l.actionCreators[O]=w,s},exposeCaseReducer(O,w){return l.sliceCaseReducersByName[O]=w,s}};c.forEach(O=>{const w=u[O],P={reducerName:O,type:Ib(a,O),createNotation:typeof i.reducers=="function"};Lb(w)?qb(P,w,s,t):Rb(P,w,s)});function f(){const[O={},w=[],P=void 0]=typeof i.extraReducers=="function"?sv(i.extraReducers):[i.extraReducers],A={...O,...l.sliceCaseReducersByType};return Tb(i.initialState,E=>{for(let j in A)E.addCase(j,A[j]);for(let j of l.sliceMatchers)E.addMatcher(j.matcher,j.reducer);for(let j of w)E.addMatcher(j.matcher,j.reducer);P&&E.addDefaultCase(P)})}const d=O=>O,h=new Map,p=new WeakMap;let v;function m(O,w){return v||(v=f()),v(O,w)}function g(){return v||(v=f()),v.getInitialState()}function b(O,w=!1){function P(E){let j=E[O];return typeof j>"u"&&w&&(j=Pn(p,P,g)),j}function A(E=d){const j=Pn(h,w,()=>new WeakMap);return Pn(j,E,()=>{const N={};for(const[T,C]of Object.entries(i.selectors??{}))N[T]=Nb(C,E,()=>Pn(p,E,g),w);return N})}return{reducerPath:O,getSelectors:A,get selectors(){return A(P)},selectSlice:P}}const x={name:a,reducer:m,actions:l.actionCreators,caseReducers:l.sliceCaseReducersByName,getInitialState:g,...b(o),injectInto(O,{reducerPath:w,...P}={}){const A=w??o;return O.inject({reducerPath:A,reducer:m},P),{...x,...b(A,!0)}}};return x}}function Nb(e,t,r,n){function i(a,...o){let u=t(a);return typeof u>"u"&&n&&(u=r()),e(u,...o)}return i.unwrapped=e,i}var tt=Db();function $b(){function e(t,r){return{_reducerDefinitionType:"asyncThunk",payloadCreator:t,...r}}return e.withTypes=()=>e,{reducer(t){return Object.assign({[t.name](...r){return t(...r)}}[t.name],{_reducerDefinitionType:"reducer"})},preparedReducer(t,r){return{_reducerDefinitionType:"reducerWithPrepare",prepare:t,reducer:r}},asyncThunk:e}}function Rb({type:e,reducerName:t,createNotation:r},n,i){let a,o;if("reducer"in n){if(r&&!Bb(n))throw new Error(Be(17));a=n.reducer,o=n.prepare}else a=n;i.addCase(e,a).exposeCaseReducer(t,a).exposeAction(t,o?Qe(e,o):Qe(e))}function Lb(e){return e._reducerDefinitionType==="asyncThunk"}function Bb(e){return e._reducerDefinitionType==="reducerWithPrepare"}function qb({type:e,reducerName:t},r,n,i){if(!i)throw new Error(Be(18));const{payloadCreator:a,fulfilled:o,pending:u,rejected:c,settled:l,options:s}=r,f=i(e,a,s);n.exposeAction(t,f),o&&n.addCase(f.fulfilled,o),u&&n.addCase(f.pending,u),c&&n.addCase(f.rejected,c),l&&n.addMatcher(f.settled,l),n.exposeCaseReducer(t,{fulfilled:o||An,pending:u||An,rejected:c||An,settled:l||An})}function An(){}var zb="task",cv="listener",lv="completed",Ju="cancelled",Kb=`task-${Ju}`,Wb=`task-${lv}`,Vo=`${cv}-${Ju}`,Fb=`${cv}-${lv}`,Si=class{constructor(e){Ji(this,"name","TaskAbortError");Ji(this,"message");this.code=e,this.message=`${zb} ${Ju} (reason: ${e})`}},Qu=(e,t)=>{if(typeof e!="function")throw new TypeError(Be(32))},Un=()=>{},fv=(e,t=Un)=>(e.catch(t),e),dv=(e,t)=>(e.addEventListener("abort",t,{once:!0}),()=>e.removeEventListener("abort",t)),Qt=(e,t)=>{const r=e.signal;r.aborted||("reason"in r||Object.defineProperty(r,"reason",{enumerable:!0,value:t,configurable:!0,writable:!0}),e.abort(t))},er=e=>{if(e.aborted){const{reason:t}=e;throw new Si(t)}};function hv(e,t){let r=Un;return new Promise((n,i)=>{const a=()=>i(new Si(e.reason));if(e.aborted){a();return}r=dv(e,a),t.finally(()=>r()).then(n,i)}).finally(()=>{r=Un})}var Ub=async(e,t)=>{try{return await Promise.resolve(),{status:"ok",value:await e()}}catch(r){return{status:r instanceof Si?"cancelled":"rejected",error:r}}finally{t==null||t()}},Hn=e=>t=>fv(hv(e,t).then(r=>(er(e),r))),vv=e=>{const t=Hn(e);return r=>t(new Promise(n=>setTimeout(n,r)))},{assign:gr}=Object,Cl={},Ei="listenerMiddleware",Hb=(e,t)=>{const r=n=>dv(e,()=>Qt(n,e.reason));return(n,i)=>{Qu(n);const a=new AbortController;r(a);const o=Ub(async()=>{er(e),er(a.signal);const u=await n({pause:Hn(a.signal),delay:vv(a.signal),signal:a.signal});return er(a.signal),u},()=>Qt(a,Wb));return i!=null&&i.autoJoin&&t.push(o.catch(Un)),{result:Hn(e)(o),cancel(){Qt(a,Kb)}}}},Yb=(e,t)=>{const r=async(n,i)=>{er(t);let a=()=>{};const u=[new Promise((c,l)=>{let s=e({predicate:n,effect:(f,d)=>{d.unsubscribe(),c([f,d.getState(),d.getOriginalState()])}});a=()=>{s(),l()}})];i!=null&&u.push(new Promise(c=>setTimeout(c,i,null)));try{const c=await hv(t,Promise.race(u));return er(t),c}finally{a()}};return(n,i)=>fv(r(n,i))},pv=e=>{let{type:t,actionCreator:r,matcher:n,predicate:i,effect:a}=e;if(t)i=Qe(t).match;else if(r)t=r.type,i=r.match;else if(n)i=n;else if(!i)throw new Error(Be(21));return Qu(a),{predicate:i,type:t,effect:a}},mv=gr(e=>{const{type:t,predicate:r,effect:n}=pv(e);return{id:kb(),effect:n,type:t,predicate:r,pending:new Set,unsubscribe:()=>{throw new Error(Be(22))}}},{withTypes:()=>mv}),kl=(e,t)=>{const{type:r,effect:n,predicate:i}=pv(t);return Array.from(e.values()).find(a=>(typeof r=="string"?a.type===r:a.predicate===i)&&a.effect===n)},Xo=e=>{e.pending.forEach(t=>{Qt(t,Vo)})},Gb=e=>()=>{e.forEach(Xo),e.clear()},Ml=(e,t,r)=>{try{e(t,r)}catch(n){setTimeout(()=>{throw n},0)}},yv=gr(Qe(`${Ei}/add`),{withTypes:()=>yv}),Vb=Qe(`${Ei}/removeAll`),gv=gr(Qe(`${Ei}/remove`),{withTypes:()=>gv}),Xb=(...e)=>{console.error(`${Ei}/error`,...e)},ln=(e={})=>{const t=new Map,{extra:r,onError:n=Xb}=e;Qu(n);const i=s=>(s.unsubscribe=()=>t.delete(s.id),t.set(s.id,s),f=>{s.unsubscribe(),f!=null&&f.cancelActive&&Xo(s)}),a=s=>{const f=kl(t,s)??mv(s);return i(f)};gr(a,{withTypes:()=>a});const o=s=>{const f=kl(t,s);return f&&(f.unsubscribe(),s.cancelActive&&Xo(f)),!!f};gr(o,{withTypes:()=>o});const u=async(s,f,d,h)=>{const p=new AbortController,v=Yb(a,p.signal),m=[];try{s.pending.add(p),await Promise.resolve(s.effect(f,gr({},d,{getOriginalState:h,condition:(g,b)=>v(g,b).then(Boolean),take:v,delay:vv(p.signal),pause:Hn(p.signal),extra:r,signal:p.signal,fork:Hb(p.signal,m),unsubscribe:s.unsubscribe,subscribe:()=>{t.set(s.id,s)},cancelActiveListeners:()=>{s.pending.forEach((g,b,x)=>{g!==p&&(Qt(g,Vo),x.delete(g))})},cancel:()=>{Qt(p,Vo),s.pending.delete(p)},throwIfCancelled:()=>{er(p.signal)}})))}catch(g){g instanceof Si||Ml(n,g,{raisedBy:"effect"})}finally{await Promise.all(m),Qt(p,Fb),s.pending.delete(p)}},c=Gb(t);return{middleware:s=>f=>d=>{if(!Jh(d))return f(d);if(yv.match(d))return a(d.payload);if(Vb.match(d)){c();return}if(gv.match(d))return o(d.payload);let h=s.getState();const p=()=>{if(h===Cl)throw new Error(Be(23));return h};let v;try{if(v=f(d),t.size>0){const m=s.getState(),g=Array.from(t.values());for(const b of g){let x=!1;try{x=b.predicate(d,m,h)}catch(O){x=!1,Ml(n,O,{raisedBy:"predicate"})}x&&u(b,d,s,p)}}}finally{h=Cl}return v},startListening:a,stopListening:o,clearListeners:c}};function Be(e){return`Minified Redux Toolkit error #${e}; visit https://redux-toolkit.js.org/Errors?code=${e} for the full message or use the non-minified dev environment for full errors. `}var Zb={layoutType:"horizontal",width:0,height:0,margin:{top:5,right:5,bottom:5,left:5},scale:1},bv=tt({name:"chartLayout",initialState:Zb,reducers:{setLayout(e,t){e.layoutType=t.payload},setChartSize(e,t){e.width=t.payload.width,e.height=t.payload.height},setMargin(e,t){e.margin.top=t.payload.top,e.margin.right=t.payload.right,e.margin.bottom=t.payload.bottom,e.margin.left=t.payload.left},setScale(e,t){e.scale=t.payload}}}),{setMargin:Jb,setLayout:Qb,setChartSize:ex,setScale:tx}=bv.actions,rx=bv.reducer;function Il(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function Dl(e){for(var t=1;te*180/Math.PI,ye=(e,t,r,n)=>({x:e+Math.cos(-Yn*n)*r,y:t+Math.sin(-Yn*n)*r}),ux=function(t,r){var n=arguments.length>2&&arguments[2]!==void 0?arguments[2]:{top:0,right:0,bottom:0,left:0};return Math.min(Math.abs(t-(n.left||0)-(n.right||0)),Math.abs(r-(n.top||0)-(n.bottom||0)))/2},sx=(e,t)=>{var{x:r,y:n}=e,{x:i,y:a}=t;return Math.sqrt((r-i)**2+(n-a)**2)},cx=(e,t)=>{var{x:r,y:n}=e,{cx:i,cy:a}=t,o=sx({x:r,y:n},{x:i,y:a});if(o<=0)return{radius:o,angle:0};var u=(r-i)/o,c=Math.acos(u);return n>a&&(c=2*Math.PI-c),{radius:o,angle:ox(c),angleInRadian:c}},lx=e=>{var{startAngle:t,endAngle:r}=e,n=Math.floor(t/360),i=Math.floor(r/360),a=Math.min(n,i);return{startAngle:t-a*360,endAngle:r-a*360}},fx=(e,t)=>{var{startAngle:r,endAngle:n}=t,i=Math.floor(r/360),a=Math.floor(n/360),o=Math.min(i,a);return e+o*360},dx=(e,t)=>{var{x:r,y:n}=e,{radius:i,angle:a}=cx({x:r,y:n},t),{innerRadius:o,outerRadius:u}=t;if(iu||i===0)return null;var{startAngle:c,endAngle:l}=lx(t),s=a,f;if(c<=l){for(;s>l;)s-=360;for(;s=c&&s<=l}else{for(;s>c;)s-=360;for(;s=l&&s<=c}return f?Dl(Dl({},t),{},{radius:i,angle:fx(s,t)}):null};function Nl(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function Ze(e){for(var t=1;t{var a,o=-1,u=(a=t==null?void 0:t.length)!==null&&a!==void 0?a:0;if(u<=1||e==null)return 0;if(n==="angleAxis"&&i!=null&&Math.abs(Math.abs(i[1]-i[0])-360)<=1e-6)for(var c=0;c0?r[c-1].coordinate:r[u-1].coordinate,s=r[c].coordinate,f=c>=u-1?r[0].coordinate:r[c+1].coordinate,d=void 0;if(je(s-l)!==je(f-s)){var h=[];if(je(f-s)===je(i[1]-i[0])){d=f;var p=s+i[1]-i[0];h[0]=Math.min(p,(p+l)/2),h[1]=Math.max(p,(p+l)/2)}else{d=l;var v=f+i[1]-i[0];h[0]=Math.min(s,(v+s)/2),h[1]=Math.max(s,(v+s)/2)}var m=[Math.min(s,(d+s)/2),Math.max(s,(d+s)/2)];if(e>m[0]&&e<=m[1]||e>=h[0]&&e<=h[1]){({index:o}=r[c]);break}}else{var g=Math.min(l,f),b=Math.max(l,f);if(e>(g+s)/2&&e<=(b+s)/2){({index:o}=r[c]);break}}}else if(t){for(var x=0;x0&&x(t[x].coordinate+t[x-1].coordinate)/2&&e<=(t[x].coordinate+t[x+1].coordinate)/2||x===u-1&&e>(t[x].coordinate+t[x-1].coordinate)/2){({index:o}=t[x]);break}}return o},yx=(e,t,r)=>{if(t&&r){var{width:n,height:i}=r,{align:a,verticalAlign:o,layout:u}=t;if((u==="vertical"||u==="horizontal"&&o==="middle")&&a!=="center"&&k(e[a]))return Ze(Ze({},e),{},{[a]:e[a]+(n||0)});if((u==="horizontal"||u==="vertical"&&a==="center")&&o!=="middle"&&k(e[o]))return Ze(Ze({},e),{},{[o]:e[o]+(i||0)})}return e},Lt=(e,t)=>e==="horizontal"&&t==="xAxis"||e==="vertical"&&t==="yAxis"||e==="centric"&&t==="angleAxis"||e==="radial"&&t==="radiusAxis",xv=(e,t,r,n)=>{if(n)return e.map(u=>u.coordinate);var i,a,o=e.map(u=>(u.coordinate===t&&(i=!0),u.coordinate===r&&(a=!0),u.coordinate));return i||o.push(t),a||o.push(r),o},wv=(e,t,r)=>{if(!e)return null;var{duplicateDomain:n,type:i,range:a,scale:o,realScaleType:u,isCategorical:c,categoricalDomain:l,tickCount:s,ticks:f,niceTicks:d,axisType:h}=e;if(!o)return null;var p=u==="scaleBand"&&o.bandwidth?o.bandwidth()/2:2,v=i==="category"&&o.bandwidth?o.bandwidth()/p:0;if(v=h==="angleAxis"&&a&&a.length>=2?je(a[0]-a[1])*2*v:v,f||d){var m=(f||d||[]).map((g,b)=>{var x=n?n.indexOf(g):g;return{coordinate:o(x)+v,value:g,offset:v,index:b}});return m.filter(g=>!qe(g.coordinate))}return c&&l?l.map((g,b)=>({coordinate:o(g)+v,value:g,index:b,offset:v})):o.ticks&&s!=null?o.ticks(s).map((g,b)=>({coordinate:o(g)+v,value:g,offset:v,index:b})):o.domain().map((g,b)=>({coordinate:o(g)+v,value:n?n[g]:g,index:b,offset:v}))},$l=1e-4,gx=e=>{var t=e.domain();if(!(!t||t.length<=2)){var r=t.length,n=e.range(),i=Math.min(n[0],n[1])-$l,a=Math.max(n[0],n[1])+$l,o=e(t[0]),u=e(t[r-1]);(oa||ua)&&e.domain([t[0],t[r-1]])}},bx=(e,t)=>{if(!t||t.length!==2||!k(t[0])||!k(t[1]))return e;var r=Math.min(t[0],t[1]),n=Math.max(t[0],t[1]),i=[e[0],e[1]];return(!k(e[0])||e[0]n)&&(i[1]=n),i[0]>n&&(i[0]=n),i[1]{var t=e.length;if(!(t<=0))for(var r=0,n=e[0].length;r=0?(e[o][r][0]=i,e[o][r][1]=i+u,i=e[o][r][1]):(e[o][r][0]=a,e[o][r][1]=a+u,a=e[o][r][1])}},wx=e=>{var t=e.length;if(!(t<=0))for(var r=0,n=e[0].length;r=0?(e[a][r][0]=i,e[a][r][1]=i+o,i=e[a][r][1]):(e[a][r][0]=0,e[a][r][1]=0)}},Ox={sign:xx,expand:Zg,none:wr,silhouette:Jg,wiggle:Qg,positive:wx},Px=(e,t,r)=>{var n=Ox[r],i=Xg().keys(t).value((a,o)=>+Pe(a,o,0)).order(qo).offset(n);return i(e)};function Ov(e){return e==null?void 0:String(e)}function Rl(e){var{axis:t,ticks:r,bandSize:n,entry:i,index:a,dataKey:o}=e;if(t.type==="category"){if(!t.allowDuplicatedCategory&&t.dataKey&&!X(i[t.dataKey])){var u=yh(r,"value",i[t.dataKey]);if(u)return u.coordinate+n/2}return r[a]?r[a].coordinate+n/2:null}var c=Pe(i,X(o)?t.dataKey:o);return X(c)?null:t.scale(c)}var Ll=e=>{var{axis:t,ticks:r,offset:n,bandSize:i,entry:a,index:o}=e;if(t.type==="category")return r[o]?r[o].coordinate+n:null;var u=Pe(a,t.dataKey,t.scale.domain()[o]);return X(u)?null:t.scale(u)-i/2+n},Ax=e=>{var{numericAxis:t}=e,r=t.scale.domain();if(t.type==="number"){var n=Math.min(r[0],r[1]),i=Math.max(r[0],r[1]);return n<=0&&i>=0?0:i<0?i:n}return r[0]},Sx=e=>{var t=e.flat(2).filter(k);return[Math.min(...t),Math.max(...t)]},Ex=e=>[e[0]===1/0?0:e[0],e[1]===-1/0?0:e[1]],_x=(e,t,r)=>{if(e!=null)return Ex(Object.keys(e).reduce((n,i)=>{var a=e[i],{stackedData:o}=a,u=o.reduce((c,l)=>{var s=Sx(l.slice(t,r+1));return[Math.min(c[0],s[0]),Math.max(c[1],s[1])]},[1/0,-1/0]);return[Math.min(u[0],n[0]),Math.max(u[1],n[1])]},[1/0,-1/0]))},Bl=/^dataMin[\s]*-[\s]*([0-9]+([.]{1}[0-9]+){0,1})$/,ql=/^dataMax[\s]*\+[\s]*([0-9]+([.]{1}[0-9]+){0,1})$/,Jr=(e,t,r)=>{if(e&&e.scale&&e.scale.bandwidth){var n=e.scale.bandwidth();if(!r||n>0)return n}if(e&&t&&t.length>=2){for(var i=xi(t,s=>s.coordinate),a=1/0,o=1,u=i.length;o=i.left&&e<=i.left+i.width&&t>=i.top&&t<=i.top+i.height;return a?{x:e,y:t}:null}return n?dx({x:e,y:t},n):null}var Tx=(e,t,r,n)=>{var i=t.find(l=>l&&l.index===r);if(i){if(e==="horizontal")return{x:i.coordinate,y:n.y};if(e==="vertical")return{x:n.x,y:i.coordinate};if(e==="centric"){var a=i.coordinate,{radius:o}=n;return Ze(Ze(Ze({},n),ye(n.cx,n.cy,o,a)),{},{angle:a,radius:o})}var u=i.coordinate,{angle:c}=n;return Ze(Ze(Ze({},n),ye(n.cx,n.cy,u,c)),{},{angle:c,radius:u})}return{x:0,y:0}},Cx=(e,t)=>t==="horizontal"?e.x:t==="vertical"?e.y:t==="centric"?e.angle:e.radius,St=e=>e.layout.width,Et=e=>e.layout.height,kx=e=>e.layout.scale,Pv=e=>e.layout.margin,es=S(e=>e.cartesianAxis.xAxis,e=>Object.values(e)),ts=S(e=>e.cartesianAxis.yAxis,e=>Object.values(e)),Mx="data-recharts-item-index",Ix="data-recharts-item-data-key",ji=60;function Kl(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function kt(e){for(var t=1;te.brush.height,ce=S([St,Et,Pv,Rx,es,ts,Gh,rb],(e,t,r,n,i,a,o,u)=>{var c=a.reduce((p,v)=>{var{orientation:m}=v;if(!v.mirror&&!v.hide){var g=typeof v.width=="number"?v.width:ji;return kt(kt({},p),{},{[m]:p[m]+g})}return p},{left:r.left||0,right:r.right||0}),l=i.reduce((p,v)=>{var{orientation:m}=v;return!v.mirror&&!v.hide?kt(kt({},p),{},{[m]:nr(p,"".concat(m))+v.height}):p},{top:r.top||0,bottom:r.bottom||0}),s=kt(kt({},l),c),f=s.bottom;s.bottom+=n,s=yx(s,o,u);var d=e-s.left-s.right,h=t-s.top-s.bottom;return kt(kt({brushBottom:f},s),{},{width:Math.max(d,0),height:Math.max(h,0)})}),Lx=S(ce,e=>({x:e.left,y:e.top,width:e.width,height:e.height})),Av=S(St,Et,(e,t)=>({x:0,y:0,width:e,height:t})),Bx=y.createContext(null),Se=()=>y.useContext(Bx)!=null,Ti=e=>e.brush,Ci=S([Ti,ce,Pv],(e,t,r)=>({height:e.height,x:k(e.x)?e.x:t.left,y:k(e.y)?e.y:t.top+t.height+t.brushBottom-((r==null?void 0:r.bottom)||0),width:k(e.width)?e.width:t.width})),rs=()=>{var e,t=Se(),r=D(Lx),n=D(Ci),i=(e=D(Ti))===null||e===void 0?void 0:e.padding;return!t||!n||!i?r:{width:n.width-i.left-i.right,height:n.height-i.top-i.bottom,x:i.left,y:i.top}},qx={top:0,bottom:0,left:0,right:0,width:0,height:0,brushBottom:0},Sv=()=>{var e;return(e=D(ce))!==null&&e!==void 0?e:qx},ns=()=>D(St),is=()=>D(Et),zx={top:0,right:0,bottom:0,left:0},Kx=()=>{var e;return(e=D(t=>t.layout.margin))!==null&&e!==void 0?e:zx},F=e=>e.layout.layoutType,ki=()=>D(F),Wx={settings:{layout:"horizontal",align:"center",verticalAlign:"middle",itemSorter:"value"},size:{width:0,height:0},payload:[]},Ev=tt({name:"legend",initialState:Wx,reducers:{setLegendSize(e,t){e.size.width=t.payload.width,e.size.height=t.payload.height},setLegendSettings(e,t){e.settings.align=t.payload.align,e.settings.layout=t.payload.layout,e.settings.verticalAlign=t.payload.verticalAlign,e.settings.itemSorter=t.payload.itemSorter},addLegendPayload(e,t){e.payload.push(t.payload)},removeLegendPayload(e,t){var r=gt(e).payload.indexOf(t.payload);r>-1&&e.payload.splice(r,1)}}}),{setLegendSize:Wl,setLegendSettings:Fx,addLegendPayload:_v,removeLegendPayload:jv}=Ev.actions,Ux=Ev.reducer,Hx=["contextPayload"];function Zo(){return Zo=Object.assign?Object.assign.bind():function(e){for(var t=1;t{t(Fx(e))},[t,e]),null}function tw(e){var t=re();return y.useEffect(()=>(t(Wl(e)),()=>{t(Wl({width:0,height:0}))}),[t,e]),null}function rw(e){var t=ab(),r=xg(),n=Kx(),{width:i,height:a,wrapperStyle:o,portal:u}=e,[c,l]=Vh([t]),s=ns(),f=is(),d=s-(n.left||0)-(n.right||0),h=os.getWidthOrHeight(e.layout,a,i,d),p=u?o:Pr(Pr({position:"absolute",width:(h==null?void 0:h.width)||i||"auto",height:(h==null?void 0:h.height)||a||"auto"},Qx(o,e,n,s,f,c)),o),v=u??r;if(v==null)return null;var m=y.createElement("div",{className:"recharts-legend-wrapper",style:p,ref:l},y.createElement(ew,{layout:e.layout,align:e.align,verticalAlign:e.verticalAlign,itemSorter:e.itemSorter}),y.createElement(tw,{width:c.width,height:c.height}),y.createElement(Jx,Zo({},e,h,{margin:n,chartWidth:s,chartHeight:f,contextPayload:t})));return hh.createPortal(m,v)}class os extends y.PureComponent{static getWidthOrHeight(t,r,n,i){return t==="vertical"&&k(r)?{height:r}:t==="horizontal"?{width:n||i}:null}render(){return y.createElement(rw,this.props)}}as(os,"displayName","Legend");as(os,"defaultProps",{align:"center",iconSize:14,itemSorter:"value",layout:"horizontal",verticalAlign:"bottom"});function Jo(){return Jo=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var{separator:t=" : ",contentStyle:r={},itemStyle:n={},labelStyle:i={},payload:a,formatter:o,itemSorter:u,wrapperClassName:c,labelClassName:l,label:s,labelFormatter:f,accessibilityLayer:d=!1}=e,h=()=>{if(a&&a.length){var w={padding:0,margin:0},P=(u?xi(a,u):a).map((A,E)=>{if(A.type==="none")return null;var j=A.formatter||o||ow,{value:N,name:T}=A,C=N,R=T;if(j){var L=j(N,T,A,E,a);if(Array.isArray(L))[C,R]=L;else if(L!=null)C=L;else return null}var Y=Qa({display:"block",paddingTop:4,paddingBottom:4,color:A.color||"#000"},n);return y.createElement("li",{className:"recharts-tooltip-item",key:"tooltip-item-".concat(E),style:Y},ft(R)?y.createElement("span",{className:"recharts-tooltip-item-name"},R):null,ft(R)?y.createElement("span",{className:"recharts-tooltip-item-separator"},t):null,y.createElement("span",{className:"recharts-tooltip-item-value"},C),y.createElement("span",{className:"recharts-tooltip-item-unit"},A.unit||""))});return y.createElement("ul",{className:"recharts-tooltip-item-list",style:w},P)}return null},p=Qa({margin:0,padding:10,backgroundColor:"#fff",border:"1px solid #ccc",whiteSpace:"nowrap"},r),v=Qa({margin:0},i),m=!X(s),g=m?s:"",b=H("recharts-default-tooltip",c),x=H("recharts-tooltip-label",l);m&&f&&a!==void 0&&a!==null&&(g=f(s,a));var O=d?{role:"status","aria-live":"assertive"}:{};return y.createElement("div",Jo({className:b,style:p},O),y.createElement("p",{className:x,style:v},y.isValidElement(g)?g:"".concat(g)),h())},Rr="recharts-tooltip-wrapper",sw={visibility:"hidden"};function cw(e){var{coordinate:t,translateX:r,translateY:n}=e;return H(Rr,{["".concat(Rr,"-right")]:k(r)&&t&&k(t.x)&&r>=t.x,["".concat(Rr,"-left")]:k(r)&&t&&k(t.x)&&r=t.y,["".concat(Rr,"-top")]:k(n)&&t&&k(t.y)&&n0?i:0),f=r[n]+i;if(t[n])return o[n]?s:f;var d=c[n];if(d==null)return 0;if(o[n]){var h=s,p=d;return hm?Math.max(s,d):Math.max(f,d)}function lw(e){var{translateX:t,translateY:r,useTranslate3d:n}=e;return{transform:n?"translate3d(".concat(t,"px, ").concat(r,"px, 0)"):"translate(".concat(t,"px, ").concat(r,"px)")}}function fw(e){var{allowEscapeViewBox:t,coordinate:r,offsetTopLeft:n,position:i,reverseDirection:a,tooltipBox:o,useTranslate3d:u,viewBox:c}=e,l,s,f;return o.height>0&&o.width>0&&r?(s=Hl({allowEscapeViewBox:t,coordinate:r,key:"x",offsetTopLeft:n,position:i,reverseDirection:a,tooltipDimension:o.width,viewBox:c,viewBoxDimension:c.width}),f=Hl({allowEscapeViewBox:t,coordinate:r,key:"y",offsetTopLeft:n,position:i,reverseDirection:a,tooltipDimension:o.height,viewBox:c,viewBoxDimension:c.height}),l=lw({translateX:s,translateY:f,useTranslate3d:u})):l=sw,{cssProperties:l,cssClasses:cw({translateX:s,translateY:f,coordinate:r})}}function Yl(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function Sn(e){for(var t=1;t{if(t.key==="Escape"){var r,n,i,a;this.setState({dismissed:!0,dismissedAtCoordinate:{x:(r=(n=this.props.coordinate)===null||n===void 0?void 0:n.x)!==null&&r!==void 0?r:0,y:(i=(a=this.props.coordinate)===null||a===void 0?void 0:a.y)!==null&&i!==void 0?i:0}})}})}componentDidMount(){document.addEventListener("keydown",this.handleKeyDown)}componentWillUnmount(){document.removeEventListener("keydown",this.handleKeyDown)}componentDidUpdate(){var t,r;this.state.dismissed&&(((t=this.props.coordinate)===null||t===void 0?void 0:t.x)!==this.state.dismissedAtCoordinate.x||((r=this.props.coordinate)===null||r===void 0?void 0:r.y)!==this.state.dismissedAtCoordinate.y)&&(this.state.dismissed=!1)}render(){var{active:t,allowEscapeViewBox:r,animationDuration:n,animationEasing:i,children:a,coordinate:o,hasPayload:u,isAnimationActive:c,offset:l,position:s,reverseDirection:f,useTranslate3d:d,viewBox:h,wrapperStyle:p,lastBoundingBox:v,innerRef:m,hasPortalFromProps:g}=this.props,{cssClasses:b,cssProperties:x}=fw({allowEscapeViewBox:r,coordinate:o,offsetTopLeft:l,position:s,reverseDirection:f,tooltipBox:{height:v.height,width:v.width},useTranslate3d:d,viewBox:h}),O=g?{}:Sn(Sn({transition:c&&t?"transform ".concat(n,"ms ").concat(i):void 0},x),{},{pointerEvents:"none",visibility:!this.state.dismissed&&t&&u?"visible":"hidden",position:"absolute",top:0,left:0}),w=Sn(Sn({},O),{},{visibility:!this.state.dismissed&&t&&u?"visible":"hidden"},p);return y.createElement("div",{xmlns:"http://www.w3.org/1999/xhtml",tabIndex:-1,className:b,style:w,ref:m},a)}}var pw=()=>!(typeof window<"u"&&window.document&&window.document.createElement&&window.setTimeout),Tr={isSsr:pw()},Tv=()=>D(e=>e.rootProps.accessibilityLayer);function Fe(e){return Number.isFinite(e)}function Gn(e){return typeof e=="number"&&e>0&&Number.isFinite(e)}function eu(){return eu=Object.assign?Object.assign.bind():function(e){for(var t=1;tFe(e.x)&&Fe(e.y),Lr=e=>e.x,Br=e=>e.y,bw=(e,t)=>{if(typeof e=="function")return e;var r="curve".concat(cn(e));return(r==="curveMonotone"||r==="curveBump")&&t?Xl["".concat(r).concat(t==="vertical"?"Y":"X")]:Xl[r]||gi},xw=e=>{var{type:t="linear",points:r=[],baseLine:n,layout:i,connectNulls:a=!1}=e,o=bw(t,i),u=a?r.filter(En):r,c;if(Array.isArray(n)){var l=a?n.filter(f=>En(f)):n,s=u.map((f,d)=>Vl(Vl({},f),{},{base:l[d]}));return i==="vertical"?c=xn().y(Br).x1(Lr).x0(f=>f.base.x):c=xn().x(Lr).y1(Br).y0(f=>f.base.y),c.defined(En).curve(o),c(s)}return i==="vertical"&&k(n)?c=xn().y(Br).x1(Lr).x0(n):k(n)?c=xn().x(Lr).y1(Br).y0(n):c=Eh().x(Lr).y(Br),c.defined(En).curve(o),c(u)},Cv=e=>{var{className:t,points:r,path:n,pathRef:i}=e;if((!r||!r.length)&&!n)return null;var a=r&&r.length?xw(e):n;return y.createElement("path",eu({},W(e,!1),Du(e),{className:H("recharts-curve",t),d:a===null?void 0:a,ref:i}))},ww=["x","y","top","left","width","height","className"];function tu(){return tu=Object.assign?Object.assign.bind():function(e){for(var t=1;t"M".concat(e,",").concat(i,"v").concat(n,"M").concat(a,",").concat(t,"h").concat(r),Tw=e=>{var{x:t=0,y:r=0,top:n=0,left:i=0,width:a=0,height:o=0,className:u}=e,c=Ew(e,ww),l=Ow({x:t,y:r,top:n,left:i,width:a,height:o},c);return!k(t)||!k(r)||!k(a)||!k(o)||!k(n)||!k(i)?null:y.createElement("path",tu({},W(l,!0),{className:H("recharts-cross",u),d:jw(t,r,a,o,n,i)}))};function Cw(e,t,r,n){var i=n/2;return{stroke:"none",fill:"#ccc",x:e==="horizontal"?t.x-i:r.left+.5,y:e==="horizontal"?r.top+.5:t.y-i,width:e==="horizontal"?n:r.width-1,height:e==="horizontal"?r.height-1:n}}function Jl(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function kw(e){for(var t=1;t(o[u]===void 0&&n[u]!==void 0&&(o[u]=n[u]),o),r);return a}var eo={},to={},ro={},Ql;function Nw(){return Ql||(Ql=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){if(!r||typeof r!="object")return!1;const n=Object.getPrototypeOf(r);return n===null||n===Object.prototype||Object.getPrototypeOf(n)===null?Object.prototype.toString.call(r)==="[object Object]":!1}e.isPlainObject=t}(ro)),ro}var ef;function $w(){return ef||(ef=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Nw(),r=zh(),n=Hu(),i=Yu(),a=Fu();function o(l,s,f){return u(l,s,void 0,void 0,void 0,void 0,f)}function u(l,s,f,d,h,p,v){const m=v(l,s,f,d,h,p);if(m!==void 0)return m;if(typeof l==typeof s)switch(typeof l){case"bigint":case"string":case"boolean":case"symbol":case"undefined":return l===s;case"number":return l===s||Object.is(l,s);case"function":return l===s;case"object":return c(l,s,p,v)}return c(l,s,p,v)}function c(l,s,f,d){if(Object.is(l,s))return!0;let h=n.getTag(l),p=n.getTag(s);if(h===i.argumentsTag&&(h=i.objectTag),p===i.argumentsTag&&(p=i.objectTag),h!==p)return!1;switch(h){case i.stringTag:return l.toString()===s.toString();case i.numberTag:{const g=l.valueOf(),b=s.valueOf();return a.eq(g,b)}case i.booleanTag:case i.dateTag:case i.symbolTag:return Object.is(l.valueOf(),s.valueOf());case i.regexpTag:return l.source===s.source&&l.flags===s.flags;case i.functionTag:return l===s}f=f??new Map;const v=f.get(l),m=f.get(s);if(v!=null&&m!=null)return v===s;f.set(l,s),f.set(s,l);try{switch(h){case i.mapTag:{if(l.size!==s.size)return!1;for(const[g,b]of l.entries())if(!s.has(g)||!u(b,s.get(g),g,l,s,f,d))return!1;return!0}case i.setTag:{if(l.size!==s.size)return!1;const g=Array.from(l.values()),b=Array.from(s.values());for(let x=0;xu(O,P,void 0,l,s,f,d));if(w===-1)return!1;b.splice(w,1)}return!0}case i.arrayTag:case i.uint8ArrayTag:case i.uint8ClampedArrayTag:case i.uint16ArrayTag:case i.uint32ArrayTag:case i.bigUint64ArrayTag:case i.int8ArrayTag:case i.int16ArrayTag:case i.int32ArrayTag:case i.bigInt64ArrayTag:case i.float32ArrayTag:case i.float64ArrayTag:{if(typeof Buffer<"u"&&Buffer.isBuffer(l)!==Buffer.isBuffer(s)||l.length!==s.length)return!1;for(let g=0;gnull,n=!1,i=null,a=o=>{if(!n){if(Array.isArray(o)){if(!o.length)return;var u=o,[c,...l]=u;if(typeof c=="number"){i=e.setTimeout(a.bind(null,l),c);return}a(c),i=e.setTimeout(a.bind(null,l));return}typeof o=="object"&&(t=o,r(t)),typeof o=="function"&&o()}};return{stop:()=>{n=!0},start:o=>{n=!1,i&&(i(),i=null),a(o)},subscribe:o=>(r=o,()=>{r=()=>null}),getTimeoutController:()=>e}}var Vn=1e-4,kv=(e,t)=>[0,3*e,3*t-6*e,3*e-3*t+1],Mv=(e,t)=>e.map((r,n)=>r*t**n).reduce((r,n)=>r+n),af=(e,t)=>r=>{var n=kv(e,t);return Mv(n,r)},Ww=(e,t)=>r=>{var n=kv(e,t),i=[...n.map((a,o)=>a*o).slice(1),0];return Mv(i,r)},of=function(){for(var t,r,n,i,a=arguments.length,o=new Array(a),u=0;uparseFloat(p)))}}else o.length===4&&([t,n,r,i]=o);var l=af(t,r),s=af(n,i),f=Ww(t,r),d=p=>p>1?1:p<0?0:p,h=p=>{for(var v=p>1?1:p,m=v,g=0;g<8;++g){var b=l(m)-v,x=f(m);if(Math.abs(b-v)0&&arguments[0]!==void 0?arguments[0]:{},{stiff:r=100,damping:n=8,dt:i=17}=t,a=(o,u,c)=>{var l=-(o-u)*r,s=c*n,f=c+(l-s)*i/1e3,d=c*i/1e3+o;return Math.abs(d-u){if(typeof e=="string")switch(e){case"ease":case"ease-in-out":case"ease-out":case"ease-in":case"linear":return of(e);case"spring":return Fw();default:if(e.split("(")[0]==="cubic-bezier")return of(e)}return typeof e=="function"?e:null};function uf(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function sf(e){for(var t=1;te.replace(/([A-Z])/g,t=>"-".concat(t.toLowerCase())),Xw=(e,t,r)=>e.map(n=>"".concat(Vw(n)," ").concat(t,"ms ").concat(r)).join(","),Zw=(e,t)=>[Object.keys(e),Object.keys(t)].reduce((r,n)=>r.filter(i=>n.includes(i))),Qr=(e,t)=>Object.keys(t).reduce((r,n)=>sf(sf({},r),{},{[n]:e(n,t[n])}),{});function cf(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function me(e){for(var t=1;te+(t-e)*r,ru=e=>{var{from:t,to:r}=e;return t!==r},Iv=(e,t,r)=>{var n=Qr((i,a)=>{if(ru(a)){var[o,u]=e(a.from,a.to,a.velocity);return me(me({},a),{},{from:o,velocity:u})}return a},t);return r<1?Qr((i,a)=>ru(a)?me(me({},a),{},{velocity:Xn(a.velocity,n[i].velocity,r),from:Xn(a.from,n[i].from,r)}):a,t):Iv(e,n,r-1)};function t1(e,t,r,n,i,a){var o,u=n.reduce((d,h)=>me(me({},d),{},{[h]:{from:e[h],velocity:0,to:t[h]}}),{}),c=()=>Qr((d,h)=>h.from,u),l=()=>!Object.values(u).filter(ru).length,s=null,f=d=>{o||(o=d);var h=d-o,p=h/r.dt;u=Iv(r,u,p),i(me(me(me({},e),t),c())),o=d,l()||(s=a.setTimeout(f))};return()=>(s=a.setTimeout(f),()=>{s()})}function r1(e,t,r,n,i,a,o){var u=null,c=i.reduce((f,d)=>me(me({},f),{},{[d]:[e[d],t[d]]}),{}),l,s=f=>{l||(l=f);var d=(f-l)/n,h=Qr((v,m)=>Xn(...m,r(d)),c);if(a(me(me(me({},e),t),h)),d<1)u=o.setTimeout(s);else{var p=Qr((v,m)=>Xn(...m,r(1)),c);a(me(me(me({},e),t),p))}};return()=>(u=o.setTimeout(s),()=>{u()})}const n1=(e,t,r,n,i,a)=>{var o=Zw(e,t);return r.isStepper===!0?t1(e,t,r,o,i,a):r1(e,t,r,n,o,i,a)};class i1{setTimeout(t){var r=arguments.length>1&&arguments[1]!==void 0?arguments[1]:0,n=performance.now(),i=null,a=o=>{o-n>=r?t(o):typeof requestAnimationFrame=="function"&&(i=requestAnimationFrame(a))};return i=requestAnimationFrame(a),()=>{cancelAnimationFrame(i)}}}var a1=["children","begin","duration","attributeName","easing","isActive","from","to","canBegin","onAnimationEnd","shouldReAnimate","onAnimationReStart","animationManager"];function nu(){return nu=Object.assign?Object.assign.bind():function(e){for(var t=1;t{this.stopJSAnimation=l()};this.manager.start([c,o,s,i,u])}runAnimation(t){var{begin:r,duration:n,attributeName:i,to:a,easing:o,onAnimationStart:u,onAnimationEnd:c,children:l}=t;if(this.unSubscribe=this.manager.subscribe(this.handleStyleChange),typeof o=="function"||typeof l=="function"||o==="spring"){this.runJSAnimation(t);return}var s=i?{[i]:a}:a,f=Xw(Object.keys(s),n,o);this.manager.start([u,r,Mt(Mt({},s),{},{transition:f}),n,c])}render(){var t=this.props,{children:r,begin:n,duration:i,attributeName:a,easing:o,isActive:u,from:c,to:l,canBegin:s,onAnimationEnd:f,shouldReAnimate:d,onAnimationReStart:h,animationManager:p}=t,v=o1(t,a1),m=y.Children.count(r),g=this.state.style;if(typeof r=="function")return r(g);if(!u||m===0||i<=0)return r;var b=x=>{var{style:O={},className:w}=x.props,P=y.cloneElement(x,Mt(Mt({},v),{},{style:Mt(Mt({},O),g),className:w}));return P};return m===1?b(y.Children.only(r)):y.createElement("div",null,y.Children.map(r,x=>b(x)))}}Vt(us,"displayName","Animate");Vt(us,"defaultProps",{begin:0,duration:1e3,attributeName:"",easing:"ease",isActive:!0,canBegin:!0,onAnimationEnd:()=>{},onAnimationStart:()=>{}});var f1=y.createContext(null);function or(e){var t,r,n=y.useContext(f1);return y.createElement(us,nu({},e,{animationManager:(t=(r=e.animationManager)!==null&&r!==void 0?r:n)!==null&&t!==void 0?t:l1()}))}function Zn(){return Zn=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var a=Math.min(Math.abs(r)/2,Math.abs(n)/2),o=n>=0?1:-1,u=r>=0?1:-1,c=n>=0&&r>=0||n<0&&r<0?1:0,l;if(a>0&&i instanceof Array){for(var s=[0,0,0,0],f=0,d=4;fa?a:i[f];l="M".concat(e,",").concat(t+o*s[0]),s[0]>0&&(l+="A ".concat(s[0],",").concat(s[0],",0,0,").concat(c,",").concat(e+u*s[0],",").concat(t)),l+="L ".concat(e+r-u*s[1],",").concat(t),s[1]>0&&(l+="A ".concat(s[1],",").concat(s[1],",0,0,").concat(c,`, + `).concat(e+r,",").concat(t+o*s[1])),l+="L ".concat(e+r,",").concat(t+n-o*s[2]),s[2]>0&&(l+="A ".concat(s[2],",").concat(s[2],",0,0,").concat(c,`, + `).concat(e+r-u*s[2],",").concat(t+n)),l+="L ".concat(e+u*s[3],",").concat(t+n),s[3]>0&&(l+="A ".concat(s[3],",").concat(s[3],",0,0,").concat(c,`, + `).concat(e,",").concat(t+n-o*s[3])),l+="Z"}else if(a>0&&i===+i&&i>0){var h=Math.min(a,i);l="M ".concat(e,",").concat(t+o*h,` + A `).concat(h,",").concat(h,",0,0,").concat(c,",").concat(e+u*h,",").concat(t,` + L `).concat(e+r-u*h,",").concat(t,` + A `).concat(h,",").concat(h,",0,0,").concat(c,",").concat(e+r,",").concat(t+o*h,` + L `).concat(e+r,",").concat(t+n-o*h,` + A `).concat(h,",").concat(h,",0,0,").concat(c,",").concat(e+r-u*h,",").concat(t+n,` + L `).concat(e+u*h,",").concat(t+n,` + A `).concat(h,",").concat(h,",0,0,").concat(c,",").concat(e,",").concat(t+n-o*h," Z")}else l="M ".concat(e,",").concat(t," h ").concat(r," v ").concat(n," h ").concat(-r," Z");return l},d1={x:0,y:0,width:0,height:0,radius:0,isAnimationActive:!1,isUpdateAnimationActive:!1,animationBegin:0,animationDuration:1500,animationEasing:"ease"},Dv=e=>{var t=_t(e,d1),r=y.useRef(null),[n,i]=y.useState(-1);y.useEffect(()=>{if(r.current&&r.current.getTotalLength)try{var g=r.current.getTotalLength();g&&i(g)}catch{}},[]);var{x:a,y:o,width:u,height:c,radius:l,className:s}=t,{animationEasing:f,animationDuration:d,animationBegin:h,isAnimationActive:p,isUpdateAnimationActive:v}=t;if(a!==+a||o!==+o||u!==+u||c!==+c||u===0||c===0)return null;var m=H("recharts-rectangle",s);return v?y.createElement(or,{canBegin:n>0,from:{width:u,height:c,x:a,y:o},to:{width:u,height:c,x:a,y:o},duration:d,animationEasing:f,isActive:v},g=>{var{width:b,height:x,x:O,y:w}=g;return y.createElement(or,{canBegin:n>0,from:"0px ".concat(n===-1?1:n,"px"),to:"".concat(n,"px 0px"),attributeName:"strokeDasharray",begin:h,duration:d,isActive:p,easing:f},y.createElement("path",Zn({},W(t,!0),{className:m,d:ff(O,w,b,x,l),ref:r})))}):y.createElement("path",Zn({},W(t,!0),{className:m,d:ff(a,o,u,c,l)}))};function Nv(e){var{cx:t,cy:r,radius:n,startAngle:i,endAngle:a}=e,o=ye(t,r,n,i),u=ye(t,r,n,a);return{points:[o,u],cx:t,cy:r,radius:n,startAngle:i,endAngle:a}}function iu(){return iu=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var r=je(t-e),n=Math.min(Math.abs(t-e),359.999);return r*n},_n=e=>{var{cx:t,cy:r,radius:n,angle:i,sign:a,isExternal:o,cornerRadius:u,cornerIsExternal:c}=e,l=u*(o?1:-1)+n,s=Math.asin(u/l)/Yn,f=c?i:i+a*s,d=ye(t,r,l,f),h=ye(t,r,n,f),p=c?i-a*s:i,v=ye(t,r,l*Math.cos(s*Yn),p);return{center:d,circleTangency:h,lineTangency:v,theta:s}},$v=e=>{var{cx:t,cy:r,innerRadius:n,outerRadius:i,startAngle:a,endAngle:o}=e,u=h1(a,o),c=a+u,l=ye(t,r,i,a),s=ye(t,r,i,c),f="M ".concat(l.x,",").concat(l.y,` + A `).concat(i,",").concat(i,`,0, + `).concat(+(Math.abs(u)>180),",").concat(+(a>c),`, + `).concat(s.x,",").concat(s.y,` + `);if(n>0){var d=ye(t,r,n,a),h=ye(t,r,n,c);f+="L ".concat(h.x,",").concat(h.y,` + A `).concat(n,",").concat(n,`,0, + `).concat(+(Math.abs(u)>180),",").concat(+(a<=c),`, + `).concat(d.x,",").concat(d.y," Z")}else f+="L ".concat(t,",").concat(r," Z");return f},v1=e=>{var{cx:t,cy:r,innerRadius:n,outerRadius:i,cornerRadius:a,forceCornerRadius:o,cornerIsExternal:u,startAngle:c,endAngle:l}=e,s=je(l-c),{circleTangency:f,lineTangency:d,theta:h}=_n({cx:t,cy:r,radius:i,angle:c,sign:s,cornerRadius:a,cornerIsExternal:u}),{circleTangency:p,lineTangency:v,theta:m}=_n({cx:t,cy:r,radius:i,angle:l,sign:-s,cornerRadius:a,cornerIsExternal:u}),g=u?Math.abs(c-l):Math.abs(c-l)-h-m;if(g<0)return o?"M ".concat(d.x,",").concat(d.y,` + a`).concat(a,",").concat(a,",0,0,1,").concat(a*2,`,0 + a`).concat(a,",").concat(a,",0,0,1,").concat(-a*2,`,0 + `):$v({cx:t,cy:r,innerRadius:n,outerRadius:i,startAngle:c,endAngle:l});var b="M ".concat(d.x,",").concat(d.y,` + A`).concat(a,",").concat(a,",0,0,").concat(+(s<0),",").concat(f.x,",").concat(f.y,` + A`).concat(i,",").concat(i,",0,").concat(+(g>180),",").concat(+(s<0),",").concat(p.x,",").concat(p.y,` + A`).concat(a,",").concat(a,",0,0,").concat(+(s<0),",").concat(v.x,",").concat(v.y,` + `);if(n>0){var{circleTangency:x,lineTangency:O,theta:w}=_n({cx:t,cy:r,radius:n,angle:c,sign:s,isExternal:!0,cornerRadius:a,cornerIsExternal:u}),{circleTangency:P,lineTangency:A,theta:E}=_n({cx:t,cy:r,radius:n,angle:l,sign:-s,isExternal:!0,cornerRadius:a,cornerIsExternal:u}),j=u?Math.abs(c-l):Math.abs(c-l)-w-E;if(j<0&&a===0)return"".concat(b,"L").concat(t,",").concat(r,"Z");b+="L".concat(A.x,",").concat(A.y,` + A`).concat(a,",").concat(a,",0,0,").concat(+(s<0),",").concat(P.x,",").concat(P.y,` + A`).concat(n,",").concat(n,",0,").concat(+(j>180),",").concat(+(s>0),",").concat(x.x,",").concat(x.y,` + A`).concat(a,",").concat(a,",0,0,").concat(+(s<0),",").concat(O.x,",").concat(O.y,"Z")}else b+="L".concat(t,",").concat(r,"Z");return b},p1={cx:0,cy:0,innerRadius:0,outerRadius:0,startAngle:0,endAngle:0,cornerRadius:0,forceCornerRadius:!1,cornerIsExternal:!1},Rv=e=>{var t=_t(e,p1),{cx:r,cy:n,innerRadius:i,outerRadius:a,cornerRadius:o,forceCornerRadius:u,cornerIsExternal:c,startAngle:l,endAngle:s,className:f}=t;if(a0&&Math.abs(l-s)<360?v=v1({cx:r,cy:n,innerRadius:i,outerRadius:a,cornerRadius:Math.min(p,h/2),forceCornerRadius:u,cornerIsExternal:c,startAngle:l,endAngle:s}):v=$v({cx:r,cy:n,innerRadius:i,outerRadius:a,startAngle:l,endAngle:s}),y.createElement("path",iu({},W(t,!0),{className:d,d:v}))};function m1(e,t,r){var n,i,a,o;if(e==="horizontal")n=t.x,a=n,i=r.top,o=r.top+r.height;else if(e==="vertical")i=t.y,o=i,n=r.left,a=r.left+r.width;else if(t.cx!=null&&t.cy!=null)if(e==="centric"){var{cx:u,cy:c,innerRadius:l,outerRadius:s,angle:f}=t,d=ye(u,c,l,f),h=ye(u,c,s,f);n=d.x,i=d.y,a=h.x,o=h.y}else return Nv(t);return[{x:n,y:i},{x:a,y:o}]}var ao={},oo={},uo={},df;function y1(){return df||(df=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Hh();function r(n){return t.isSymbol(n)?NaN:Number(n)}e.toNumber=r}(uo)),uo}var hf;function g1(){return hf||(hf=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=y1();function r(n){return n?(n=t.toNumber(n),n===1/0||n===-1/0?(n<0?-1:1)*Number.MAX_VALUE:n===n?n:0):n===0?n:0}e.toFinite=r}(oo)),oo}var vf;function b1(){return vf||(vf=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=Yh(),r=g1();function n(i,a,o){o&&typeof o!="number"&&t.isIterateeCall(i,a,o)&&(a=o=void 0),i=r.toFinite(i),a===void 0?(a=i,i=0):a=r.toFinite(a),o=o===void 0?it?1:e>=t?0:NaN}function O1(e,t){return e==null||t==null?NaN:te?1:t>=e?0:NaN}function ss(e){let t,r,n;e.length!==2?(t=Nt,r=(u,c)=>Nt(e(u),c),n=(u,c)=>e(u)-c):(t=e===Nt||e===O1?e:P1,r=e,n=e);function i(u,c,l=0,s=u.length){if(l>>1;r(u[f],c)<0?l=f+1:s=f}while(l>>1;r(u[f],c)<=0?l=f+1:s=f}while(ll&&n(u[f-1],c)>-n(u[f],c)?f-1:f}return{left:i,center:o,right:a}}function P1(){return 0}function Bv(e){return e===null?NaN:+e}function*A1(e,t){for(let r of e)r!=null&&(r=+r)>=r&&(yield r)}const S1=ss(Nt),fn=S1.right;ss(Bv).center;class mf extends Map{constructor(t,r=j1){if(super(),Object.defineProperties(this,{_intern:{value:new Map},_key:{value:r}}),t!=null)for(const[n,i]of t)this.set(n,i)}get(t){return super.get(yf(this,t))}has(t){return super.has(yf(this,t))}set(t,r){return super.set(E1(this,t),r)}delete(t){return super.delete(_1(this,t))}}function yf({_intern:e,_key:t},r){const n=t(r);return e.has(n)?e.get(n):r}function E1({_intern:e,_key:t},r){const n=t(r);return e.has(n)?e.get(n):(e.set(n,r),r)}function _1({_intern:e,_key:t},r){const n=t(r);return e.has(n)&&(r=e.get(n),e.delete(n)),r}function j1(e){return e!==null&&typeof e=="object"?e.valueOf():e}function T1(e=Nt){if(e===Nt)return qv;if(typeof e!="function")throw new TypeError("compare is not a function");return(t,r)=>{const n=e(t,r);return n||n===0?n:(e(r,r)===0)-(e(t,t)===0)}}function qv(e,t){return(e==null||!(e>=e))-(t==null||!(t>=t))||(et?1:0)}const C1=Math.sqrt(50),k1=Math.sqrt(10),M1=Math.sqrt(2);function Jn(e,t,r){const n=(t-e)/Math.max(0,r),i=Math.floor(Math.log10(n)),a=n/Math.pow(10,i),o=a>=C1?10:a>=k1?5:a>=M1?2:1;let u,c,l;return i<0?(l=Math.pow(10,-i)/o,u=Math.round(e*l),c=Math.round(t*l),u/lt&&--c,l=-l):(l=Math.pow(10,i)*o,u=Math.round(e/l),c=Math.round(t/l),u*lt&&--c),c0))return[];if(e===t)return[e];const n=t=i))return[];const u=a-i+1,c=new Array(u);if(n)if(o<0)for(let l=0;l=n)&&(r=n);return r}function bf(e,t){let r;for(const n of e)n!=null&&(r>n||r===void 0&&n>=n)&&(r=n);return r}function zv(e,t,r=0,n=1/0,i){if(t=Math.floor(t),r=Math.floor(Math.max(0,r)),n=Math.floor(Math.min(e.length-1,n)),!(r<=t&&t<=n))return e;for(i=i===void 0?qv:T1(i);n>r;){if(n-r>600){const c=n-r+1,l=t-r+1,s=Math.log(c),f=.5*Math.exp(2*s/3),d=.5*Math.sqrt(s*f*(c-f)/c)*(l-c/2<0?-1:1),h=Math.max(r,Math.floor(t-l*f/c+d)),p=Math.min(n,Math.floor(t+(c-l)*f/c+d));zv(e,t,h,p,i)}const a=e[t];let o=r,u=n;for(qr(e,r,t),i(e[n],a)>0&&qr(e,r,n);o0;)--u}i(e[r],a)===0?qr(e,r,u):(++u,qr(e,u,n)),u<=t&&(r=u+1),t<=u&&(n=u-1)}return e}function qr(e,t,r){const n=e[t];e[t]=e[r],e[r]=n}function I1(e,t,r){if(e=Float64Array.from(A1(e)),!(!(n=e.length)||isNaN(t=+t))){if(t<=0||n<2)return bf(e);if(t>=1)return gf(e);var n,i=(n-1)*t,a=Math.floor(i),o=gf(zv(e,a).subarray(0,a+1)),u=bf(e.subarray(a+1));return o+(u-o)*(i-a)}}function D1(e,t,r=Bv){if(!(!(n=e.length)||isNaN(t=+t))){if(t<=0||n<2)return+r(e[0],0,e);if(t>=1)return+r(e[n-1],n-1,e);var n,i=(n-1)*t,a=Math.floor(i),o=+r(e[a],a,e),u=+r(e[a+1],a+1,e);return o+(u-o)*(i-a)}}function N1(e,t,r){e=+e,t=+t,r=(i=arguments.length)<2?(t=e,e=0,1):i<3?1:+r;for(var n=-1,i=Math.max(0,Math.ceil((t-e)/r))|0,a=new Array(i);++n>8&15|t>>4&240,t>>4&15|t&240,(t&15)<<4|t&15,1):r===8?jn(t>>24&255,t>>16&255,t>>8&255,(t&255)/255):r===4?jn(t>>12&15|t>>8&240,t>>8&15|t>>4&240,t>>4&15|t&240,((t&15)<<4|t&15)/255):null):(t=L1.exec(e))?new Te(t[1],t[2],t[3],1):(t=B1.exec(e))?new Te(t[1]*255/100,t[2]*255/100,t[3]*255/100,1):(t=q1.exec(e))?jn(t[1],t[2],t[3],t[4]):(t=z1.exec(e))?jn(t[1]*255/100,t[2]*255/100,t[3]*255/100,t[4]):(t=K1.exec(e))?Ef(t[1],t[2]/100,t[3]/100,1):(t=W1.exec(e))?Ef(t[1],t[2]/100,t[3]/100,t[4]):xf.hasOwnProperty(e)?Pf(xf[e]):e==="transparent"?new Te(NaN,NaN,NaN,0):null}function Pf(e){return new Te(e>>16&255,e>>8&255,e&255,1)}function jn(e,t,r,n){return n<=0&&(e=t=r=NaN),new Te(e,t,r,n)}function H1(e){return e instanceof dn||(e=rn(e)),e?(e=e.rgb(),new Te(e.r,e.g,e.b,e.opacity)):new Te}function cu(e,t,r,n){return arguments.length===1?H1(e):new Te(e,t,r,n??1)}function Te(e,t,r,n){this.r=+e,this.g=+t,this.b=+r,this.opacity=+n}fs(Te,cu,Wv(dn,{brighter(e){return e=e==null?Qn:Math.pow(Qn,e),new Te(this.r*e,this.g*e,this.b*e,this.opacity)},darker(e){return e=e==null?en:Math.pow(en,e),new Te(this.r*e,this.g*e,this.b*e,this.opacity)},rgb(){return this},clamp(){return new Te(tr(this.r),tr(this.g),tr(this.b),ei(this.opacity))},displayable(){return-.5<=this.r&&this.r<255.5&&-.5<=this.g&&this.g<255.5&&-.5<=this.b&&this.b<255.5&&0<=this.opacity&&this.opacity<=1},hex:Af,formatHex:Af,formatHex8:Y1,formatRgb:Sf,toString:Sf}));function Af(){return`#${Xt(this.r)}${Xt(this.g)}${Xt(this.b)}`}function Y1(){return`#${Xt(this.r)}${Xt(this.g)}${Xt(this.b)}${Xt((isNaN(this.opacity)?1:this.opacity)*255)}`}function Sf(){const e=ei(this.opacity);return`${e===1?"rgb(":"rgba("}${tr(this.r)}, ${tr(this.g)}, ${tr(this.b)}${e===1?")":`, ${e})`}`}function ei(e){return isNaN(e)?1:Math.max(0,Math.min(1,e))}function tr(e){return Math.max(0,Math.min(255,Math.round(e)||0))}function Xt(e){return e=tr(e),(e<16?"0":"")+e.toString(16)}function Ef(e,t,r,n){return n<=0?e=t=r=NaN:r<=0||r>=1?e=t=NaN:t<=0&&(e=NaN),new ot(e,t,r,n)}function Fv(e){if(e instanceof ot)return new ot(e.h,e.s,e.l,e.opacity);if(e instanceof dn||(e=rn(e)),!e)return new ot;if(e instanceof ot)return e;e=e.rgb();var t=e.r/255,r=e.g/255,n=e.b/255,i=Math.min(t,r,n),a=Math.max(t,r,n),o=NaN,u=a-i,c=(a+i)/2;return u?(t===a?o=(r-n)/u+(r0&&c<1?0:o,new ot(o,u,c,e.opacity)}function G1(e,t,r,n){return arguments.length===1?Fv(e):new ot(e,t,r,n??1)}function ot(e,t,r,n){this.h=+e,this.s=+t,this.l=+r,this.opacity=+n}fs(ot,G1,Wv(dn,{brighter(e){return e=e==null?Qn:Math.pow(Qn,e),new ot(this.h,this.s,this.l*e,this.opacity)},darker(e){return e=e==null?en:Math.pow(en,e),new ot(this.h,this.s,this.l*e,this.opacity)},rgb(){var e=this.h%360+(this.h<0)*360,t=isNaN(e)||isNaN(this.s)?0:this.s,r=this.l,n=r+(r<.5?r:1-r)*t,i=2*r-n;return new Te(co(e>=240?e-240:e+120,i,n),co(e,i,n),co(e<120?e+240:e-120,i,n),this.opacity)},clamp(){return new ot(_f(this.h),Tn(this.s),Tn(this.l),ei(this.opacity))},displayable(){return(0<=this.s&&this.s<=1||isNaN(this.s))&&0<=this.l&&this.l<=1&&0<=this.opacity&&this.opacity<=1},formatHsl(){const e=ei(this.opacity);return`${e===1?"hsl(":"hsla("}${_f(this.h)}, ${Tn(this.s)*100}%, ${Tn(this.l)*100}%${e===1?")":`, ${e})`}`}}));function _f(e){return e=(e||0)%360,e<0?e+360:e}function Tn(e){return Math.max(0,Math.min(1,e||0))}function co(e,t,r){return(e<60?t+(r-t)*e/60:e<180?r:e<240?t+(r-t)*(240-e)/60:t)*255}const ds=e=>()=>e;function V1(e,t){return function(r){return e+r*t}}function X1(e,t,r){return e=Math.pow(e,r),t=Math.pow(t,r)-e,r=1/r,function(n){return Math.pow(e+n*t,r)}}function Z1(e){return(e=+e)==1?Uv:function(t,r){return r-t?X1(t,r,e):ds(isNaN(t)?r:t)}}function Uv(e,t){var r=t-e;return r?V1(e,r):ds(isNaN(e)?t:e)}const jf=function e(t){var r=Z1(t);function n(i,a){var o=r((i=cu(i)).r,(a=cu(a)).r),u=r(i.g,a.g),c=r(i.b,a.b),l=Uv(i.opacity,a.opacity);return function(s){return i.r=o(s),i.g=u(s),i.b=c(s),i.opacity=l(s),i+""}}return n.gamma=e,n}(1);function J1(e,t){t||(t=[]);var r=e?Math.min(t.length,e.length):0,n=t.slice(),i;return function(a){for(i=0;ir&&(a=t.slice(r,a),u[o]?u[o]+=a:u[++o]=a),(n=n[0])===(i=i[0])?u[o]?u[o]+=i:u[++o]=i:(u[++o]=null,c.push({i:o,x:ti(n,i)})),r=lo.lastIndex;return rt&&(r=e,e=t,t=r),function(n){return Math.max(e,Math.min(t,n))}}function cO(e,t,r){var n=e[0],i=e[1],a=t[0],o=t[1];return i2?lO:cO,c=l=null,f}function f(d){return d==null||isNaN(d=+d)?a:(c||(c=u(e.map(n),t,r)))(n(o(d)))}return f.invert=function(d){return o(i((l||(l=u(t,e.map(n),ti)))(d)))},f.domain=function(d){return arguments.length?(e=Array.from(d,ri),s()):e.slice()},f.range=function(d){return arguments.length?(t=Array.from(d),s()):t.slice()},f.rangeRound=function(d){return t=Array.from(d),r=hs,s()},f.clamp=function(d){return arguments.length?(o=d?!0:Ae,s()):o!==Ae},f.interpolate=function(d){return arguments.length?(r=d,s()):r},f.unknown=function(d){return arguments.length?(a=d,f):a},function(d,h){return n=d,i=h,s()}}function vs(){return Mi()(Ae,Ae)}function fO(e){return Math.abs(e=Math.round(e))>=1e21?e.toLocaleString("en").replace(/,/g,""):e.toString(10)}function ni(e,t){if((r=(e=t?e.toExponential(t-1):e.toExponential()).indexOf("e"))<0)return null;var r,n=e.slice(0,r);return[n.length>1?n[0]+n.slice(2):n,+e.slice(r+1)]}function Ar(e){return e=ni(Math.abs(e)),e?e[1]:NaN}function dO(e,t){return function(r,n){for(var i=r.length,a=[],o=0,u=e[0],c=0;i>0&&u>0&&(c+u+1>n&&(u=Math.max(1,n-c)),a.push(r.substring(i-=u,i+u)),!((c+=u+1)>n));)u=e[o=(o+1)%e.length];return a.reverse().join(t)}}function hO(e){return function(t){return t.replace(/[0-9]/g,function(r){return e[+r]})}}var vO=/^(?:(.)?([<>=^]))?([+\-( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?(~)?([a-z%])?$/i;function nn(e){if(!(t=vO.exec(e)))throw new Error("invalid format: "+e);var t;return new ps({fill:t[1],align:t[2],sign:t[3],symbol:t[4],zero:t[5],width:t[6],comma:t[7],precision:t[8]&&t[8].slice(1),trim:t[9],type:t[10]})}nn.prototype=ps.prototype;function ps(e){this.fill=e.fill===void 0?" ":e.fill+"",this.align=e.align===void 0?">":e.align+"",this.sign=e.sign===void 0?"-":e.sign+"",this.symbol=e.symbol===void 0?"":e.symbol+"",this.zero=!!e.zero,this.width=e.width===void 0?void 0:+e.width,this.comma=!!e.comma,this.precision=e.precision===void 0?void 0:+e.precision,this.trim=!!e.trim,this.type=e.type===void 0?"":e.type+""}ps.prototype.toString=function(){return this.fill+this.align+this.sign+this.symbol+(this.zero?"0":"")+(this.width===void 0?"":Math.max(1,this.width|0))+(this.comma?",":"")+(this.precision===void 0?"":"."+Math.max(0,this.precision|0))+(this.trim?"~":"")+this.type};function pO(e){e:for(var t=e.length,r=1,n=-1,i;r0&&(n=0);break}return n>0?e.slice(0,n)+e.slice(i+1):e}var Hv;function mO(e,t){var r=ni(e,t);if(!r)return e+"";var n=r[0],i=r[1],a=i-(Hv=Math.max(-8,Math.min(8,Math.floor(i/3)))*3)+1,o=n.length;return a===o?n:a>o?n+new Array(a-o+1).join("0"):a>0?n.slice(0,a)+"."+n.slice(a):"0."+new Array(1-a).join("0")+ni(e,Math.max(0,t+a-1))[0]}function Cf(e,t){var r=ni(e,t);if(!r)return e+"";var n=r[0],i=r[1];return i<0?"0."+new Array(-i).join("0")+n:n.length>i+1?n.slice(0,i+1)+"."+n.slice(i+1):n+new Array(i-n.length+2).join("0")}const kf={"%":(e,t)=>(e*100).toFixed(t),b:e=>Math.round(e).toString(2),c:e=>e+"",d:fO,e:(e,t)=>e.toExponential(t),f:(e,t)=>e.toFixed(t),g:(e,t)=>e.toPrecision(t),o:e=>Math.round(e).toString(8),p:(e,t)=>Cf(e*100,t),r:Cf,s:mO,X:e=>Math.round(e).toString(16).toUpperCase(),x:e=>Math.round(e).toString(16)};function Mf(e){return e}var If=Array.prototype.map,Df=["y","z","a","f","p","n","µ","m","","k","M","G","T","P","E","Z","Y"];function yO(e){var t=e.grouping===void 0||e.thousands===void 0?Mf:dO(If.call(e.grouping,Number),e.thousands+""),r=e.currency===void 0?"":e.currency[0]+"",n=e.currency===void 0?"":e.currency[1]+"",i=e.decimal===void 0?".":e.decimal+"",a=e.numerals===void 0?Mf:hO(If.call(e.numerals,String)),o=e.percent===void 0?"%":e.percent+"",u=e.minus===void 0?"−":e.minus+"",c=e.nan===void 0?"NaN":e.nan+"";function l(f){f=nn(f);var d=f.fill,h=f.align,p=f.sign,v=f.symbol,m=f.zero,g=f.width,b=f.comma,x=f.precision,O=f.trim,w=f.type;w==="n"?(b=!0,w="g"):kf[w]||(x===void 0&&(x=12),O=!0,w="g"),(m||d==="0"&&h==="=")&&(m=!0,d="0",h="=");var P=v==="$"?r:v==="#"&&/[boxX]/.test(w)?"0"+w.toLowerCase():"",A=v==="$"?n:/[%p]/.test(w)?o:"",E=kf[w],j=/[defgprs%]/.test(w);x=x===void 0?6:/[gprs]/.test(w)?Math.max(1,Math.min(21,x)):Math.max(0,Math.min(20,x));function N(T){var C=P,R=A,L,Y,te;if(w==="c")R=E(T)+R,T="";else{T=+T;var B=T<0||1/T<0;if(T=isNaN(T)?c:E(Math.abs(T),x),O&&(T=pO(T)),B&&+T==0&&p!=="+"&&(B=!1),C=(B?p==="("?p:u:p==="-"||p==="("?"":p)+C,R=(w==="s"?Df[8+Hv/3]:"")+R+(B&&p==="("?")":""),j){for(L=-1,Y=T.length;++Lte||te>57){R=(te===46?i+T.slice(L+1):T.slice(L))+R,T=T.slice(0,L);break}}}b&&!m&&(T=t(T,1/0));var he=C.length+T.length+R.length,ne=he>1)+C+T+R+ne.slice(he);break;default:T=ne+C+T+R;break}return a(T)}return N.toString=function(){return f+""},N}function s(f,d){var h=l((f=nn(f),f.type="f",f)),p=Math.max(-8,Math.min(8,Math.floor(Ar(d)/3)))*3,v=Math.pow(10,-p),m=Df[8+p/3];return function(g){return h(v*g)+m}}return{format:l,formatPrefix:s}}var Cn,ms,Yv;gO({thousands:",",grouping:[3],currency:["$",""]});function gO(e){return Cn=yO(e),ms=Cn.format,Yv=Cn.formatPrefix,Cn}function bO(e){return Math.max(0,-Ar(Math.abs(e)))}function xO(e,t){return Math.max(0,Math.max(-8,Math.min(8,Math.floor(Ar(t)/3)))*3-Ar(Math.abs(e)))}function wO(e,t){return e=Math.abs(e),t=Math.abs(t)-e,Math.max(0,Ar(t)-Ar(e))+1}function Gv(e,t,r,n){var i=uu(e,t,r),a;switch(n=nn(n??",f"),n.type){case"s":{var o=Math.max(Math.abs(e),Math.abs(t));return n.precision==null&&!isNaN(a=xO(i,o))&&(n.precision=a),Yv(n,o)}case"":case"e":case"g":case"p":case"r":{n.precision==null&&!isNaN(a=wO(i,Math.max(Math.abs(e),Math.abs(t))))&&(n.precision=a-(n.type==="e"));break}case"f":case"%":{n.precision==null&&!isNaN(a=bO(i))&&(n.precision=a-(n.type==="%")*2);break}}return ms(n)}function Bt(e){var t=e.domain;return e.ticks=function(r){var n=t();return au(n[0],n[n.length-1],r??10)},e.tickFormat=function(r,n){var i=t();return Gv(i[0],i[i.length-1],r??10,n)},e.nice=function(r){r==null&&(r=10);var n=t(),i=0,a=n.length-1,o=n[i],u=n[a],c,l,s=10;for(u0;){if(l=ou(o,u,r),l===c)return n[i]=o,n[a]=u,t(n);if(l>0)o=Math.floor(o/l)*l,u=Math.ceil(u/l)*l;else if(l<0)o=Math.ceil(o*l)/l,u=Math.floor(u*l)/l;else break;c=l}return e},e}function Vv(){var e=vs();return e.copy=function(){return hn(e,Vv())},rt.apply(e,arguments),Bt(e)}function Xv(e){var t;function r(n){return n==null||isNaN(n=+n)?t:n}return r.invert=r,r.domain=r.range=function(n){return arguments.length?(e=Array.from(n,ri),r):e.slice()},r.unknown=function(n){return arguments.length?(t=n,r):t},r.copy=function(){return Xv(e).unknown(t)},e=arguments.length?Array.from(e,ri):[0,1],Bt(r)}function Zv(e,t){e=e.slice();var r=0,n=e.length-1,i=e[r],a=e[n],o;return aMath.pow(e,t)}function EO(e){return e===Math.E?Math.log:e===10&&Math.log10||e===2&&Math.log2||(e=Math.log(e),t=>Math.log(t)/e)}function Rf(e){return(t,r)=>-e(-t,r)}function ys(e){const t=e(Nf,$f),r=t.domain;let n=10,i,a;function o(){return i=EO(n),a=SO(n),r()[0]<0?(i=Rf(i),a=Rf(a),e(OO,PO)):e(Nf,$f),t}return t.base=function(u){return arguments.length?(n=+u,o()):n},t.domain=function(u){return arguments.length?(r(u),o()):r()},t.ticks=u=>{const c=r();let l=c[0],s=c[c.length-1];const f=s0){for(;d<=h;++d)for(p=1;ps)break;g.push(v)}}else for(;d<=h;++d)for(p=n-1;p>=1;--p)if(v=d>0?p/a(-d):p*a(d),!(vs)break;g.push(v)}g.length*2{if(u==null&&(u=10),c==null&&(c=n===10?"s":","),typeof c!="function"&&(!(n%1)&&(c=nn(c)).precision==null&&(c.trim=!0),c=ms(c)),u===1/0)return c;const l=Math.max(1,n*u/t.ticks().length);return s=>{let f=s/a(Math.round(i(s)));return f*nr(Zv(r(),{floor:u=>a(Math.floor(i(u))),ceil:u=>a(Math.ceil(i(u)))})),t}function Jv(){const e=ys(Mi()).domain([1,10]);return e.copy=()=>hn(e,Jv()).base(e.base()),rt.apply(e,arguments),e}function Lf(e){return function(t){return Math.sign(t)*Math.log1p(Math.abs(t/e))}}function Bf(e){return function(t){return Math.sign(t)*Math.expm1(Math.abs(t))*e}}function gs(e){var t=1,r=e(Lf(t),Bf(t));return r.constant=function(n){return arguments.length?e(Lf(t=+n),Bf(t)):t},Bt(r)}function Qv(){var e=gs(Mi());return e.copy=function(){return hn(e,Qv()).constant(e.constant())},rt.apply(e,arguments)}function qf(e){return function(t){return t<0?-Math.pow(-t,e):Math.pow(t,e)}}function _O(e){return e<0?-Math.sqrt(-e):Math.sqrt(e)}function jO(e){return e<0?-e*e:e*e}function bs(e){var t=e(Ae,Ae),r=1;function n(){return r===1?e(Ae,Ae):r===.5?e(_O,jO):e(qf(r),qf(1/r))}return t.exponent=function(i){return arguments.length?(r=+i,n()):r},Bt(t)}function xs(){var e=bs(Mi());return e.copy=function(){return hn(e,xs()).exponent(e.exponent())},rt.apply(e,arguments),e}function TO(){return xs.apply(null,arguments).exponent(.5)}function zf(e){return Math.sign(e)*e*e}function CO(e){return Math.sign(e)*Math.sqrt(Math.abs(e))}function ep(){var e=vs(),t=[0,1],r=!1,n;function i(a){var o=CO(e(a));return isNaN(o)?n:r?Math.round(o):o}return i.invert=function(a){return e.invert(zf(a))},i.domain=function(a){return arguments.length?(e.domain(a),i):e.domain()},i.range=function(a){return arguments.length?(e.range((t=Array.from(a,ri)).map(zf)),i):t.slice()},i.rangeRound=function(a){return i.range(a).round(!0)},i.round=function(a){return arguments.length?(r=!!a,i):r},i.clamp=function(a){return arguments.length?(e.clamp(a),i):e.clamp()},i.unknown=function(a){return arguments.length?(n=a,i):n},i.copy=function(){return ep(e.domain(),t).round(r).clamp(e.clamp()).unknown(n)},rt.apply(i,arguments),Bt(i)}function tp(){var e=[],t=[],r=[],n;function i(){var o=0,u=Math.max(1,t.length);for(r=new Array(u-1);++o0?r[u-1]:e[0],u=r?[n[r-1],t]:[n[l-1],n[l]]},o.unknown=function(c){return arguments.length&&(a=c),o},o.thresholds=function(){return n.slice()},o.copy=function(){return rp().domain([e,t]).range(i).unknown(a)},rt.apply(Bt(o),arguments)}function np(){var e=[.5],t=[0,1],r,n=1;function i(a){return a!=null&&a<=a?t[fn(e,a,0,n)]:r}return i.domain=function(a){return arguments.length?(e=Array.from(a),n=Math.min(e.length,t.length-1),i):e.slice()},i.range=function(a){return arguments.length?(t=Array.from(a),n=Math.min(e.length,t.length-1),i):t.slice()},i.invertExtent=function(a){var o=t.indexOf(a);return[e[o-1],e[o]]},i.unknown=function(a){return arguments.length?(r=a,i):r},i.copy=function(){return np().domain(e).range(t).unknown(r)},rt.apply(i,arguments)}const fo=new Date,ho=new Date;function ue(e,t,r,n){function i(a){return e(a=arguments.length===0?new Date:new Date(+a)),a}return i.floor=a=>(e(a=new Date(+a)),a),i.ceil=a=>(e(a=new Date(a-1)),t(a,1),e(a),a),i.round=a=>{const o=i(a),u=i.ceil(a);return a-o(t(a=new Date(+a),o==null?1:Math.floor(o)),a),i.range=(a,o,u)=>{const c=[];if(a=i.ceil(a),u=u==null?1:Math.floor(u),!(a0))return c;let l;do c.push(l=new Date(+a)),t(a,u),e(a);while(lue(o=>{if(o>=o)for(;e(o),!a(o);)o.setTime(o-1)},(o,u)=>{if(o>=o)if(u<0)for(;++u<=0;)for(;t(o,-1),!a(o););else for(;--u>=0;)for(;t(o,1),!a(o););}),r&&(i.count=(a,o)=>(fo.setTime(+a),ho.setTime(+o),e(fo),e(ho),Math.floor(r(fo,ho))),i.every=a=>(a=Math.floor(a),!isFinite(a)||!(a>0)?null:a>1?i.filter(n?o=>n(o)%a===0:o=>i.count(0,o)%a===0):i)),i}const ii=ue(()=>{},(e,t)=>{e.setTime(+e+t)},(e,t)=>t-e);ii.every=e=>(e=Math.floor(e),!isFinite(e)||!(e>0)?null:e>1?ue(t=>{t.setTime(Math.floor(t/e)*e)},(t,r)=>{t.setTime(+t+r*e)},(t,r)=>(r-t)/e):ii);ii.range;const mt=1e3,Je=mt*60,yt=Je*60,wt=yt*24,ws=wt*7,Kf=wt*30,vo=wt*365,Zt=ue(e=>{e.setTime(e-e.getMilliseconds())},(e,t)=>{e.setTime(+e+t*mt)},(e,t)=>(t-e)/mt,e=>e.getUTCSeconds());Zt.range;const Os=ue(e=>{e.setTime(e-e.getMilliseconds()-e.getSeconds()*mt)},(e,t)=>{e.setTime(+e+t*Je)},(e,t)=>(t-e)/Je,e=>e.getMinutes());Os.range;const Ps=ue(e=>{e.setUTCSeconds(0,0)},(e,t)=>{e.setTime(+e+t*Je)},(e,t)=>(t-e)/Je,e=>e.getUTCMinutes());Ps.range;const As=ue(e=>{e.setTime(e-e.getMilliseconds()-e.getSeconds()*mt-e.getMinutes()*Je)},(e,t)=>{e.setTime(+e+t*yt)},(e,t)=>(t-e)/yt,e=>e.getHours());As.range;const Ss=ue(e=>{e.setUTCMinutes(0,0,0)},(e,t)=>{e.setTime(+e+t*yt)},(e,t)=>(t-e)/yt,e=>e.getUTCHours());Ss.range;const vn=ue(e=>e.setHours(0,0,0,0),(e,t)=>e.setDate(e.getDate()+t),(e,t)=>(t-e-(t.getTimezoneOffset()-e.getTimezoneOffset())*Je)/wt,e=>e.getDate()-1);vn.range;const Ii=ue(e=>{e.setUTCHours(0,0,0,0)},(e,t)=>{e.setUTCDate(e.getUTCDate()+t)},(e,t)=>(t-e)/wt,e=>e.getUTCDate()-1);Ii.range;const ip=ue(e=>{e.setUTCHours(0,0,0,0)},(e,t)=>{e.setUTCDate(e.getUTCDate()+t)},(e,t)=>(t-e)/wt,e=>Math.floor(e/wt));ip.range;function cr(e){return ue(t=>{t.setDate(t.getDate()-(t.getDay()+7-e)%7),t.setHours(0,0,0,0)},(t,r)=>{t.setDate(t.getDate()+r*7)},(t,r)=>(r-t-(r.getTimezoneOffset()-t.getTimezoneOffset())*Je)/ws)}const Di=cr(0),ai=cr(1),kO=cr(2),MO=cr(3),Sr=cr(4),IO=cr(5),DO=cr(6);Di.range;ai.range;kO.range;MO.range;Sr.range;IO.range;DO.range;function lr(e){return ue(t=>{t.setUTCDate(t.getUTCDate()-(t.getUTCDay()+7-e)%7),t.setUTCHours(0,0,0,0)},(t,r)=>{t.setUTCDate(t.getUTCDate()+r*7)},(t,r)=>(r-t)/ws)}const Ni=lr(0),oi=lr(1),NO=lr(2),$O=lr(3),Er=lr(4),RO=lr(5),LO=lr(6);Ni.range;oi.range;NO.range;$O.range;Er.range;RO.range;LO.range;const Es=ue(e=>{e.setDate(1),e.setHours(0,0,0,0)},(e,t)=>{e.setMonth(e.getMonth()+t)},(e,t)=>t.getMonth()-e.getMonth()+(t.getFullYear()-e.getFullYear())*12,e=>e.getMonth());Es.range;const _s=ue(e=>{e.setUTCDate(1),e.setUTCHours(0,0,0,0)},(e,t)=>{e.setUTCMonth(e.getUTCMonth()+t)},(e,t)=>t.getUTCMonth()-e.getUTCMonth()+(t.getUTCFullYear()-e.getUTCFullYear())*12,e=>e.getUTCMonth());_s.range;const Ot=ue(e=>{e.setMonth(0,1),e.setHours(0,0,0,0)},(e,t)=>{e.setFullYear(e.getFullYear()+t)},(e,t)=>t.getFullYear()-e.getFullYear(),e=>e.getFullYear());Ot.every=e=>!isFinite(e=Math.floor(e))||!(e>0)?null:ue(t=>{t.setFullYear(Math.floor(t.getFullYear()/e)*e),t.setMonth(0,1),t.setHours(0,0,0,0)},(t,r)=>{t.setFullYear(t.getFullYear()+r*e)});Ot.range;const Pt=ue(e=>{e.setUTCMonth(0,1),e.setUTCHours(0,0,0,0)},(e,t)=>{e.setUTCFullYear(e.getUTCFullYear()+t)},(e,t)=>t.getUTCFullYear()-e.getUTCFullYear(),e=>e.getUTCFullYear());Pt.every=e=>!isFinite(e=Math.floor(e))||!(e>0)?null:ue(t=>{t.setUTCFullYear(Math.floor(t.getUTCFullYear()/e)*e),t.setUTCMonth(0,1),t.setUTCHours(0,0,0,0)},(t,r)=>{t.setUTCFullYear(t.getUTCFullYear()+r*e)});Pt.range;function ap(e,t,r,n,i,a){const o=[[Zt,1,mt],[Zt,5,5*mt],[Zt,15,15*mt],[Zt,30,30*mt],[a,1,Je],[a,5,5*Je],[a,15,15*Je],[a,30,30*Je],[i,1,yt],[i,3,3*yt],[i,6,6*yt],[i,12,12*yt],[n,1,wt],[n,2,2*wt],[r,1,ws],[t,1,Kf],[t,3,3*Kf],[e,1,vo]];function u(l,s,f){const d=sm).right(o,d);if(h===o.length)return e.every(uu(l/vo,s/vo,f));if(h===0)return ii.every(Math.max(uu(l,s,f),1));const[p,v]=o[d/o[h-1][2]53)return null;"w"in _||(_.w=1),"Z"in _?(Z=mo(zr(_.y,0,1)),Me=Z.getUTCDay(),Z=Me>4||Me===0?oi.ceil(Z):oi(Z),Z=Ii.offset(Z,(_.V-1)*7),_.y=Z.getUTCFullYear(),_.m=Z.getUTCMonth(),_.d=Z.getUTCDate()+(_.w+6)%7):(Z=po(zr(_.y,0,1)),Me=Z.getDay(),Z=Me>4||Me===0?ai.ceil(Z):ai(Z),Z=vn.offset(Z,(_.V-1)*7),_.y=Z.getFullYear(),_.m=Z.getMonth(),_.d=Z.getDate()+(_.w+6)%7)}else("W"in _||"U"in _)&&("w"in _||(_.w="u"in _?_.u%7:"W"in _?1:0),Me="Z"in _?mo(zr(_.y,0,1)).getUTCDay():po(zr(_.y,0,1)).getDay(),_.m=0,_.d="W"in _?(_.w+6)%7+_.W*7-(Me+5)%7:_.w+_.U*7-(Me+6)%7);return"Z"in _?(_.H+=_.Z/100|0,_.M+=_.Z%100,mo(_)):po(_)}}function E(I,q,z,_){for(var Ee=0,Z=q.length,Me=z.length,Ie,Ft;Ee=Me)return-1;if(Ie=q.charCodeAt(Ee++),Ie===37){if(Ie=q.charAt(Ee++),Ft=w[Ie in Wf?q.charAt(Ee++):Ie],!Ft||(_=Ft(I,z,_))<0)return-1}else if(Ie!=z.charCodeAt(_++))return-1}return _}function j(I,q,z){var _=l.exec(q.slice(z));return _?(I.p=s.get(_[0].toLowerCase()),z+_[0].length):-1}function N(I,q,z){var _=h.exec(q.slice(z));return _?(I.w=p.get(_[0].toLowerCase()),z+_[0].length):-1}function T(I,q,z){var _=f.exec(q.slice(z));return _?(I.w=d.get(_[0].toLowerCase()),z+_[0].length):-1}function C(I,q,z){var _=g.exec(q.slice(z));return _?(I.m=b.get(_[0].toLowerCase()),z+_[0].length):-1}function R(I,q,z){var _=v.exec(q.slice(z));return _?(I.m=m.get(_[0].toLowerCase()),z+_[0].length):-1}function L(I,q,z){return E(I,t,q,z)}function Y(I,q,z){return E(I,r,q,z)}function te(I,q,z){return E(I,n,q,z)}function B(I){return o[I.getDay()]}function he(I){return a[I.getDay()]}function ne(I){return c[I.getMonth()]}function ke(I){return u[I.getMonth()]}function Ue(I){return i[+(I.getHours()>=12)]}function $(I){return 1+~~(I.getMonth()/3)}function ve(I){return o[I.getUTCDay()]}function Wt(I){return a[I.getUTCDay()]}function He(I){return c[I.getUTCMonth()]}function Zy(I){return u[I.getUTCMonth()]}function Jy(I){return i[+(I.getUTCHours()>=12)]}function Qy(I){return 1+~~(I.getUTCMonth()/3)}return{format:function(I){var q=P(I+="",x);return q.toString=function(){return I},q},parse:function(I){var q=A(I+="",!1);return q.toString=function(){return I},q},utcFormat:function(I){var q=P(I+="",O);return q.toString=function(){return I},q},utcParse:function(I){var q=A(I+="",!0);return q.toString=function(){return I},q}}}var Wf={"-":"",_:" ",0:"0"},le=/^\s*\d+/,FO=/^%/,UO=/[\\^$*+?|[\]().{}]/g;function K(e,t,r){var n=e<0?"-":"",i=(n?-e:e)+"",a=i.length;return n+(a[t.toLowerCase(),r]))}function YO(e,t,r){var n=le.exec(t.slice(r,r+1));return n?(e.w=+n[0],r+n[0].length):-1}function GO(e,t,r){var n=le.exec(t.slice(r,r+1));return n?(e.u=+n[0],r+n[0].length):-1}function VO(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.U=+n[0],r+n[0].length):-1}function XO(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.V=+n[0],r+n[0].length):-1}function ZO(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.W=+n[0],r+n[0].length):-1}function Ff(e,t,r){var n=le.exec(t.slice(r,r+4));return n?(e.y=+n[0],r+n[0].length):-1}function Uf(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.y=+n[0]+(+n[0]>68?1900:2e3),r+n[0].length):-1}function JO(e,t,r){var n=/^(Z)|([+-]\d\d)(?::?(\d\d))?/.exec(t.slice(r,r+6));return n?(e.Z=n[1]?0:-(n[2]+(n[3]||"00")),r+n[0].length):-1}function QO(e,t,r){var n=le.exec(t.slice(r,r+1));return n?(e.q=n[0]*3-3,r+n[0].length):-1}function eP(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.m=n[0]-1,r+n[0].length):-1}function Hf(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.d=+n[0],r+n[0].length):-1}function tP(e,t,r){var n=le.exec(t.slice(r,r+3));return n?(e.m=0,e.d=+n[0],r+n[0].length):-1}function Yf(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.H=+n[0],r+n[0].length):-1}function rP(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.M=+n[0],r+n[0].length):-1}function nP(e,t,r){var n=le.exec(t.slice(r,r+2));return n?(e.S=+n[0],r+n[0].length):-1}function iP(e,t,r){var n=le.exec(t.slice(r,r+3));return n?(e.L=+n[0],r+n[0].length):-1}function aP(e,t,r){var n=le.exec(t.slice(r,r+6));return n?(e.L=Math.floor(n[0]/1e3),r+n[0].length):-1}function oP(e,t,r){var n=FO.exec(t.slice(r,r+1));return n?r+n[0].length:-1}function uP(e,t,r){var n=le.exec(t.slice(r));return n?(e.Q=+n[0],r+n[0].length):-1}function sP(e,t,r){var n=le.exec(t.slice(r));return n?(e.s=+n[0],r+n[0].length):-1}function Gf(e,t){return K(e.getDate(),t,2)}function cP(e,t){return K(e.getHours(),t,2)}function lP(e,t){return K(e.getHours()%12||12,t,2)}function fP(e,t){return K(1+vn.count(Ot(e),e),t,3)}function op(e,t){return K(e.getMilliseconds(),t,3)}function dP(e,t){return op(e,t)+"000"}function hP(e,t){return K(e.getMonth()+1,t,2)}function vP(e,t){return K(e.getMinutes(),t,2)}function pP(e,t){return K(e.getSeconds(),t,2)}function mP(e){var t=e.getDay();return t===0?7:t}function yP(e,t){return K(Di.count(Ot(e)-1,e),t,2)}function up(e){var t=e.getDay();return t>=4||t===0?Sr(e):Sr.ceil(e)}function gP(e,t){return e=up(e),K(Sr.count(Ot(e),e)+(Ot(e).getDay()===4),t,2)}function bP(e){return e.getDay()}function xP(e,t){return K(ai.count(Ot(e)-1,e),t,2)}function wP(e,t){return K(e.getFullYear()%100,t,2)}function OP(e,t){return e=up(e),K(e.getFullYear()%100,t,2)}function PP(e,t){return K(e.getFullYear()%1e4,t,4)}function AP(e,t){var r=e.getDay();return e=r>=4||r===0?Sr(e):Sr.ceil(e),K(e.getFullYear()%1e4,t,4)}function SP(e){var t=e.getTimezoneOffset();return(t>0?"-":(t*=-1,"+"))+K(t/60|0,"0",2)+K(t%60,"0",2)}function Vf(e,t){return K(e.getUTCDate(),t,2)}function EP(e,t){return K(e.getUTCHours(),t,2)}function _P(e,t){return K(e.getUTCHours()%12||12,t,2)}function jP(e,t){return K(1+Ii.count(Pt(e),e),t,3)}function sp(e,t){return K(e.getUTCMilliseconds(),t,3)}function TP(e,t){return sp(e,t)+"000"}function CP(e,t){return K(e.getUTCMonth()+1,t,2)}function kP(e,t){return K(e.getUTCMinutes(),t,2)}function MP(e,t){return K(e.getUTCSeconds(),t,2)}function IP(e){var t=e.getUTCDay();return t===0?7:t}function DP(e,t){return K(Ni.count(Pt(e)-1,e),t,2)}function cp(e){var t=e.getUTCDay();return t>=4||t===0?Er(e):Er.ceil(e)}function NP(e,t){return e=cp(e),K(Er.count(Pt(e),e)+(Pt(e).getUTCDay()===4),t,2)}function $P(e){return e.getUTCDay()}function RP(e,t){return K(oi.count(Pt(e)-1,e),t,2)}function LP(e,t){return K(e.getUTCFullYear()%100,t,2)}function BP(e,t){return e=cp(e),K(e.getUTCFullYear()%100,t,2)}function qP(e,t){return K(e.getUTCFullYear()%1e4,t,4)}function zP(e,t){var r=e.getUTCDay();return e=r>=4||r===0?Er(e):Er.ceil(e),K(e.getUTCFullYear()%1e4,t,4)}function KP(){return"+0000"}function Xf(){return"%"}function Zf(e){return+e}function Jf(e){return Math.floor(+e/1e3)}var vr,lp,fp;WP({dateTime:"%x, %X",date:"%-m/%-d/%Y",time:"%-I:%M:%S %p",periods:["AM","PM"],days:["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"],shortDays:["Sun","Mon","Tue","Wed","Thu","Fri","Sat"],months:["January","February","March","April","May","June","July","August","September","October","November","December"],shortMonths:["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"]});function WP(e){return vr=WO(e),lp=vr.format,vr.parse,fp=vr.utcFormat,vr.utcParse,vr}function FP(e){return new Date(e)}function UP(e){return e instanceof Date?+e:+new Date(+e)}function js(e,t,r,n,i,a,o,u,c,l){var s=vs(),f=s.invert,d=s.domain,h=l(".%L"),p=l(":%S"),v=l("%I:%M"),m=l("%I %p"),g=l("%a %d"),b=l("%b %d"),x=l("%B"),O=l("%Y");function w(P){return(c(P)t(i/(e.length-1)))},r.quantiles=function(n){return Array.from({length:n+1},(i,a)=>I1(e,a/n))},r.copy=function(){return pp(t).domain(e)},jt.apply(r,arguments)}function Ri(){var e=0,t=.5,r=1,n=1,i,a,o,u,c,l=Ae,s,f=!1,d;function h(v){return isNaN(v=+v)?d:(v=.5+((v=+s(v))-a)*(n*ve.chartData,XP=S([fr],e=>{var t=e.chartData!=null?e.chartData.length-1:0;return{chartData:e.chartData,computedData:e.computedData,dataEndIndex:t,dataStartIndex:0}}),ks=(e,t,r,n)=>n?XP(e):fr(e);function _r(e){if(Array.isArray(e)&&e.length===2){var[t,r]=e;if(Fe(t)&&Fe(r))return!0}return!1}function Qf(e,t,r){return r?e:[Math.min(e[0],t[0]),Math.max(e[1],t[1])]}function ZP(e,t){if(t&&typeof e!="function"&&Array.isArray(e)&&e.length===2){var[r,n]=e,i,a;if(Fe(r))i=r;else if(typeof r=="function")return;if(Fe(n))a=n;else if(typeof n=="function")return;var o=[i,a];if(_r(o))return o}}function JP(e,t,r){if(!(!r&&t==null)){if(typeof e=="function"&&t!=null)try{var n=e(t,r);if(_r(n))return Qf(n,t,r)}catch{}if(Array.isArray(e)&&e.length===2){var[i,a]=e,o,u;if(i==="auto")t!=null&&(o=Math.min(...t));else if(k(i))o=i;else if(typeof i=="function")try{t!=null&&(o=i(t==null?void 0:t[0]))}catch{}else if(typeof i=="string"&&Bl.test(i)){var c=Bl.exec(i);if(c==null||t==null)o=void 0;else{var l=+c[1];o=t[0]-l}}else o=t==null?void 0:t[0];if(a==="auto")t!=null&&(u=Math.max(...t));else if(k(a))u=a;else if(typeof a=="function")try{t!=null&&(u=a(t==null?void 0:t[1]))}catch{}else if(typeof a=="string"&&ql.test(a)){var s=ql.exec(a);if(s==null||t==null)u=void 0;else{var f=+s[1];u=t[1]+f}}else u=t==null?void 0:t[1];var d=[o,u];if(_r(d))return t==null?d:Qf(d,t,r)}}}var kr=1e9,QP={precision:20,rounding:4,toExpNeg:-7,toExpPos:21,LN10:"2.302585092994045684017991454684364207601101488628772976033327900967572609677352480235997205089598298341967784042286"},Is,ee=!0,et="[DecimalError] ",rr=et+"Invalid argument: ",Ms=et+"Exponent out of range: ",Mr=Math.floor,Yt=Math.pow,eA=/^(\d+(\.\d*)?|\.\d+)(e[+-]?\d+)?$/i,Le,se=1e7,Q=7,bp=9007199254740991,ui=Mr(bp/Q),M={};M.absoluteValue=M.abs=function(){var e=new this.constructor(this);return e.s&&(e.s=1),e};M.comparedTo=M.cmp=function(e){var t,r,n,i,a=this;if(e=new a.constructor(e),a.s!==e.s)return a.s||-e.s;if(a.e!==e.e)return a.e>e.e^a.s<0?1:-1;for(n=a.d.length,i=e.d.length,t=0,r=ne.d[t]^a.s<0?1:-1;return n===i?0:n>i^a.s<0?1:-1};M.decimalPlaces=M.dp=function(){var e=this,t=e.d.length-1,r=(t-e.e)*Q;if(t=e.d[t],t)for(;t%10==0;t/=10)r--;return r<0?0:r};M.dividedBy=M.div=function(e){return bt(this,new this.constructor(e))};M.dividedToIntegerBy=M.idiv=function(e){var t=this,r=t.constructor;return V(bt(t,new r(e),0,1),r.precision)};M.equals=M.eq=function(e){return!this.cmp(e)};M.exponent=function(){return ae(this)};M.greaterThan=M.gt=function(e){return this.cmp(e)>0};M.greaterThanOrEqualTo=M.gte=function(e){return this.cmp(e)>=0};M.isInteger=M.isint=function(){return this.e>this.d.length-2};M.isNegative=M.isneg=function(){return this.s<0};M.isPositive=M.ispos=function(){return this.s>0};M.isZero=function(){return this.s===0};M.lessThan=M.lt=function(e){return this.cmp(e)<0};M.lessThanOrEqualTo=M.lte=function(e){return this.cmp(e)<1};M.logarithm=M.log=function(e){var t,r=this,n=r.constructor,i=n.precision,a=i+5;if(e===void 0)e=new n(10);else if(e=new n(e),e.s<1||e.eq(Le))throw Error(et+"NaN");if(r.s<1)throw Error(et+(r.s?"NaN":"-Infinity"));return r.eq(Le)?new n(0):(ee=!1,t=bt(an(r,a),an(e,a),a),ee=!0,V(t,i))};M.minus=M.sub=function(e){var t=this;return e=new t.constructor(e),t.s==e.s?Op(t,e):xp(t,(e.s=-e.s,e))};M.modulo=M.mod=function(e){var t,r=this,n=r.constructor,i=n.precision;if(e=new n(e),!e.s)throw Error(et+"NaN");return r.s?(ee=!1,t=bt(r,e,0,1).times(e),ee=!0,r.minus(t)):V(new n(r),i)};M.naturalExponential=M.exp=function(){return wp(this)};M.naturalLogarithm=M.ln=function(){return an(this)};M.negated=M.neg=function(){var e=new this.constructor(this);return e.s=-e.s||0,e};M.plus=M.add=function(e){var t=this;return e=new t.constructor(e),t.s==e.s?xp(t,e):Op(t,(e.s=-e.s,e))};M.precision=M.sd=function(e){var t,r,n,i=this;if(e!==void 0&&e!==!!e&&e!==1&&e!==0)throw Error(rr+e);if(t=ae(i)+1,n=i.d.length-1,r=n*Q+1,n=i.d[n],n){for(;n%10==0;n/=10)r--;for(n=i.d[0];n>=10;n/=10)r++}return e&&t>r?t:r};M.squareRoot=M.sqrt=function(){var e,t,r,n,i,a,o,u=this,c=u.constructor;if(u.s<1){if(!u.s)return new c(0);throw Error(et+"NaN")}for(e=ae(u),ee=!1,i=Math.sqrt(+u),i==0||i==1/0?(t=ct(u.d),(t.length+e)%2==0&&(t+="0"),i=Math.sqrt(t),e=Mr((e+1)/2)-(e<0||e%2),i==1/0?t="5e"+e:(t=i.toExponential(),t=t.slice(0,t.indexOf("e")+1)+e),n=new c(t)):n=new c(i.toString()),r=c.precision,i=o=r+3;;)if(a=n,n=a.plus(bt(u,a,o+2)).times(.5),ct(a.d).slice(0,o)===(t=ct(n.d)).slice(0,o)){if(t=t.slice(o-3,o+1),i==o&&t=="4999"){if(V(a,r+1,0),a.times(a).eq(u)){n=a;break}}else if(t!="9999")break;o+=4}return ee=!0,V(n,r)};M.times=M.mul=function(e){var t,r,n,i,a,o,u,c,l,s=this,f=s.constructor,d=s.d,h=(e=new f(e)).d;if(!s.s||!e.s)return new f(0);for(e.s*=s.s,r=s.e+e.e,c=d.length,l=h.length,c=0;){for(t=0,i=c+n;i>n;)u=a[i]+h[n]*d[i-n-1]+t,a[i--]=u%se|0,t=u/se|0;a[i]=(a[i]+t)%se|0}for(;!a[--o];)a.pop();return t?++r:a.shift(),e.d=a,e.e=r,ee?V(e,f.precision):e};M.toDecimalPlaces=M.todp=function(e,t){var r=this,n=r.constructor;return r=new n(r),e===void 0?r:(dt(e,0,kr),t===void 0?t=n.rounding:dt(t,0,8),V(r,e+ae(r)+1,t))};M.toExponential=function(e,t){var r,n=this,i=n.constructor;return e===void 0?r=ur(n,!0):(dt(e,0,kr),t===void 0?t=i.rounding:dt(t,0,8),n=V(new i(n),e+1,t),r=ur(n,!0,e+1)),r};M.toFixed=function(e,t){var r,n,i=this,a=i.constructor;return e===void 0?ur(i):(dt(e,0,kr),t===void 0?t=a.rounding:dt(t,0,8),n=V(new a(i),e+ae(i)+1,t),r=ur(n.abs(),!1,e+ae(n)+1),i.isneg()&&!i.isZero()?"-"+r:r)};M.toInteger=M.toint=function(){var e=this,t=e.constructor;return V(new t(e),ae(e)+1,t.rounding)};M.toNumber=function(){return+this};M.toPower=M.pow=function(e){var t,r,n,i,a,o,u=this,c=u.constructor,l=12,s=+(e=new c(e));if(!e.s)return new c(Le);if(u=new c(u),!u.s){if(e.s<1)throw Error(et+"Infinity");return u}if(u.eq(Le))return u;if(n=c.precision,e.eq(Le))return V(u,n);if(t=e.e,r=e.d.length-1,o=t>=r,a=u.s,o){if((r=s<0?-s:s)<=bp){for(i=new c(Le),t=Math.ceil(n/Q+4),ee=!1;r%2&&(i=i.times(u),td(i.d,t)),r=Mr(r/2),r!==0;)u=u.times(u),td(u.d,t);return ee=!0,e.s<0?new c(Le).div(i):V(i,n)}}else if(a<0)throw Error(et+"NaN");return a=a<0&&e.d[Math.max(t,r)]&1?-1:1,u.s=1,ee=!1,i=e.times(an(u,n+l)),ee=!0,i=wp(i),i.s=a,i};M.toPrecision=function(e,t){var r,n,i=this,a=i.constructor;return e===void 0?(r=ae(i),n=ur(i,r<=a.toExpNeg||r>=a.toExpPos)):(dt(e,1,kr),t===void 0?t=a.rounding:dt(t,0,8),i=V(new a(i),e,t),r=ae(i),n=ur(i,e<=r||r<=a.toExpNeg,e)),n};M.toSignificantDigits=M.tosd=function(e,t){var r=this,n=r.constructor;return e===void 0?(e=n.precision,t=n.rounding):(dt(e,1,kr),t===void 0?t=n.rounding:dt(t,0,8)),V(new n(r),e,t)};M.toString=M.valueOf=M.val=M.toJSON=M[Symbol.for("nodejs.util.inspect.custom")]=function(){var e=this,t=ae(e),r=e.constructor;return ur(e,t<=r.toExpNeg||t>=r.toExpPos)};function xp(e,t){var r,n,i,a,o,u,c,l,s=e.constructor,f=s.precision;if(!e.s||!t.s)return t.s||(t=new s(e)),ee?V(t,f):t;if(c=e.d,l=t.d,o=e.e,i=t.e,c=c.slice(),a=o-i,a){for(a<0?(n=c,a=-a,u=l.length):(n=l,i=o,u=c.length),o=Math.ceil(f/Q),u=o>u?o+1:u+1,a>u&&(a=u,n.length=1),n.reverse();a--;)n.push(0);n.reverse()}for(u=c.length,a=l.length,u-a<0&&(a=u,n=l,l=c,c=n),r=0;a;)r=(c[--a]=c[a]+l[a]+r)/se|0,c[a]%=se;for(r&&(c.unshift(r),++i),u=c.length;c[--u]==0;)c.pop();return t.d=c,t.e=i,ee?V(t,f):t}function dt(e,t,r){if(e!==~~e||er)throw Error(rr+e)}function ct(e){var t,r,n,i=e.length-1,a="",o=e[0];if(i>0){for(a+=o,t=1;to?1:-1;else for(u=c=0;ui[u]?1:-1;break}return c}function r(n,i,a){for(var o=0;a--;)n[a]-=o,o=n[a]1;)n.shift()}return function(n,i,a,o){var u,c,l,s,f,d,h,p,v,m,g,b,x,O,w,P,A,E,j=n.constructor,N=n.s==i.s?1:-1,T=n.d,C=i.d;if(!n.s)return new j(n);if(!i.s)throw Error(et+"Division by zero");for(c=n.e-i.e,A=C.length,w=T.length,h=new j(N),p=h.d=[],l=0;C[l]==(T[l]||0);)++l;if(C[l]>(T[l]||0)&&--c,a==null?b=a=j.precision:o?b=a+(ae(n)-ae(i))+1:b=a,b<0)return new j(0);if(b=b/Q+2|0,l=0,A==1)for(s=0,C=C[0],b++;(l1&&(C=e(C,s),T=e(T,s),A=C.length,w=T.length),O=A,v=T.slice(0,A),m=v.length;m=se/2&&++P;do s=0,u=t(C,v,A,m),u<0?(g=v[0],A!=m&&(g=g*se+(v[1]||0)),s=g/P|0,s>1?(s>=se&&(s=se-1),f=e(C,s),d=f.length,m=v.length,u=t(f,v,d,m),u==1&&(s--,r(f,A16)throw Error(Ms+ae(e));if(!e.s)return new s(Le);for(ee=!1,u=f,o=new s(.03125);e.abs().gte(.1);)e=e.times(o),l+=5;for(n=Math.log(Yt(2,l))/Math.LN10*2+5|0,u+=n,r=i=a=new s(Le),s.precision=u;;){if(i=V(i.times(e),u),r=r.times(++c),o=a.plus(bt(i,r,u)),ct(o.d).slice(0,u)===ct(a.d).slice(0,u)){for(;l--;)a=V(a.times(a),u);return s.precision=f,t==null?(ee=!0,V(a,f)):a}a=o}}function ae(e){for(var t=e.e*Q,r=e.d[0];r>=10;r/=10)t++;return t}function yo(e,t,r){if(t>e.LN10.sd())throw ee=!0,r&&(e.precision=r),Error(et+"LN10 precision limit exceeded");return V(new e(e.LN10),t)}function It(e){for(var t="";e--;)t+="0";return t}function an(e,t){var r,n,i,a,o,u,c,l,s,f=1,d=10,h=e,p=h.d,v=h.constructor,m=v.precision;if(h.s<1)throw Error(et+(h.s?"NaN":"-Infinity"));if(h.eq(Le))return new v(0);if(t==null?(ee=!1,l=m):l=t,h.eq(10))return t==null&&(ee=!0),yo(v,l);if(l+=d,v.precision=l,r=ct(p),n=r.charAt(0),a=ae(h),Math.abs(a)<15e14){for(;n<7&&n!=1||n==1&&r.charAt(1)>3;)h=h.times(e),r=ct(h.d),n=r.charAt(0),f++;a=ae(h),n>1?(h=new v("0."+r),a++):h=new v(n+"."+r.slice(1))}else return c=yo(v,l+2,m).times(a+""),h=an(new v(n+"."+r.slice(1)),l-d).plus(c),v.precision=m,t==null?(ee=!0,V(h,m)):h;for(u=o=h=bt(h.minus(Le),h.plus(Le),l),s=V(h.times(h),l),i=3;;){if(o=V(o.times(s),l),c=u.plus(bt(o,new v(i),l)),ct(c.d).slice(0,l)===ct(u.d).slice(0,l))return u=u.times(2),a!==0&&(u=u.plus(yo(v,l+2,m).times(a+""))),u=bt(u,new v(f),l),v.precision=m,t==null?(ee=!0,V(u,m)):u;u=c,i+=2}}function ed(e,t){var r,n,i;for((r=t.indexOf("."))>-1&&(t=t.replace(".","")),(n=t.search(/e/i))>0?(r<0&&(r=n),r+=+t.slice(n+1),t=t.substring(0,n)):r<0&&(r=t.length),n=0;t.charCodeAt(n)===48;)++n;for(i=t.length;t.charCodeAt(i-1)===48;)--i;if(t=t.slice(n,i),t){if(i-=n,r=r-n-1,e.e=Mr(r/Q),e.d=[],n=(r+1)%Q,r<0&&(n+=Q),nui||e.e<-ui))throw Error(Ms+r)}else e.s=0,e.e=0,e.d=[0];return e}function V(e,t,r){var n,i,a,o,u,c,l,s,f=e.d;for(o=1,a=f[0];a>=10;a/=10)o++;if(n=t-o,n<0)n+=Q,i=t,l=f[s=0];else{if(s=Math.ceil((n+1)/Q),a=f.length,s>=a)return e;for(l=a=f[s],o=1;a>=10;a/=10)o++;n%=Q,i=n-Q+o}if(r!==void 0&&(a=Yt(10,o-i-1),u=l/a%10|0,c=t<0||f[s+1]!==void 0||l%a,c=r<4?(u||c)&&(r==0||r==(e.s<0?3:2)):u>5||u==5&&(r==4||c||r==6&&(n>0?i>0?l/Yt(10,o-i):0:f[s-1])%10&1||r==(e.s<0?8:7))),t<1||!f[0])return c?(a=ae(e),f.length=1,t=t-a-1,f[0]=Yt(10,(Q-t%Q)%Q),e.e=Mr(-t/Q)||0):(f.length=1,f[0]=e.e=e.s=0),e;if(n==0?(f.length=s,a=1,s--):(f.length=s+1,a=Yt(10,Q-n),f[s]=i>0?(l/Yt(10,o-i)%Yt(10,i)|0)*a:0),c)for(;;)if(s==0){(f[0]+=a)==se&&(f[0]=1,++e.e);break}else{if(f[s]+=a,f[s]!=se)break;f[s--]=0,a=1}for(n=f.length;f[--n]===0;)f.pop();if(ee&&(e.e>ui||e.e<-ui))throw Error(Ms+ae(e));return e}function Op(e,t){var r,n,i,a,o,u,c,l,s,f,d=e.constructor,h=d.precision;if(!e.s||!t.s)return t.s?t.s=-t.s:t=new d(e),ee?V(t,h):t;if(c=e.d,f=t.d,n=t.e,l=e.e,c=c.slice(),o=l-n,o){for(s=o<0,s?(r=c,o=-o,u=f.length):(r=f,n=l,u=c.length),i=Math.max(Math.ceil(h/Q),u)+2,o>i&&(o=i,r.length=1),r.reverse(),i=o;i--;)r.push(0);r.reverse()}else{for(i=c.length,u=f.length,s=i0;--i)c[u++]=0;for(i=f.length;i>o;){if(c[--i]0?a=a.charAt(0)+"."+a.slice(1)+It(n):o>1&&(a=a.charAt(0)+"."+a.slice(1)),a=a+(i<0?"e":"e+")+i):i<0?(a="0."+It(-i-1)+a,r&&(n=r-o)>0&&(a+=It(n))):i>=o?(a+=It(i+1-o),r&&(n=r-i-1)>0&&(a=a+"."+It(n))):((n=i+1)0&&(i+1===o&&(a+="."),a+=It(n))),e.s<0?"-"+a:a}function td(e,t){if(e.length>t)return e.length=t,!0}function Pp(e){var t,r,n;function i(a){var o=this;if(!(o instanceof i))return new i(a);if(o.constructor=i,a instanceof i){o.s=a.s,o.e=a.e,o.d=(a=a.d)?a.slice():a;return}if(typeof a=="number"){if(a*0!==0)throw Error(rr+a);if(a>0)o.s=1;else if(a<0)a=-a,o.s=-1;else{o.s=0,o.e=0,o.d=[0];return}if(a===~~a&&a<1e7){o.e=0,o.d=[a];return}return ed(o,a.toString())}else if(typeof a!="string")throw Error(rr+a);if(a.charCodeAt(0)===45?(a=a.slice(1),o.s=-1):o.s=1,eA.test(a))ed(o,a);else throw Error(rr+a)}if(i.prototype=M,i.ROUND_UP=0,i.ROUND_DOWN=1,i.ROUND_CEIL=2,i.ROUND_FLOOR=3,i.ROUND_HALF_UP=4,i.ROUND_HALF_DOWN=5,i.ROUND_HALF_EVEN=6,i.ROUND_HALF_CEIL=7,i.ROUND_HALF_FLOOR=8,i.clone=Pp,i.config=i.set=tA,e===void 0&&(e={}),e)for(n=["precision","rounding","toExpNeg","toExpPos","LN10"],t=0;t=i[t+1]&&n<=i[t+2])this[r]=n;else throw Error(rr+r+": "+n);if((n=e[r="LN10"])!==void 0)if(n==Math.LN10)this[r]=new this(n);else throw Error(rr+r+": "+n);return this}var Is=Pp(QP);Le=new Is(1);const U=Is;var rA=e=>e,Ap={},Sp=e=>e===Ap,rd=e=>function t(){return arguments.length===0||arguments.length===1&&Sp(arguments.length<=0?void 0:arguments[0])?t:e(...arguments)},Ep=(e,t)=>e===1?t:rd(function(){for(var r=arguments.length,n=new Array(r),i=0;io!==Ap).length;return a>=e?t(...n):Ep(e-a,rd(function(){for(var o=arguments.length,u=new Array(o),c=0;cSp(s)?u.shift():s);return t(...l,...u)}))}),Li=e=>Ep(e.length,e),du=(e,t)=>{for(var r=[],n=e;nArray.isArray(t)?t.map(e):Object.keys(t).map(r=>t[r]).map(e)),iA=function(){for(var t=arguments.length,r=new Array(t),n=0;nc(u),a(...arguments))}},hu=e=>Array.isArray(e)?e.reverse():e.split("").reverse().join(""),_p=e=>{var t=null,r=null;return function(){for(var n=arguments.length,i=new Array(n),a=0;a{var c;return o===((c=t)===null||c===void 0?void 0:c[u])})||(t=i,r=e(...i)),r}};function jp(e){var t;return e===0?t=1:t=Math.floor(new U(e).abs().log(10).toNumber())+1,t}function Tp(e,t,r){for(var n=new U(e),i=0,a=[];n.lt(t)&&i<1e5;)a.push(n.toNumber()),n=n.add(r),i++;return a}Li((e,t,r)=>{var n=+e,i=+t;return n+r*(i-n)});Li((e,t,r)=>{var n=t-+e;return n=n||1/0,(r-e)/n});Li((e,t,r)=>{var n=t-+e;return n=n||1/0,Math.max(0,Math.min(1,(r-e)/n))});var Cp=e=>{var[t,r]=e,[n,i]=[t,r];return t>r&&([n,i]=[r,t]),[n,i]},kp=(e,t,r)=>{if(e.lte(0))return new U(0);var n=jp(e.toNumber()),i=new U(10).pow(n),a=e.div(i),o=n!==1?.05:.1,u=new U(Math.ceil(a.div(o).toNumber())).add(r).mul(o),c=u.mul(i);return t?new U(c.toNumber()):new U(Math.ceil(c.toNumber()))},aA=(e,t,r)=>{var n=new U(1),i=new U(e);if(!i.isint()&&r){var a=Math.abs(e);a<1?(n=new U(10).pow(jp(e)-1),i=new U(Math.floor(i.div(n).toNumber())).mul(n)):a>1&&(i=new U(Math.floor(e)))}else e===0?i=new U(Math.floor((t-1)/2)):r||(i=new U(Math.floor(e)));var o=Math.floor((t-1)/2),u=iA(nA(c=>i.add(new U(c-o).mul(n)).toNumber()),du);return u(0,t)},Mp=function(t,r,n,i){var a=arguments.length>4&&arguments[4]!==void 0?arguments[4]:0;if(!Number.isFinite((r-t)/(n-1)))return{step:new U(0),tickMin:new U(0),tickMax:new U(0)};var o=kp(new U(r).sub(t).div(n-1),i,a),u;t<=0&&r>=0?u=new U(0):(u=new U(t).add(r).div(2),u=u.sub(new U(u).mod(o)));var c=Math.ceil(u.sub(t).div(o).toNumber()),l=Math.ceil(new U(r).sub(u).div(o).toNumber()),s=c+l+1;return s>n?Mp(t,r,n,i,a+1):(s0?l+(n-s):l,c=r>0?c:c+(n-s)),{step:o,tickMin:u.sub(new U(c).mul(o)),tickMax:u.add(new U(l).mul(o))})};function oA(e){var[t,r]=e,n=arguments.length>1&&arguments[1]!==void 0?arguments[1]:6,i=arguments.length>2&&arguments[2]!==void 0?arguments[2]:!0,a=Math.max(n,2),[o,u]=Cp([t,r]);if(o===-1/0||u===1/0){var c=u===1/0?[o,...du(0,n-1).map(()=>1/0)]:[...du(0,n-1).map(()=>-1/0),u];return t>r?hu(c):c}if(o===u)return aA(o,n,i);var{step:l,tickMin:s,tickMax:f}=Mp(o,u,a,i,0),d=Tp(s,f.add(new U(.1).mul(l)),l);return t>r?hu(d):d}function uA(e,t){var[r,n]=e,i=arguments.length>2&&arguments[2]!==void 0?arguments[2]:!0,[a,o]=Cp([r,n]);if(a===-1/0||o===1/0)return[r,n];if(a===o)return[a];var u=Math.max(t,2),c=kp(new U(o).sub(a).div(u-1),i,0),l=[...Tp(new U(a),new U(o),c),o];return i===!1&&(l=l.map(s=>Math.round(s))),r>n?hu(l):l}var sA=_p(oA),cA=_p(uA),Ip=e=>e.rootProps.maxBarSize,lA=e=>e.rootProps.barGap,Dp=e=>e.rootProps.barCategoryGap,fA=e=>e.rootProps.barSize,Bi=e=>e.rootProps.stackOffset,Ds=e=>e.options.chartName,Ns=e=>e.rootProps.syncId,Np=e=>e.rootProps.syncMethod,$s=e=>e.options.eventEmitter,vt={allowDuplicatedCategory:!0,angleAxisId:0,reversed:!1,scale:"auto",tick:!0,type:"category"},$e={allowDataOverflow:!1,allowDuplicatedCategory:!0,radiusAxisId:0,scale:"auto",tick:!0,tickCount:5,type:"number"},qi=(e,t)=>{if(!(!e||!t))return e!=null&&e.reversed?[t[1],t[0]]:t},dA={allowDataOverflow:!1,allowDecimals:!1,allowDuplicatedCategory:!1,dataKey:void 0,domain:void 0,id:vt.angleAxisId,includeHidden:!1,name:void 0,reversed:vt.reversed,scale:vt.scale,tick:vt.tick,tickCount:void 0,ticks:void 0,type:vt.type,unit:void 0},hA={allowDataOverflow:$e.allowDataOverflow,allowDecimals:!1,allowDuplicatedCategory:$e.allowDuplicatedCategory,dataKey:void 0,domain:void 0,id:$e.radiusAxisId,includeHidden:!1,name:void 0,reversed:!1,scale:$e.scale,tick:$e.tick,tickCount:$e.tickCount,ticks:void 0,type:$e.type,unit:void 0},vA={allowDataOverflow:!1,allowDecimals:!1,allowDuplicatedCategory:vt.allowDuplicatedCategory,dataKey:void 0,domain:void 0,id:vt.angleAxisId,includeHidden:!1,name:void 0,reversed:!1,scale:vt.scale,tick:vt.tick,tickCount:void 0,ticks:void 0,type:"number",unit:void 0},pA={allowDataOverflow:$e.allowDataOverflow,allowDecimals:!1,allowDuplicatedCategory:$e.allowDuplicatedCategory,dataKey:void 0,domain:void 0,id:$e.radiusAxisId,includeHidden:!1,name:void 0,reversed:!1,scale:$e.scale,tick:$e.tick,tickCount:$e.tickCount,ticks:void 0,type:"category",unit:void 0},Rs=(e,t)=>e.polarAxis.angleAxis[t]!=null?e.polarAxis.angleAxis[t]:e.layout.layoutType==="radial"?vA:dA,Ls=(e,t)=>e.polarAxis.radiusAxis[t]!=null?e.polarAxis.radiusAxis[t]:e.layout.layoutType==="radial"?pA:hA,zi=e=>e.polarOptions,Bs=S([St,Et,ce],ux),$p=S([zi,Bs],(e,t)=>{if(e!=null)return ut(e.innerRadius,t,0)}),Rp=S([zi,Bs],(e,t)=>{if(e!=null)return ut(e.outerRadius,t,t*.8)}),mA=e=>{if(e==null)return[0,0];var{startAngle:t,endAngle:r}=e;return[t,r]},Lp=S([zi],mA);S([Rs,Lp],qi);var Bp=S([Bs,$p,Rp],(e,t,r)=>{if(!(e==null||t==null||r==null))return[t,r]});S([Ls,Bp],qi);var yA=S([F,zi,$p,Rp,St,Et],(e,t,r,n,i,a)=>{if(!(e!=="centric"&&e!=="radial"||t==null||r==null||n==null)){var{cx:o,cy:u,startAngle:c,endAngle:l}=t;return{cx:ut(o,i,i/2),cy:ut(u,a,a/2),innerRadius:r,outerRadius:n,startAngle:c,endAngle:l,clockWise:!1}}}),fe=(e,t)=>t,Ki=(e,t,r)=>r;function nd(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function si(e){for(var t=1;t{var r=e.cartesianAxis.xAxis[t];return r??De},Ne={allowDataOverflow:!1,allowDecimals:!0,allowDuplicatedCategory:!0,angle:0,dataKey:void 0,domain:vu,hide:!0,id:0,includeHidden:!1,interval:"preserveEnd",minTickGap:5,mirror:!1,name:void 0,orientation:"left",padding:{top:0,bottom:0},reversed:!1,scale:"auto",tick:!0,tickCount:5,tickFormatter:void 0,ticks:void 0,type:"number",unit:void 0,width:ji},zt=(e,t)=>{var r=e.cartesianAxis.yAxis[t];return r??Ne},wA={domain:[0,"auto"],includeHidden:!1,reversed:!1,allowDataOverflow:!1,allowDuplicatedCategory:!1,dataKey:void 0,id:0,name:"",range:[64,64],scale:"auto",type:"number",unit:""},qs=(e,t)=>{var r=e.cartesianAxis.zAxis[t];return r??wA},Ce=(e,t,r)=>{switch(t){case"xAxis":return Tt(e,r);case"yAxis":return zt(e,r);case"zAxis":return qs(e,r);case"angleAxis":return Rs(e,r);case"radiusAxis":return Ls(e,r);default:throw new Error("Unexpected axis type: ".concat(t))}},OA=(e,t,r)=>{switch(t){case"xAxis":return Tt(e,r);case"yAxis":return zt(e,r);default:throw new Error("Unexpected axis type: ".concat(t))}},pn=(e,t,r)=>{switch(t){case"xAxis":return Tt(e,r);case"yAxis":return zt(e,r);case"angleAxis":return Rs(e,r);case"radiusAxis":return Ls(e,r);default:throw new Error("Unexpected axis type: ".concat(t))}},qp=e=>e.graphicalItems.countOfBars>0;function zp(e,t){return r=>{switch(e){case"xAxis":return"xAxisId"in r&&r.xAxisId===t;case"yAxis":return"yAxisId"in r&&r.yAxisId===t;case"zAxis":return"zAxisId"in r&&r.zAxisId===t;case"angleAxis":return"angleAxisId"in r&&r.angleAxisId===t;case"radiusAxis":return"radiusAxisId"in r&&r.radiusAxisId===t;default:return!1}}}var Wi=e=>e.graphicalItems.cartesianItems,PA=S([fe,Ki],zp),Kp=(e,t,r)=>e.filter(r).filter(n=>(t==null?void 0:t.includeHidden)===!0?!0:!n.hide),mn=S([Wi,Ce,PA],Kp),Wp=e=>e.filter(t=>t.stackId===void 0),AA=S([mn],Wp),Fp=e=>e.map(t=>t.data).filter(Boolean).flat(1),SA=S([mn],Fp),Up=(e,t)=>{var{chartData:r=[],dataStartIndex:n,dataEndIndex:i}=t;return e.length>0?e:r.slice(n,i+1)},Fi=S([SA,ks],Up),Hp=(e,t,r)=>(t==null?void 0:t.dataKey)!=null?e.map(n=>({value:Pe(n,t.dataKey)})):r.length>0?r.map(n=>n.dataKey).flatMap(n=>e.map(i=>({value:Pe(i,n)}))):e.map(n=>({value:n})),Ui=S([Fi,Ce,mn],Hp);function Yp(e,t){switch(e){case"xAxis":return t.direction==="x";case"yAxis":return t.direction==="y";default:return!1}}function dr(e){return e.filter(t=>ft(t)||t instanceof Date).map(Number).filter(t=>qe(t)===!1)}function EA(e,t,r){return!r||typeof t!="number"||qe(t)?[]:r.length?dr(r.flatMap(n=>{var i=Pe(e,n.dataKey),a,o;if(Array.isArray(i)?[a,o]=i:a=o=i,!(!Fe(a)||!Fe(o)))return[t-a,t+o]})):[]}var Gp=(e,t,r)=>{var n={},i=t.reduce((a,o)=>(o.stackId==null||(a[o.stackId]==null&&(a[o.stackId]=[]),a[o.stackId].push(o)),a),n);return Object.fromEntries(Object.entries(i).map(a=>{var[o,u]=a,c=u.map(l=>l.dataKey);return[o,{stackedData:Px(e,c,r),graphicalItems:u}]}))},pu=S([Fi,mn,Bi],Gp),Vp=(e,t,r)=>{var{dataStartIndex:n,dataEndIndex:i}=t;if(r!=="zAxis"){var a=_x(e,n,i);if(!(a!=null&&a[0]===0&&a[1]===0))return a}},_A=S([pu,fr,fe],Vp),Xp=(e,t,r,n)=>r.length>0?e.flatMap(i=>r.flatMap(a=>{var o,u,c=(o=a.errorBars)===null||o===void 0?void 0:o.filter(s=>Yp(n,s)),l=Pe(i,(u=t.dataKey)!==null&&u!==void 0?u:a.dataKey);return{value:l,errorDomain:EA(i,l,c)}})).filter(Boolean):(t==null?void 0:t.dataKey)!=null?e.map(i=>({value:Pe(i,t.dataKey),errorDomain:[]})):e.map(i=>({value:i,errorDomain:[]})),jA=S(Fi,Ce,AA,fe,Xp);function TA(e){var{value:t}=e;if(ft(t)||t instanceof Date)return t}var CA=e=>{var t=e.flatMap(n=>[n.value,n.errorDomain]).flat(1),r=dr(t);if(r.length!==0)return[Math.min(...r),Math.max(...r)]},kA=(e,t,r)=>{var n=e.map(TA).filter(i=>i!=null);return r&&(t.dataKey==null||t.allowDuplicatedCategory&&mh(n))?Lv(0,e.length):t.allowDuplicatedCategory?n:Array.from(new Set(n))},zs=e=>{var t;if(e==null||!("domain"in e))return vu;if(e.domain!=null)return e.domain;if(e.ticks!=null){if(e.type==="number"){var r=dr(e.ticks);return[Math.min(...r),Math.max(...r)]}if(e.type==="category")return e.ticks.map(String)}return(t=e==null?void 0:e.domain)!==null&&t!==void 0?t:vu},Ks=function(){for(var t=arguments.length,r=new Array(t),n=0;ne.referenceElements.dots,Ir=(e,t,r)=>e.filter(n=>n.ifOverflow==="extendDomain").filter(n=>t==="xAxis"?n.xAxisId===r:n.yAxisId===r),MA=S([Zp,fe,Ki],Ir),Jp=e=>e.referenceElements.areas,IA=S([Jp,fe,Ki],Ir),Qp=e=>e.referenceElements.lines,DA=S([Qp,fe,Ki],Ir),em=(e,t)=>{var r=dr(e.map(n=>t==="xAxis"?n.x:n.y));if(r.length!==0)return[Math.min(...r),Math.max(...r)]},NA=S(MA,fe,em),tm=(e,t)=>{var r=dr(e.flatMap(n=>[t==="xAxis"?n.x1:n.y1,t==="xAxis"?n.x2:n.y2]));if(r.length!==0)return[Math.min(...r),Math.max(...r)]},$A=S([IA,fe],tm),rm=(e,t)=>{var r=dr(e.map(n=>t==="xAxis"?n.x:n.y));if(r.length!==0)return[Math.min(...r),Math.max(...r)]},RA=S(DA,fe,rm),LA=S(NA,RA,$A,(e,t,r)=>Ks(e,r,t)),BA=S([Ce],zs),nm=(e,t,r,n,i)=>{var a=ZP(t,e.allowDataOverflow);return a??JP(t,Ks(r,i,CA(n)),e.allowDataOverflow)},qA=S([Ce,BA,_A,jA,LA],nm),zA=[0,1],im=(e,t,r,n,i,a,o)=>{if(!(e==null||r==null||r.length===0)){var{dataKey:u,type:c}=e,l=Lt(t,a);return l&&u==null?Lv(0,r.length):c==="category"?kA(n,e,l):i==="expand"?zA:o}},Ws=S([Ce,F,Fi,Ui,Bi,fe,qA],im),am=(e,t,r,n,i)=>{if(e!=null){var{scale:a,type:o}=e;if(a==="auto")return t==="radial"&&i==="radiusAxis"?"band":t==="radial"&&i==="angleAxis"?"linear":o==="category"&&n&&(n.indexOf("LineChart")>=0||n.indexOf("AreaChart")>=0||n.indexOf("ComposedChart")>=0&&!r)?"point":o==="category"?"band":"linear";if(typeof a=="string"){var u="scale".concat(cn(a));return u in Hr?u:"point"}}},yn=S([Ce,F,qp,Ds,fe],am);function KA(e){if(e!=null){if(e in Hr)return Hr[e]();var t="scale".concat(cn(e));if(t in Hr)return Hr[t]()}}function Fs(e,t,r,n){if(!(r==null||n==null)){if(typeof e.scale=="function")return e.scale.copy().domain(r).range(n);var i=KA(t);if(i!=null){var a=i.domain(r).range(n);return gx(a),a}}}var om=(e,t,r)=>{var n=zs(t);if(!(r!=="auto"&&r!=="linear")){if(t!=null&&t.tickCount&&Array.isArray(n)&&(n[0]==="auto"||n[1]==="auto")&&_r(e))return sA(e,t.tickCount,t.allowDecimals);if(t!=null&&t.tickCount&&t.type==="number"&&_r(e))return cA(e,t.tickCount,t.allowDecimals)}},Us=S([Ws,pn,yn],om),um=(e,t,r,n)=>{if(n!=="angleAxis"&&(e==null?void 0:e.type)==="number"&&_r(t)&&Array.isArray(r)&&r.length>0){var i=t[0],a=r[0],o=t[1],u=r[r.length-1];return[Math.min(i,a),Math.max(o,u)]}return t},WA=S([Ce,Ws,Us,fe],um),FA=S(Ui,Ce,(e,t)=>{if(!(!t||t.type!=="number")){var r=1/0,n=Array.from(dr(e.map(u=>u.value))).sort((u,c)=>u-c);if(n.length<2)return 1/0;var i=n[n.length-1]-n[0];if(i===0)return 1/0;for(var a=0;an,(e,t,r,n,i)=>{if(!Fe(e))return 0;var a=t==="vertical"?n.height:n.width;if(i==="gap")return e*a/2;if(i==="no-gap"){var o=ut(r,e*a),u=e*a/2;return u-o-(u-o)/a*o}return 0}),UA=(e,t)=>{var r=Tt(e,t);return r==null||typeof r.padding!="string"?0:sm(e,"xAxis",t,r.padding)},HA=(e,t)=>{var r=zt(e,t);return r==null||typeof r.padding!="string"?0:sm(e,"yAxis",t,r.padding)},YA=S(Tt,UA,(e,t)=>{var r,n;if(e==null)return{left:0,right:0};var{padding:i}=e;return typeof i=="string"?{left:t,right:t}:{left:((r=i.left)!==null&&r!==void 0?r:0)+t,right:((n=i.right)!==null&&n!==void 0?n:0)+t}}),GA=S(zt,HA,(e,t)=>{var r,n;if(e==null)return{top:0,bottom:0};var{padding:i}=e;return typeof i=="string"?{top:t,bottom:t}:{top:((r=i.top)!==null&&r!==void 0?r:0)+t,bottom:((n=i.bottom)!==null&&n!==void 0?n:0)+t}}),VA=S([ce,YA,Ci,Ti,(e,t,r)=>r],(e,t,r,n,i)=>{var{padding:a}=n;return i?[a.left,r.width-a.right]:[e.left+t.left,e.left+e.width-t.right]}),XA=S([ce,F,GA,Ci,Ti,(e,t,r)=>r],(e,t,r,n,i,a)=>{var{padding:o}=i;return a?[n.height-o.bottom,o.top]:t==="horizontal"?[e.top+e.height-r.bottom,e.top+r.top]:[e.top+r.top,e.top+e.height-r.bottom]}),gn=(e,t,r,n)=>{var i;switch(t){case"xAxis":return VA(e,r,n);case"yAxis":return XA(e,r,n);case"zAxis":return(i=qs(e,r))===null||i===void 0?void 0:i.range;case"angleAxis":return Lp(e);case"radiusAxis":return Bp(e,r);default:return}},cm=S([Ce,gn],qi),Dr=S([Ce,yn,WA,cm],Fs);S(mn,fe,(e,t)=>e.flatMap(r=>{var n;return(n=r.errorBars)!==null&&n!==void 0?n:[]}).filter(r=>Yp(t,r)));function lm(e,t){return e.idt.id?1:0}var Hi=(e,t)=>t,Yi=(e,t,r)=>r,ZA=S(es,Hi,Yi,(e,t,r)=>e.filter(n=>n.orientation===t).filter(n=>n.mirror===r).sort(lm)),JA=S(ts,Hi,Yi,(e,t,r)=>e.filter(n=>n.orientation===t).filter(n=>n.mirror===r).sort(lm)),fm=(e,t)=>({width:e.width,height:t.height}),QA=(e,t)=>{var r=typeof t.width=="number"?t.width:ji;return{width:r,height:e.height}},dm=S(ce,Tt,fm),eS=(e,t,r)=>{switch(t){case"top":return e.top;case"bottom":return r-e.bottom;default:return 0}},tS=(e,t,r)=>{switch(t){case"left":return e.left;case"right":return r-e.right;default:return 0}},rS=S(Et,ce,ZA,Hi,Yi,(e,t,r,n,i)=>{var a={},o;return r.forEach(u=>{var c=fm(t,u);o==null&&(o=eS(t,n,e));var l=n==="top"&&!i||n==="bottom"&&i;a[u.id]=o-Number(l)*c.height,o+=(l?-1:1)*c.height}),a}),nS=S(St,ce,JA,Hi,Yi,(e,t,r,n,i)=>{var a={},o;return r.forEach(u=>{var c=QA(t,u);o==null&&(o=tS(t,n,e));var l=n==="left"&&!i||n==="right"&&i;a[u.id]=o-Number(l)*c.width,o+=(l?-1:1)*c.width}),a}),iS=(e,t)=>{var r=ce(e),n=Tt(e,t);if(n!=null){var i=rS(e,n.orientation,n.mirror),a=i[t];return a==null?{x:r.left,y:0}:{x:r.left,y:a}}},aS=(e,t)=>{var r=ce(e),n=zt(e,t);if(n!=null){var i=nS(e,n.orientation,n.mirror),a=i[t];return a==null?{x:0,y:r.top}:{x:a,y:r.top}}},hm=S(ce,zt,(e,t)=>{var r=typeof t.width=="number"?t.width:ji;return{width:r,height:e.height}}),id=(e,t,r)=>{switch(t){case"xAxis":return dm(e,r).width;case"yAxis":return hm(e,r).height;default:return}},vm=(e,t,r,n)=>{if(r!=null){var{allowDuplicatedCategory:i,type:a,dataKey:o}=r,u=Lt(e,n),c=t.map(l=>l.value);if(o&&u&&a==="category"&&i&&mh(c))return c}},Hs=S([F,Ui,Ce,fe],vm),pm=(e,t,r,n)=>{if(!(r==null||r.dataKey==null)){var{type:i,scale:a}=r,o=Lt(e,n);if(o&&(i==="number"||a!=="auto"))return t.map(u=>u.value)}},Ys=S([F,Ui,pn,fe],pm),ad=S([F,OA,yn,Dr,Hs,Ys,gn,Us,fe],(e,t,r,n,i,a,o,u,c)=>{if(t==null)return null;var l=Lt(e,c);return{angle:t.angle,interval:t.interval,minTickGap:t.minTickGap,orientation:t.orientation,tick:t.tick,tickCount:t.tickCount,tickFormatter:t.tickFormatter,ticks:t.ticks,type:t.type,unit:t.unit,axisType:c,categoricalDomain:a,duplicateDomain:i,isCategorical:l,niceTicks:u,range:o,realScaleType:r,scale:n}}),oS=(e,t,r,n,i,a,o,u,c)=>{if(!(t==null||n==null)){var l=Lt(e,c),{type:s,ticks:f,tickCount:d}=t,h=r==="scaleBand"&&typeof n.bandwidth=="function"?n.bandwidth()/2:2,p=s==="category"&&n.bandwidth?n.bandwidth()/h:0;p=c==="angleAxis"&&a!=null&&a.length>=2?je(a[0]-a[1])*2*p:p;var v=f||i;if(v){var m=v.map((g,b)=>{var x=o?o.indexOf(g):g;return{index:b,coordinate:n(x)+p,value:g,offset:p}});return m.filter(g=>!qe(g.coordinate))}return l&&u?u.map((g,b)=>({coordinate:n(g)+p,value:g,index:b,offset:p})):n.ticks?n.ticks(d).map(g=>({coordinate:n(g)+p,value:g,offset:p})):n.domain().map((g,b)=>({coordinate:n(g)+p,value:o?o[g]:g,index:b,offset:p}))}},mm=S([F,pn,yn,Dr,Us,gn,Hs,Ys,fe],oS),uS=(e,t,r,n,i,a,o)=>{if(!(t==null||r==null||n==null||n[0]===n[1])){var u=Lt(e,o),{tickCount:c}=t,l=0;return l=o==="angleAxis"&&(n==null?void 0:n.length)>=2?je(n[0]-n[1])*2*l:l,u&&a?a.map((s,f)=>({coordinate:r(s)+l,value:s,index:f,offset:l})):r.ticks?r.ticks(c).map(s=>({coordinate:r(s)+l,value:s,offset:l})):r.domain().map((s,f)=>({coordinate:r(s)+l,value:i?i[s]:s,index:f,offset:l}))}},Rt=S([F,pn,Dr,gn,Hs,Ys,fe],uS),ht=S(Ce,Dr,(e,t)=>{if(!(e==null||t==null))return si(si({},e),{},{scale:t})}),sS=S([Ce,yn,Ws,cm],Fs),CM=S((e,t,r)=>qs(e,r),sS,(e,t)=>{if(!(e==null||t==null))return si(si({},e),{},{scale:t})}),cS=S([F,es,ts],(e,t,r)=>{switch(e){case"horizontal":return t.some(n=>n.reversed)?"right-to-left":"left-to-right";case"vertical":return r.some(n=>n.reversed)?"bottom-to-top":"top-to-bottom";case"centric":case"radial":return"left-to-right";default:return}}),ym=e=>e.options.defaultTooltipEventType,gm=e=>e.options.validateTooltipEventTypes;function bm(e,t,r){if(e==null)return t;var n=e?"axis":"item";return r==null?t:r.includes(n)?n:t}function Gs(e,t){var r=ym(e),n=gm(e);return bm(t,r,n)}function lS(e){return D(t=>Gs(t,e))}var xm=(e,t)=>{var r,n=Number(t);if(!(qe(n)||t==null))return n>=0?e==null||(r=e[n])===null||r===void 0?void 0:r.value:void 0},fS=e=>e.tooltip.settings,Dt={active:!1,index:null,dataKey:void 0,coordinate:void 0},dS={itemInteraction:{click:Dt,hover:Dt},axisInteraction:{click:Dt,hover:Dt},keyboardInteraction:Dt,syncInteraction:{active:!1,index:null,dataKey:void 0,label:void 0,coordinate:void 0},tooltipItemPayloads:[],settings:{shared:void 0,trigger:"hover",axisId:0,active:!1,defaultIndex:void 0}},wm=tt({name:"tooltip",initialState:dS,reducers:{addTooltipEntrySettings(e,t){e.tooltipItemPayloads.push(t.payload)},removeTooltipEntrySettings(e,t){var r=gt(e).tooltipItemPayloads.indexOf(t.payload);r>-1&&e.tooltipItemPayloads.splice(r,1)},setTooltipSettingsState(e,t){e.settings=t.payload},setActiveMouseOverItemIndex(e,t){e.syncInteraction.active=!1,e.keyboardInteraction.active=!1,e.itemInteraction.hover.active=!0,e.itemInteraction.hover.index=t.payload.activeIndex,e.itemInteraction.hover.dataKey=t.payload.activeDataKey,e.itemInteraction.hover.coordinate=t.payload.activeCoordinate},mouseLeaveChart(e){e.itemInteraction.hover.active=!1,e.axisInteraction.hover.active=!1},mouseLeaveItem(e){e.itemInteraction.hover.active=!1},setActiveClickItemIndex(e,t){e.syncInteraction.active=!1,e.itemInteraction.click.active=!0,e.keyboardInteraction.active=!1,e.itemInteraction.click.index=t.payload.activeIndex,e.itemInteraction.click.dataKey=t.payload.activeDataKey,e.itemInteraction.click.coordinate=t.payload.activeCoordinate},setMouseOverAxisIndex(e,t){e.syncInteraction.active=!1,e.axisInteraction.hover.active=!0,e.keyboardInteraction.active=!1,e.axisInteraction.hover.index=t.payload.activeIndex,e.axisInteraction.hover.dataKey=t.payload.activeDataKey,e.axisInteraction.hover.coordinate=t.payload.activeCoordinate},setMouseClickAxisIndex(e,t){e.syncInteraction.active=!1,e.keyboardInteraction.active=!1,e.axisInteraction.click.active=!0,e.axisInteraction.click.index=t.payload.activeIndex,e.axisInteraction.click.dataKey=t.payload.activeDataKey,e.axisInteraction.click.coordinate=t.payload.activeCoordinate},setSyncInteraction(e,t){e.syncInteraction=t.payload},setKeyboardInteraction(e,t){e.keyboardInteraction.active=t.payload.active,e.keyboardInteraction.index=t.payload.activeIndex,e.keyboardInteraction.coordinate=t.payload.activeCoordinate,e.keyboardInteraction.dataKey=t.payload.activeDataKey}}}),{addTooltipEntrySettings:hS,removeTooltipEntrySettings:vS,setTooltipSettingsState:pS,setActiveMouseOverItemIndex:Om,mouseLeaveItem:mS,mouseLeaveChart:Pm,setActiveClickItemIndex:yS,setMouseOverAxisIndex:Am,setMouseClickAxisIndex:gS,setSyncInteraction:mu,setKeyboardInteraction:yu}=wm.actions,bS=wm.reducer;function od(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function kn(e){for(var t=1;t{if(t==null)return Dt;var i=PS(e,t,r);if(i==null)return Dt;if(i.active)return i;if(e.keyboardInteraction.active)return e.keyboardInteraction;if(e.syncInteraction.active&&e.syncInteraction.index!=null)return e.syncInteraction;var a=e.settings.active===!0;if(AS(i)){if(a)return kn(kn({},i),{},{active:!0})}else if(n!=null)return{active:!0,coordinate:void 0,dataKey:void 0,index:n};return kn(kn({},Dt),{},{coordinate:i.coordinate})},Vs=(e,t)=>{var r=e==null?void 0:e.index;if(r==null)return null;var n=Number(r);if(!Fe(n))return r;var i=0,a=1/0;return t.length>0&&(a=t.length-1),String(Math.max(i,Math.min(n,a)))},Em=(e,t,r,n,i,a,o,u)=>{if(!(a==null||u==null)){var c=o[0],l=c==null?void 0:u(c.positions,a);if(l!=null)return l;var s=i==null?void 0:i[Number(a)];if(s)switch(r){case"horizontal":return{x:s.coordinate,y:(n.top+t)/2};default:return{x:(n.left+e)/2,y:s.coordinate}}}},_m=(e,t,r,n)=>{if(t==="axis")return e.tooltipItemPayloads;if(e.tooltipItemPayloads.length===0)return[];var i;return r==="hover"?i=e.itemInteraction.hover.dataKey:i=e.itemInteraction.click.dataKey,i==null&&n!=null?[e.tooltipItemPayloads[0]]:e.tooltipItemPayloads.filter(a=>{var o;return((o=a.settings)===null||o===void 0?void 0:o.dataKey)===i})},bn=e=>e.options.tooltipPayloadSearcher,Nr=e=>e.tooltip;function ud(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function sd(e){for(var t=1;t{if(!(t==null||a==null)){var{chartData:u,computedData:c,dataStartIndex:l,dataEndIndex:s}=r,f=[];return e.reduce((d,h)=>{var p,{dataDefinedOnItem:v,settings:m}=h,g=TS(v,u),b=jS(g,l,s),x=(p=m==null?void 0:m.dataKey)!==null&&p!==void 0?p:n==null?void 0:n.dataKey,O=m==null?void 0:m.nameKey,w;if(n!=null&&n.dataKey&&Array.isArray(b)&&!Array.isArray(b[0])&&o==="axis"?w=yh(b,n.dataKey,i):w=a(b,t,c,O),Array.isArray(w))w.forEach(A=>{var E=sd(sd({},m),{},{name:A.name,unit:A.unit,color:void 0,fill:void 0});d.push(zl({tooltipEntrySettings:E,dataKey:A.dataKey,payload:A.payload,value:Pe(A.payload,A.dataKey),name:A.name}))});else{var P;d.push(zl({tooltipEntrySettings:m,dataKey:x,payload:w,value:Pe(w,x),name:(P=Pe(w,O))!==null&&P!==void 0?P:m==null?void 0:m.name}))}return d},f)}},de=e=>{var t=F(e);return t==="horizontal"?"xAxis":t==="vertical"?"yAxis":t==="centric"?"angleAxis":"radiusAxis"},$r=e=>e.tooltip.settings.axisId,ge=e=>{var t=de(e),r=$r(e);return pn(e,t,r)},Xs=S([ge,F,qp,Ds,de],am),CS=S([e=>e.graphicalItems.cartesianItems,e=>e.graphicalItems.polarItems],(e,t)=>[...e,...t]),kS=S([de,$r],zp),Gi=S([CS,ge,kS],Kp),MS=S([Gi],Fp),hr=S([MS,fr],Up),Zs=S([hr,ge,Gi],Hp),IS=S([ge],zs),DS=S([hr,Gi,Bi],Gp),NS=S([DS,fr,de],Vp),$S=S([Gi],Wp),RS=S([hr,ge,$S,de],Xp),LS=S([Zp,de,$r],Ir),BS=S([LS,de],em),qS=S([Jp,de,$r],Ir),zS=S([qS,de],tm),KS=S([Qp,de,$r],Ir),WS=S([KS,de],rm),FS=S([BS,WS,zS],Ks),US=S([ge,IS,NS,RS,FS],nm),Tm=S([ge,F,hr,Zs,Bi,de,US],im),HS=S([Tm,ge,Xs],om),YS=S([ge,Tm,HS,de],um),Cm=e=>{var t=de(e),r=$r(e),n=!1;return gn(e,t,r,n)},km=S([ge,Cm],qi),Mm=S([ge,Xs,YS,km],Fs),GS=S([F,Zs,ge,de],vm),VS=S([F,Zs,ge,de],pm),XS=(e,t,r,n,i,a,o,u)=>{if(t){var{type:c}=t,l=Lt(e,u);if(n){var s=r==="scaleBand"&&n.bandwidth?n.bandwidth()/2:2,f=c==="category"&&n.bandwidth?n.bandwidth()/s:0;return f=u==="angleAxis"&&i!=null&&(i==null?void 0:i.length)>=2?je(i[0]-i[1])*2*f:f,l&&o?o.map((d,h)=>({coordinate:n(d)+f,value:d,index:h,offset:f})):n.domain().map((d,h)=>({coordinate:n(d)+f,value:a?a[d]:d,index:h,offset:f}))}}},Ct=S([F,ge,Xs,Mm,Cm,GS,VS,de],XS),Js=S([ym,gm,fS],(e,t,r)=>bm(r.shared,e,t)),Im=e=>e.tooltip.settings.trigger,Qs=e=>e.tooltip.settings.defaultIndex,Vi=S([Nr,Js,Im,Qs],Sm),sr=S([Vi,hr],Vs),Dm=S([Ct,sr],xm),Nm=S([Vi],e=>{if(e)return e.dataKey}),$m=S([Nr,Js,Im,Qs],_m),ZS=S([St,Et,F,ce,Ct,Qs,$m,bn],Em),JS=S([Vi,ZS],(e,t)=>e!=null&&e.coordinate?e.coordinate:t),QS=S([Vi],e=>e.active),eE=S([$m,sr,fr,ge,Dm,bn,Js],jm),tE=S([eE],e=>{if(e!=null){var t=e.map(r=>r.payload).filter(r=>r!=null);return Array.from(new Set(t))}});function cd(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function ld(e){for(var t=1;tD(ge),oE=()=>{var e=aE(),t=D(Ct),r=D(Mm);return Jr(ld(ld({},e),{},{scale:r}),t)},uE=()=>D(Ds),ec=(e,t)=>t,Rm=(e,t,r)=>r,tc=(e,t,r,n)=>n,sE=S(Ct,e=>xi(e,t=>t.coordinate)),rc=S([Nr,ec,Rm,tc],Sm),Lm=S([rc,hr],Vs),cE=(e,t,r)=>{if(t!=null){var n=Nr(e);return t==="axis"?r==="hover"?n.axisInteraction.hover.dataKey:n.axisInteraction.click.dataKey:r==="hover"?n.itemInteraction.hover.dataKey:n.itemInteraction.click.dataKey}},Bm=S([Nr,ec,Rm,tc],_m),ci=S([St,Et,F,ce,Ct,tc,Bm,bn],Em),lE=S([rc,ci],(e,t)=>{var r;return(r=e.coordinate)!==null&&r!==void 0?r:t}),qm=S(Ct,Lm,xm),fE=S([Bm,Lm,fr,ge,qm,bn,ec],jm),dE=S([rc],e=>({isActive:e.active,activeIndex:e.index})),hE=(e,t,r,n,i,a,o,u)=>{if(!(!e||!t||!n||!i||!a)){var c=jx(e.chartX,e.chartY,t,r,u);if(c){var l=Cx(c,t),s=mx(l,o,a,n,i),f=Tx(t,a,s,c);return{activeIndex:String(s),activeCoordinate:f}}}};function gu(){return gu=Object.assign?Object.assign.bind():function(e){for(var t=1;ty.useContext(zm),go={exports:{}},dd;function xE(){return dd||(dd=1,function(e){var t=Object.prototype.hasOwnProperty,r="~";function n(){}Object.create&&(n.prototype=Object.create(null),new n().__proto__||(r=!1));function i(c,l,s){this.fn=c,this.context=l,this.once=s||!1}function a(c,l,s,f,d){if(typeof s!="function")throw new TypeError("The listener must be a function");var h=new i(s,f||c,d),p=r?r+l:l;return c._events[p]?c._events[p].fn?c._events[p]=[c._events[p],h]:c._events[p].push(h):(c._events[p]=h,c._eventsCount++),c}function o(c,l){--c._eventsCount===0?c._events=new n:delete c._events[l]}function u(){this._events=new n,this._eventsCount=0}u.prototype.eventNames=function(){var l=[],s,f;if(this._eventsCount===0)return l;for(f in s=this._events)t.call(s,f)&&l.push(r?f.slice(1):f);return Object.getOwnPropertySymbols?l.concat(Object.getOwnPropertySymbols(s)):l},u.prototype.listeners=function(l){var s=r?r+l:l,f=this._events[s];if(!f)return[];if(f.fn)return[f.fn];for(var d=0,h=f.length,p=new Array(h);d{e.eventEmitter==null&&(e.eventEmitter=Symbol("rechartsEventEmitter"))}}}),SE=Km.reducer,{createEventEmitter:EE}=Km.actions;function _E(e){return e.tooltip.syncInteraction}var jE={chartData:void 0,computedData:void 0,dataStartIndex:0,dataEndIndex:0},Wm=tt({name:"chartData",initialState:jE,reducers:{setChartData(e,t){if(e.chartData=t.payload,t.payload==null){e.dataStartIndex=0,e.dataEndIndex=0;return}t.payload.length>0&&e.dataEndIndex!==t.payload.length-1&&(e.dataEndIndex=t.payload.length-1)},setComputedData(e,t){e.computedData=t.payload},setDataStartEndIndexes(e,t){var{startIndex:r,endIndex:n}=t.payload;r!=null&&(e.dataStartIndex=r),n!=null&&(e.dataEndIndex=n)}}}),{setChartData:vd,setDataStartEndIndexes:TE,setComputedData:kM}=Wm.actions,CE=Wm.reducer,Fm=()=>{};function kE(){var e=D(Ns),t=D($s),r=re(),n=D(Np),i=D(Ct),a=ki(),o=rs(),u=D(c=>c.rootProps.className);y.useEffect(()=>{if(e==null)return Fm;var c=(l,s,f)=>{if(t!==f&&e===l){if(n==="index"){r(s);return}if(i!=null){var d;if(typeof n=="function"){var h={activeTooltipIndex:s.payload.index==null?void 0:Number(s.payload.index),isTooltipActive:s.payload.active,activeIndex:s.payload.index==null?void 0:Number(s.payload.index),activeLabel:s.payload.label,activeDataKey:s.payload.dataKey,activeCoordinate:s.payload.coordinate},p=n(i,h);d=i[p]}else n==="value"&&(d=i.find(P=>String(P.value)===s.payload.label));var{coordinate:v}=s.payload;if(d==null||s.payload.active===!1||v==null||o==null){r(mu({active:!1,coordinate:void 0,dataKey:void 0,index:null,label:void 0}));return}var{x:m,y:g}=v,b=Math.min(m,o.x+o.width),x=Math.min(g,o.y+o.height),O={x:a==="horizontal"?d.coordinate:b,y:a==="horizontal"?x:d.coordinate},w=mu({active:s.payload.active,coordinate:O,dataKey:s.payload.dataKey,index:String(d.index),label:s.payload.label});r(w)}}};return on.on(bu,c),()=>{on.off(bu,c)}},[u,r,t,e,n,i,a,o])}function ME(){var e=D(Ns),t=D($s),r=re();y.useEffect(()=>{if(e==null)return Fm;var n=(i,a,o)=>{t!==o&&e===i&&r(TE(a))};return on.on(hd,n),()=>{on.off(hd,n)}},[r,t,e])}function IE(){var e=re();y.useEffect(()=>{e(EE())},[e]),kE(),ME()}function DE(e,t,r,n,i,a){var o=D(d=>cE(d,e,t)),u=D($s),c=D(Ns),l=D(Np),s=D(_E),f=s==null?void 0:s.active;y.useEffect(()=>{if(!f&&c!=null&&u!=null){var d=mu({active:a,coordinate:r,dataKey:o,index:i,label:typeof n=="number"?String(n):n});on.emit(bu,c,d,u)}},[f,r,o,i,n,u,c,l,a])}function pd(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function md(e){for(var t=1;t{w(pS({shared:m,trigger:g,axisId:O,active:r,defaultIndex:P}))},[w,m,g,O,r,P]);var A=rs(),E=Tv(),j=lS(m),{activeIndex:N,isActive:T}=D(He=>dE(He,j,g,P)),C=D(He=>fE(He,j,g,P)),R=D(He=>qm(He,j,g,P)),L=D(He=>lE(He,j,g,P)),Y=C,te=bE(),B=r??T,[he,ne]=Vh([Y,B]),ke=j==="axis"?R:void 0;DE(j,g,L,ke,N,B);var Ue=x??te;if(Ue==null)return null;var $=Y??yd;B||($=yd),u&&$.length&&($=Fh(Y.filter(He=>He.value!=null&&(He.hide!==!0||t.includeHidden)),s,LE));var ve=$.length>0,Wt=y.createElement(vw,{allowEscapeViewBox:n,animationDuration:i,animationEasing:a,isAnimationActive:c,active:B,coordinate:L,hasPayload:ve,offset:l,position:f,reverseDirection:d,useTranslate3d:h,viewBox:A,wrapperStyle:p,lastBoundingBox:he,innerRef:ne,hasPortalFromProps:!!x},BE(o,md(md({},t),{},{payload:$,label:ke,active:B,coordinate:L,accessibilityLayer:E})));return y.createElement(y.Fragment,null,hh.createPortal(Wt,Ue),B&&y.createElement(gE,{cursor:v,tooltipEventType:j,coordinate:L,payload:Y,index:N}))}var bo={},xo={},gd;function zE(){return gd||(gd=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r,n=0,i={}){typeof i!="object"&&(i={});let a=null,o=null,u=null,c=0,l=null,s;const{leading:f=!1,trailing:d=!0,maxWait:h}=i,p="maxWait"in i,v=p?Math.max(Number(h)||0,n):0,m=A=>(a!==null&&(s=r.apply(o,a)),a=o=null,c=A,s),g=A=>(c=A,l=setTimeout(w,n),f&&a!==null?m(A):s),b=A=>(l=null,d&&a!==null?m(A):s),x=A=>{if(u===null)return!0;const E=A-u,j=E>=n||E<0,N=p&&A-c>=v;return j||N},O=A=>{const E=u===null?0:A-u,j=n-E,N=v-(A-c);return p?Math.min(j,N):j},w=()=>{const A=Date.now();if(x(A))return b(A);l=setTimeout(w,O(A))},P=function(...A){const E=Date.now(),j=x(E);if(a=A,o=this,u=E,j){if(l===null)return g(E);if(p)return clearTimeout(l),l=setTimeout(w,n),m(E)}return l===null&&(l=setTimeout(w,n)),s};return P.cancel=()=>{l!==null&&clearTimeout(l),c=0,u=a=o=l=null},P.flush=()=>l===null?s:b(Date.now()),P}e.debounce=t}(xo)),xo}var bd;function KE(){return bd||(bd=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=zE();function r(n,i=0,a={}){const{leading:o=!0,trailing:u=!0}=a;return t.debounce(n,i,{leading:o,maxWait:i,trailing:u})}e.throttle=r}(bo)),bo}var wo,xd;function WE(){return xd||(xd=1,wo=KE().throttle),wo}var FE=WE();const UE=At(FE);var Gr=function(t,r){for(var n=arguments.length,i=new Array(n>2?n-2:0),a=2;a{var{aspect:r,initialDimension:n={width:-1,height:-1},width:i="100%",height:a="100%",minWidth:o=0,minHeight:u,maxHeight:c,children:l,debounce:s=0,id:f,className:d,onResize:h,style:p={}}=e,v=y.useRef(null),m=y.useRef();m.current=h,y.useImperativeHandle(t,()=>v.current);var[g,b]=y.useState({containerWidth:n.width,containerHeight:n.height}),x=y.useCallback((w,P)=>{b(A=>{var E=Math.round(w),j=Math.round(P);return A.containerWidth===E&&A.containerHeight===j?A:{containerWidth:E,containerHeight:j}})},[]);y.useEffect(()=>{var w=j=>{var N,{width:T,height:C}=j[0].contentRect;x(T,C),(N=m.current)===null||N===void 0||N.call(m,T,C)};s>0&&(w=UE(w,s,{trailing:!0,leading:!1}));var P=new ResizeObserver(w),{width:A,height:E}=v.current.getBoundingClientRect();return x(A,E),P.observe(v.current),()=>{P.disconnect()}},[x,s]);var O=y.useMemo(()=>{var{containerWidth:w,containerHeight:P}=g;if(w<0||P<0)return null;Gr(Gt(i)||Gt(a),`The width(%s) and height(%s) are both fixed numbers, + maybe you don't need to use a ResponsiveContainer.`,i,a),Gr(!r||r>0,"The aspect(%s) must be greater than zero.",r);var A=Gt(i)?w:i,E=Gt(a)?P:a;return r&&r>0&&(A?E=A/r:E&&(A=E*r),c&&E>c&&(E=c)),Gr(A>0||E>0,`The width(%s) and height(%s) of chart should be greater than 0, + please check the style of container, or the props width(%s) and height(%s), + or add a minWidth(%s) or minHeight(%s) or use aspect(%s) to control the + height and width.`,A,E,i,a,o,u,r),y.Children.map(l,j=>y.cloneElement(j,{width:A,height:E,style:Oo({width:A,height:E},j.props.style)}))},[r,l,a,c,u,o,g,i]);return y.createElement("div",{id:f?"".concat(f):void 0,className:H("recharts-responsive-container",d),style:Oo(Oo({},p),{},{width:i,height:a,minWidth:o,minHeight:u,maxHeight:c}),ref:v},y.createElement("div",{style:{width:0,height:0,overflow:"visible"}},O))}),Um=e=>null;Um.displayName="Cell";function Od(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function xu(e){for(var t=1;t{t[r]||delete t[r]}),t}var Vr=function(t){var r=arguments.length>1&&arguments[1]!==void 0?arguments[1]:{};if(t==null||Tr.isSsr)return{width:0,height:0};var n=e_(r),i=JSON.stringify({text:t,copyStyle:n});if(pr.widthCache[i])return pr.widthCache[i];try{var a=document.getElementById(Pd);a||(a=document.createElement("span"),a.setAttribute("id",Pd),a.setAttribute("aria-hidden","true"),document.body.appendChild(a));var o=xu(xu({},QE),n);Object.assign(a.style,o),a.textContent="".concat(t);var u=a.getBoundingClientRect(),c={width:u.width,height:u.height};return pr.widthCache[i]=c,++pr.cacheCount>JE&&(pr.cacheCount=0,pr.widthCache={}),c}catch{return{width:0,height:0}}},Ad=/(-?\d+(?:\.\d+)?[a-zA-Z%]*)([*/])(-?\d+(?:\.\d+)?[a-zA-Z%]*)/,Sd=/(-?\d+(?:\.\d+)?[a-zA-Z%]*)([+-])(-?\d+(?:\.\d+)?[a-zA-Z%]*)/,t_=/^px|cm|vh|vw|em|rem|%|mm|in|pt|pc|ex|ch|vmin|vmax|Q$/,r_=/(-?\d+(?:\.\d+)?)([a-zA-Z%]+)?/,Hm={cm:96/2.54,mm:96/25.4,pt:96/72,pc:96/6,in:96,Q:96/(2.54*40),px:1},n_=Object.keys(Hm),mr="NaN";function i_(e,t){return e*Hm[t]}class _e{static parse(t){var r,[,n,i]=(r=r_.exec(t))!==null&&r!==void 0?r:[];return new _e(parseFloat(n),i??"")}constructor(t,r){this.num=t,this.unit=r,this.num=t,this.unit=r,qe(t)&&(this.unit=""),r!==""&&!t_.test(r)&&(this.num=NaN,this.unit=""),n_.includes(r)&&(this.num=i_(t,r),this.unit="px")}add(t){return this.unit!==t.unit?new _e(NaN,""):new _e(this.num+t.num,this.unit)}subtract(t){return this.unit!==t.unit?new _e(NaN,""):new _e(this.num-t.num,this.unit)}multiply(t){return this.unit!==""&&t.unit!==""&&this.unit!==t.unit?new _e(NaN,""):new _e(this.num*t.num,this.unit||t.unit)}divide(t){return this.unit!==""&&t.unit!==""&&this.unit!==t.unit?new _e(NaN,""):new _e(this.num/t.num,this.unit||t.unit)}toString(){return"".concat(this.num).concat(this.unit)}isNaN(){return qe(this.num)}}function Ym(e){if(e.includes(mr))return mr;for(var t=e;t.includes("*")||t.includes("/");){var r,[,n,i,a]=(r=Ad.exec(t))!==null&&r!==void 0?r:[],o=_e.parse(n??""),u=_e.parse(a??""),c=i==="*"?o.multiply(u):o.divide(u);if(c.isNaN())return mr;t=t.replace(Ad,c.toString())}for(;t.includes("+")||/.-\d+(?:\.\d+)?/.test(t);){var l,[,s,f,d]=(l=Sd.exec(t))!==null&&l!==void 0?l:[],h=_e.parse(s??""),p=_e.parse(d??""),v=f==="+"?h.add(p):h.subtract(p);if(v.isNaN())return mr;t=t.replace(Sd,v.toString())}return t}var Ed=/\(([^()]*)\)/;function a_(e){for(var t=e,r;(r=Ed.exec(t))!=null;){var[,n]=r;t=t.replace(Ed,Ym(n))}return t}function o_(e){var t=e.replace(/\s+/g,"");return t=a_(t),t=Ym(t),t}function u_(e){try{return o_(e)}catch{return mr}}function Po(e){var t=u_(e.slice(5,-1));return t===mr?"":t}var s_=["x","y","lineHeight","capHeight","scaleToFit","textAnchor","verticalAnchor","fill"],c_=["dx","dy","angle","className","breakAll"];function wu(){return wu=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var{children:t,breakAll:r,style:n}=e;try{var i=[];X(t)||(r?i=t.toString().split(""):i=t.toString().split(Gm));var a=i.map(u=>({word:u,width:Vr(u,n).width})),o=r?0:Vr(" ",n).width;return{wordsWithComputedWidth:a,spaceWidth:o}}catch{return null}},f_=(e,t,r,n,i)=>{var{maxLines:a,children:o,style:u,breakAll:c}=e,l=k(a),s=o,f=function(){var T=arguments.length>0&&arguments[0]!==void 0?arguments[0]:[];return T.reduce((C,R)=>{var{word:L,width:Y}=R,te=C[C.length-1];if(te&&(n==null||i||te.width+Y+rN.reduce((T,C)=>T.width>C.width?T:C);if(!l||i)return d;var p=d.length>a||h(d).width>Number(n);if(!p)return d;for(var v="…",m=N=>{var T=s.slice(0,N),C=Vm({breakAll:c,style:u,children:T+v}).wordsWithComputedWidth,R=f(C),L=R.length>a||h(R).width>Number(n);return[L,R]},g=0,b=s.length-1,x=0,O;g<=b&&x<=s.length-1;){var w=Math.floor((g+b)/2),P=w-1,[A,E]=m(P),[j]=m(w);if(!A&&!j&&(g=w+1),A&&j&&(b=w-1),!A&&j){O=E;break}x++}return O||d},jd=e=>{var t=X(e)?[]:e.toString().split(Gm);return[{words:t}]},d_=e=>{var{width:t,scaleToFit:r,children:n,style:i,breakAll:a,maxLines:o}=e;if((t||r)&&!Tr.isSsr){var u,c,l=Vm({breakAll:a,children:n,style:i});if(l){var{wordsWithComputedWidth:s,spaceWidth:f}=l;u=s,c=f}else return jd(n);return f_({breakAll:a,children:n,maxLines:o,style:i},u,c,t,r)}return jd(n)},Td="#808080",nc=y.forwardRef((e,t)=>{var{x:r=0,y:n=0,lineHeight:i="1em",capHeight:a="0.71em",scaleToFit:o=!1,textAnchor:u="start",verticalAnchor:c="end",fill:l=Td}=e,s=_d(e,s_),f=y.useMemo(()=>d_({breakAll:s.breakAll,children:s.children,maxLines:s.maxLines,scaleToFit:o,style:s.style,width:s.width}),[s.breakAll,s.children,s.maxLines,o,s.style,s.width]),{dx:d,dy:h,angle:p,className:v,breakAll:m}=s,g=_d(s,c_);if(!ft(r)||!ft(n))return null;var b=r+(k(d)?d:0),x=n+(k(h)?h:0),O;switch(c){case"start":O=Po("calc(".concat(a,")"));break;case"middle":O=Po("calc(".concat((f.length-1)/2," * -").concat(i," + (").concat(a," / 2))"));break;default:O=Po("calc(".concat(f.length-1," * -").concat(i,")"));break}var w=[];if(o){var P=f[0].width,{width:A}=s;w.push("scale(".concat(k(A)?A/P:1,")"))}return p&&w.push("rotate(".concat(p,", ").concat(b,", ").concat(x,")")),w.length&&(g.transform=w.join(" ")),y.createElement("text",wu({},W(g,!0),{ref:t,x:b,y:x,className:H("recharts-text",v),textAnchor:u,fill:l.includes("url")?Td:l}),f.map((E,j)=>{var N=E.words.join(m?"":" ");return y.createElement("tspan",{x:b,dy:j===0?O:i,key:"".concat(N,"-").concat(j)},N)}))});nc.displayName="Text";var h_=["offset"],v_=["labelRef"];function Cd(e,t){if(e==null)return{};var r,n,i=p_(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n{var{value:t,formatter:r}=e,n=X(e.children)?t:e.children;return typeof r=="function"?r(n):n},ic=e=>e!=null&&typeof e=="function",x_=(e,t)=>{var r=je(t-e),n=Math.min(Math.abs(t-e),360);return r*n},w_=(e,t,r)=>{var{position:n,viewBox:i,offset:a,className:o}=e,{cx:u,cy:c,innerRadius:l,outerRadius:s,startAngle:f,endAngle:d,clockWise:h}=i,p=(l+s)/2,v=x_(f,d),m=v>=0?1:-1,g,b;n==="insideStart"?(g=f+m*a,b=h):n==="insideEnd"?(g=d-m*a,b=!h):n==="end"&&(g=d+m*a,b=h),b=v<=0?b:!b;var x=ye(u,c,p,g),O=ye(u,c,p,g+(b?1:-1)*359),w="M".concat(x.x,",").concat(x.y,` + A`).concat(p,",").concat(p,",0,1,").concat(b?0:1,`, + `).concat(O.x,",").concat(O.y),P=X(e.id)?xr("recharts-radial-line-"):e.id;return y.createElement("text",pt({},r,{dominantBaseline:"central",className:H("recharts-radial-bar-label",o)}),y.createElement("defs",null,y.createElement("path",{id:P,d:w})),y.createElement("textPath",{xlinkHref:"#".concat(P)},t))},O_=e=>{var{viewBox:t,offset:r,position:n}=e,{cx:i,cy:a,innerRadius:o,outerRadius:u,startAngle:c,endAngle:l}=t,s=(c+l)/2;if(n==="outside"){var{x:f,y:d}=ye(i,a,u+r,s);return{x:f,y:d,textAnchor:f>=i?"start":"end",verticalAnchor:"middle"}}if(n==="center")return{x:i,y:a,textAnchor:"middle",verticalAnchor:"middle"};if(n==="centerTop")return{x:i,y:a,textAnchor:"middle",verticalAnchor:"start"};if(n==="centerBottom")return{x:i,y:a,textAnchor:"middle",verticalAnchor:"end"};var h=(o+u)/2,{x:p,y:v}=ye(i,a,h,s);return{x:p,y:v,textAnchor:"middle",verticalAnchor:"middle"}},P_=(e,t)=>{var{parentViewBox:r,offset:n,position:i}=e,{x:a,y:o,width:u,height:c}=t,l=c>=0?1:-1,s=l*n,f=l>0?"end":"start",d=l>0?"start":"end",h=u>=0?1:-1,p=h*n,v=h>0?"end":"start",m=h>0?"start":"end";if(i==="top"){var g={x:a+u/2,y:o-l*n,textAnchor:"middle",verticalAnchor:f};return ie(ie({},g),r?{height:Math.max(o-r.y,0),width:u}:{})}if(i==="bottom"){var b={x:a+u/2,y:o+c+s,textAnchor:"middle",verticalAnchor:d};return ie(ie({},b),r?{height:Math.max(r.y+r.height-(o+c),0),width:u}:{})}if(i==="left"){var x={x:a-p,y:o+c/2,textAnchor:v,verticalAnchor:"middle"};return ie(ie({},x),r?{width:Math.max(x.x-r.x,0),height:c}:{})}if(i==="right"){var O={x:a+u+p,y:o+c/2,textAnchor:m,verticalAnchor:"middle"};return ie(ie({},O),r?{width:Math.max(r.x+r.width-O.x,0),height:c}:{})}var w=r?{width:u,height:c}:{};return i==="insideLeft"?ie({x:a+p,y:o+c/2,textAnchor:m,verticalAnchor:"middle"},w):i==="insideRight"?ie({x:a+u-p,y:o+c/2,textAnchor:v,verticalAnchor:"middle"},w):i==="insideTop"?ie({x:a+u/2,y:o+s,textAnchor:"middle",verticalAnchor:d},w):i==="insideBottom"?ie({x:a+u/2,y:o+c-s,textAnchor:"middle",verticalAnchor:f},w):i==="insideTopLeft"?ie({x:a+p,y:o+s,textAnchor:m,verticalAnchor:d},w):i==="insideTopRight"?ie({x:a+u-p,y:o+s,textAnchor:v,verticalAnchor:d},w):i==="insideBottomLeft"?ie({x:a+p,y:o+c-s,textAnchor:m,verticalAnchor:f},w):i==="insideBottomRight"?ie({x:a+u-p,y:o+c-s,textAnchor:v,verticalAnchor:f},w):i&&typeof i=="object"&&(k(i.x)||Gt(i.x))&&(k(i.y)||Gt(i.y))?ie({x:a+ut(i.x,u),y:o+ut(i.y,c),textAnchor:"end",verticalAnchor:"end"},w):ie({x:a+u/2,y:o+c/2,textAnchor:"middle",verticalAnchor:"middle"},w)},A_=e=>"cx"in e&&k(e.cx);function Re(e){var{offset:t=5}=e,r=Cd(e,h_),n=ie({offset:t},r),{viewBox:i,position:a,value:o,children:u,content:c,className:l="",textBreakAll:s,labelRef:f}=n,d=rs(),h=i||d;if(!h||X(o)&&X(u)&&!y.isValidElement(c)&&typeof c!="function")return null;if(y.isValidElement(c)){var{labelRef:p}=n,v=Cd(n,v_);return y.cloneElement(c,v)}var m;if(typeof c=="function"){if(m=y.createElement(c,n),y.isValidElement(m))return m}else m=b_(n);var g=A_(h),b=W(n,!0);if(g&&(a==="insideStart"||a==="insideEnd"||a==="end"))return w_(n,m,b);var x=g?O_(n):P_(n,h);return y.createElement(nc,pt({ref:f,className:H("recharts-label",l)},b,x,{breakAll:s}),m)}Re.displayName="Label";var Xm=e=>{var{cx:t,cy:r,angle:n,startAngle:i,endAngle:a,r:o,radius:u,innerRadius:c,outerRadius:l,x:s,y:f,top:d,left:h,width:p,height:v,clockWise:m,labelViewBox:g}=e;if(g)return g;if(k(p)&&k(v)){if(k(s)&&k(f))return{x:s,y:f,width:p,height:v};if(k(d)&&k(h))return{x:d,y:h,width:p,height:v}}if(k(s)&&k(f))return{x:s,y:f,width:0,height:0};if(k(t)&&k(r))return{cx:t,cy:r,startAngle:i||n||0,endAngle:a||n||0,innerRadius:c||0,outerRadius:l||u||o||0,clockWise:m};if(e.viewBox)return e.viewBox},S_=(e,t,r)=>{if(!e)return null;var n={viewBox:t,labelRef:r};return e===!0?y.createElement(Re,pt({key:"label-implicit"},n)):ft(e)?y.createElement(Re,pt({key:"label-implicit",value:e},n)):y.isValidElement(e)?e.type===Re?y.cloneElement(e,ie({key:"label-implicit"},n)):y.createElement(Re,pt({key:"label-implicit",content:e},n)):ic(e)?y.createElement(Re,pt({key:"label-implicit",content:e},n)):e&&typeof e=="object"?y.createElement(Re,pt({},e,{key:"label-implicit"},n)):null},E_=function(t,r){var n=arguments.length>2&&arguments[2]!==void 0?arguments[2]:!0;if(!t||!t.children&&n&&!t.label)return null;var{children:i,labelRef:a}=t,o=Xm(t),u=Nu(i,Re).map((l,s)=>y.cloneElement(l,{viewBox:r||o,key:"label-".concat(s)}));if(!n)return u;var c=S_(t.label,r||o,a);return[c,...u]};Re.parseViewBox=Xm;Re.renderCallByParent=E_;var Ao={},So={},Md;function __(){return Md||(Md=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return r[r.length-1]}e.last=t}(So)),So}var Eo={},Id;function j_(){return Id||(Id=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){return Array.isArray(r)?r:Array.from(r)}e.toArray=t}(Eo)),Eo}var Dd;function T_(){return Dd||(Dd=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});const t=__(),r=j_(),n=Wu();function i(a){if(n.isArrayLike(a))return t.last(r.toArray(a))}e.last=i}(Ao)),Ao}var _o,Nd;function C_(){return Nd||(Nd=1,_o=T_().last),_o}var k_=C_();const M_=At(k_);var I_=["valueAccessor"],D_=["data","dataKey","clockWise","id","textBreakAll"];function li(){return li=Object.assign?Object.assign.bind():function(e){for(var t=1;tArray.isArray(e.value)?M_(e.value):e.value;function $t(e){var{valueAccessor:t=B_}=e,r=Ld(e,I_),{data:n,dataKey:i,clockWise:a,id:o,textBreakAll:u}=r,c=Ld(r,D_);return!n||!n.length?null:y.createElement(ze,{className:"recharts-label-list"},n.map((l,s)=>{var f=X(i)?t(l,s):Pe(l&&l.payload,i),d=X(o)?{}:{id:"".concat(o,"-").concat(s)};return y.createElement(Re,li({},W(l,!0),c,d,{parentViewBox:l.parentViewBox,value:f,textBreakAll:u,viewBox:Re.parseViewBox(X(a)?l:Rd(Rd({},l),{},{clockWise:a})),key:"label-".concat(s),index:s}))}))}$t.displayName="LabelList";function q_(e,t){return e?e===!0?y.createElement($t,{key:"labelList-implicit",data:t}):y.isValidElement(e)||ic(e)?y.createElement($t,{key:"labelList-implicit",data:t,content:e}):typeof e=="object"?y.createElement($t,li({data:t},e,{key:"labelList-implicit"})):null:null}function z_(e,t){var r=arguments.length>2&&arguments[2]!==void 0?arguments[2]:!0;if(!e||!e.children&&r&&!e.label)return null;var{children:n}=e,i=Nu(n,$t).map((o,u)=>y.cloneElement(o,{data:t,key:"labelList-".concat(u)}));if(!r)return i;var a=q_(e.label,t);return[a,...i]}$t.renderCallByParent=z_;function Ou(){return Ou=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var{cx:t,cy:r,r:n,className:i}=e,a=H("recharts-dot",i);return t===+t&&r===+r&&n===+n?y.createElement("circle",Ou({},W(e,!1),Du(e),{className:a,cx:t,cy:r,r:n})):null},K_={radiusAxis:{},angleAxis:{}},Jm=tt({name:"polarAxis",initialState:K_,reducers:{addRadiusAxis(e,t){e.radiusAxis[t.payload.id]=t.payload},removeRadiusAxis(e,t){delete e.radiusAxis[t.payload.id]},addAngleAxis(e,t){e.angleAxis[t.payload.id]=t.payload},removeAngleAxis(e,t){delete e.angleAxis[t.payload.id]}}}),{addRadiusAxis:DM,removeRadiusAxis:NM,addAngleAxis:$M,removeAngleAxis:RM}=Jm.actions,W_=Jm.reducer,F_={countOfBars:0,cartesianItems:[],polarItems:[]},Qm=tt({name:"graphicalItems",initialState:F_,reducers:{addBar(e){e.countOfBars+=1},removeBar(e){e.countOfBars-=1},addCartesianGraphicalItem(e,t){e.cartesianItems.push(t.payload)},replaceCartesianGraphicalItem(e,t){var{prev:r,next:n}=t.payload,i=gt(e).cartesianItems.indexOf(r);i>-1&&(e.cartesianItems[i]=n)},removeCartesianGraphicalItem(e,t){var r=gt(e).cartesianItems.indexOf(t.payload);r>-1&&e.cartesianItems.splice(r,1)},addPolarGraphicalItem(e,t){e.polarItems.push(t.payload)},removePolarGraphicalItem(e,t){var r=gt(e).polarItems.indexOf(t.payload);r>-1&&e.polarItems.splice(r,1)}}}),{addBar:U_,removeBar:H_,addCartesianGraphicalItem:Y_,replaceCartesianGraphicalItem:G_,removeCartesianGraphicalItem:V_,addPolarGraphicalItem:X_,removePolarGraphicalItem:Z_}=Qm.actions,J_=Qm.reducer;function Bd(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function qd(e){for(var t=1;t{var n=qd(qd({},e),{},{stackId:Ov(e.stackId)});r.current===null?t(Y_(n)):r.current!==n&&t(G_({prev:r.current,next:n})),r.current=n},[t,e]),y.useEffect(()=>()=>{r.current&&(t(V_(r.current)),r.current=null)},[t]),null}function LM(e){var t=re();return y.useEffect(()=>(t(X_(e)),()=>{t(Z_(e))}),[t,e]),null}var jo={},zd;function nj(){return zd||(zd=1,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:"Module"});function t(r){var i;if(typeof r!="object"||r==null)return!1;if(Object.getPrototypeOf(r)===null)return!0;if(Object.prototype.toString.call(r)!=="[object Object]"){const a=r[Symbol.toStringTag];return a==null||!((i=Object.getOwnPropertyDescriptor(r,Symbol.toStringTag))!=null&&i.writable)?!1:r.toString()===`[object ${a}]`}let n=r;for(;Object.getPrototypeOf(n)!==null;)n=Object.getPrototypeOf(n);return Object.getPrototypeOf(r)===n}e.isPlainObject=t}(jo)),jo}var To,Kd;function ij(){return Kd||(Kd=1,To=nj().isPlainObject),To}var aj=ij();const oj=At(aj);function fi(){return fi=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var a=r-n,o;return o="M ".concat(e,",").concat(t),o+="L ".concat(e+r,",").concat(t),o+="L ".concat(e+r-a/2,",").concat(t+i),o+="L ".concat(e+r-a/2-n,",").concat(t+i),o+="L ".concat(e,",").concat(t," Z"),o},uj={x:0,y:0,upperWidth:0,lowerWidth:0,height:0,isUpdateAnimationActive:!1,animationBegin:0,animationDuration:1500,animationEasing:"ease"},sj=e=>{var t=_t(e,uj),r=y.useRef(),[n,i]=y.useState(-1);y.useEffect(()=>{if(r.current&&r.current.getTotalLength)try{var m=r.current.getTotalLength();m&&i(m)}catch{}},[]);var{x:a,y:o,upperWidth:u,lowerWidth:c,height:l,className:s}=t,{animationEasing:f,animationDuration:d,animationBegin:h,isUpdateAnimationActive:p}=t;if(a!==+a||o!==+o||u!==+u||c!==+c||l!==+l||u===0&&c===0||l===0)return null;var v=H("recharts-trapezoid",s);return p?y.createElement(or,{canBegin:n>0,from:{upperWidth:0,lowerWidth:0,height:l,x:a,y:o},to:{upperWidth:u,lowerWidth:c,height:l,x:a,y:o},duration:d,animationEasing:f,isActive:p},m=>{var{upperWidth:g,lowerWidth:b,height:x,x:O,y:w}=m;return y.createElement(or,{canBegin:n>0,from:"0px ".concat(n===-1?1:n,"px"),to:"".concat(n,"px 0px"),attributeName:"strokeDasharray",begin:h,duration:d,easing:f},y.createElement("path",fi({},W(t,!0),{className:v,d:Wd(O,w,g,b,x),ref:r})))}):y.createElement("g",null,y.createElement("path",fi({},W(t,!0),{className:v,d:Wd(a,o,u,c,l)})))},cj=["option","shapeType","propTransformer","activeClassName","isActive"];function lj(e,t){if(e==null)return{};var r,n,i=fj(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n{var r=re();return(n,i)=>a=>{e==null||e(n,i,a),r(Om({activeIndex:String(i),activeDataKey:t,activeCoordinate:n.tooltipPosition}))}},ty=e=>{var t=re();return(r,n)=>i=>{e==null||e(r,n,i),t(mS())}},ry=(e,t)=>{var r=re();return(n,i)=>a=>{e==null||e(n,i,a),r(yS({activeIndex:String(i),activeDataKey:t,activeCoordinate:n.tooltipPosition}))}};function ny(e){var{fn:t,args:r}=e,n=re(),i=Se();return y.useEffect(()=>{if(!i){var a=t(r);return n(hS(a)),()=>{n(vS(a))}}},[t,r,n,i]),null}var iy=()=>{};function ay(e){var{legendPayload:t}=e,r=re(),n=Se();return y.useEffect(()=>n?iy:(r(_v(t)),()=>{r(jv(t))}),[r,n,t]),null}function BM(e){var{legendPayload:t}=e,r=re(),n=D(F);return y.useEffect(()=>n!=="centric"&&n!=="radial"?iy:(r(_v(t)),()=>{r(jv(t))}),[r,n,t]),null}function oy(e){var t=arguments.length>1&&arguments[1]!==void 0?arguments[1]:"animation-",r=y.useRef(xr(t)),n=y.useRef(e);return n.current!==e&&(r.current=xr(t),n.current=e),r.current}var bj=S([ce],e=>{if(e)return{top:e.top,bottom:e.bottom,left:e.left,right:e.right}}),xj=S([bj,St,Et],(e,t,r)=>{if(!(!e||t==null||r==null))return{x:e.left,y:e.top,width:Math.max(0,t-e.left-e.right),height:Math.max(0,r-e.top-e.bottom)}}),wj=e=>{var t=Se();return D(r=>ht(r,"xAxis",e,t))},Oj=e=>{var t=Se();return D(r=>ht(r,"yAxis",e,t))},ac=()=>D(xj),Pj=()=>D(tE);function Hd(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function Yd(e){for(var t=1;t{var{point:t,childIndex:r,mainColor:n,activeDot:i,dataKey:a}=e;if(i===!1||t.x==null||t.y==null)return null;var o=Yd(Yd({index:r,dataKey:a,cx:t.x,cy:t.y,r:4,fill:n??"none",strokeWidth:2,stroke:"#fff",payload:t.payload,value:t.value},W(i,!1)),Du(i)),u;return y.isValidElement(i)?u=y.cloneElement(i,o):typeof i=="function"?u=i(o):u=y.createElement(Zm,o),y.createElement(ze,{className:"recharts-active-dot"},u)};function jj(e){var{points:t,mainColor:r,activeDot:n,itemDataKey:i}=e,a=D(sr),o=Pj();if(t==null||o==null)return null;var u=t.find(c=>o.includes(c.payload));return X(u)?null:_j({point:u,childIndex:Number(a),mainColor:r,dataKey:i,activeDot:n})}var Tj=()=>{var e=re();return y.useEffect(()=>(e(U_()),()=>{e(H_())})),null},Cj=["children"];function kj(e,t){if(e==null)return{};var r,n,i=Mj(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n{},uy=y.createContext({addErrorBar:Gd,removeErrorBar:Gd}),Ij={data:[],xAxisId:"xAxis-0",yAxisId:"yAxis-0",dataPointFormatter:()=>({x:0,y:0,value:0}),errorBarOffset:0},sy=y.createContext(Ij);function cy(e){var{children:t}=e,r=kj(e,Cj);return y.createElement(sy.Provider,{value:r},t)}var Dj=()=>y.useContext(sy),ly=e=>{var{children:t,xAxisId:r,yAxisId:n,zAxisId:i,dataKey:a,data:o,stackId:u,hide:c,type:l,barSize:s}=e,[f,d]=y.useState([]),h=y.useCallback(m=>{d(g=>[...g,m])},[d]),p=y.useCallback(m=>{d(g=>g.filter(b=>b!==m))},[d]),v=Se();return y.createElement(uy.Provider,{value:{addErrorBar:h,removeErrorBar:p}},y.createElement(rj,{type:l,data:o,xAxisId:r,yAxisId:n,zAxisId:i,dataKey:a,errorBars:f,stackId:u,hide:c,barSize:s,isPanorama:v}),t)};function Nj(e){var{addErrorBar:t,removeErrorBar:r}=y.useContext(uy);return y.useEffect(()=>(t(e),()=>{r(e)}),[t,r,e]),null}var $j=["direction","width","dataKey","isAnimationActive","animationBegin","animationDuration","animationEasing"];function fy(e,t,r){return(t=Rj(t))in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function Rj(e){var t=Lj(e,"string");return typeof t=="symbol"?t:t+""}function Lj(e,t){if(typeof e!="object"||!e)return e;var r=e[Symbol.toPrimitive];if(r!==void 0){var n=r.call(e,t);if(typeof n!="object")return n;throw new TypeError("@@toPrimitive must return a primitive value.")}return(t==="string"?String:Number)(e)}function un(){return un=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var{x,y:O,value:w,errorVal:P}=f(b,n,t);if(!P)return null;var A=[],E,j;if(Array.isArray(P)?[E,j]=P:E=j=P,t==="x"){var{scale:N}=v,T=O+p,C=T+r,R=T-r,L=N(w-E),Y=N(w+j);A.push({x1:Y,y1:C,x2:Y,y2:R}),A.push({x1:L,y1:T,x2:Y,y2:T}),A.push({x1:L,y1:C,x2:L,y2:R})}else if(t==="y"){var{scale:te}=m,B=x+p,he=B-r,ne=B+r,ke=te(w-E),Ue=te(w+j);A.push({x1:he,y1:Ue,x2:ne,y2:Ue}),A.push({x1:B,y1:ke,x2:B,y2:Ue}),A.push({x1:he,y1:ke,x2:ne,y2:ke})}var $="".concat(x+p,"px ").concat(O+p,"px");return y.createElement(ze,un({className:"recharts-errorBar",key:"bar-".concat(A.map(ve=>"".concat(ve.x1,"-").concat(ve.x2,"-").concat(ve.y1,"-").concat(ve.y2)))},l),A.map(ve=>{var Wt=i?{transformOrigin:"".concat(ve.x1-5,"px")}:void 0;return y.createElement(or,{from:{transform:"scaleY(0)",transformOrigin:$},to:{transform:"scaleY(1)",transformOrigin:$},begin:a,easing:u,isActive:i,duration:o,key:"line-".concat(ve.x1,"-").concat(ve.x2,"-").concat(ve.y1,"-").concat(ve.y2),style:{transformOrigin:$}},y.createElement("line",un({},ve,{style:Wt})))}))});return y.createElement(ze,{className:"recharts-errorBars"},g)}var dy=y.createContext(void 0);function Kj(e){var t=y.useContext(dy);return e??t??"x"}function hy(e){var{direction:t,children:r}=e;return y.createElement(dy.Provider,{value:t},r)}var vy={stroke:"black",strokeWidth:1.5,width:5,offset:0,isAnimationActive:!0,animationBegin:0,animationDuration:400,animationEasing:"ease-in-out"};function Wj(e){var t=Kj(e.direction),{width:r,isAnimationActive:n,animationBegin:i,animationDuration:a,animationEasing:o}=_t(e,vy);return y.createElement(y.Fragment,null,y.createElement(Nj,{dataKey:e.dataKey,direction:t}),y.createElement(zj,un({},e,{direction:t,width:r,isAnimationActive:n,animationBegin:i,animationDuration:a,animationEasing:o})))}class py extends y.Component{render(){return y.createElement(Wj,this.props)}}fy(py,"defaultProps",vy);fy(py,"displayName","ErrorBar");var Fj="Invariant failed";function Uj(e,t){throw new Error(Fj)}var Hj=["x","y"];function Pu(){return Pu=Object.assign?Object.assign.bind():function(e){for(var t=1;t1&&arguments[1]!==void 0?arguments[1]:0;return(n,i)=>{if(k(t))return t;var a=k(n)||X(n);return a?t(n,i):(a||Uj(),r)}};function oc(e,t){var r,n,i=D(l=>Tt(l,e)),a=D(l=>zt(l,t)),o=(r=i==null?void 0:i.allowDataOverflow)!==null&&r!==void 0?r:De.allowDataOverflow,u=(n=a==null?void 0:a.allowDataOverflow)!==null&&n!==void 0?n:Ne.allowDataOverflow,c=o||u;return{needClip:c,needClipX:o,needClipY:u}}function yy(e){var{xAxisId:t,yAxisId:r,clipPathId:n}=e,i=ac(),{needClipX:a,needClipY:o,needClip:u}=oc(t,r);if(!u)return null;var{x:c,y:l,width:s,height:f}=i;return y.createElement("clipPath",{id:"clipPath-".concat(n)},y.createElement("rect",{x:a?c:c-s/2,y:o?l:l-f/2,width:a?s:s*2,height:o?f:f*2}))}var eT=["onMouseEnter","onMouseLeave","onClick"],tT=["value","background","tooltipPosition"],rT=["onMouseEnter","onClick","onMouseLeave"];function sn(){return sn=Object.assign?Object.assign.bind():function(e){for(var t=1;t{var{dataKey:t,name:r,fill:n,legendType:i,hide:a}=e;return[{inactive:a,dataKey:t,type:i,color:n,value:_i(r,t),payload:e}]};function uT(e){var{dataKey:t,stroke:r,strokeWidth:n,fill:i,name:a,hide:o,unit:u}=e;return{dataDefinedOnItem:void 0,positions:void 0,settings:{stroke:r,strokeWidth:n,fill:i,dataKey:t,nameKey:void 0,name:_i(a,t),hide:o,type:e.tooltipType,color:e.fill,unit:u}}}function sT(e){var t=D(sr),{data:r,dataKey:n,background:i,allOtherBarProps:a}=e,{onMouseEnter:o,onMouseLeave:u,onClick:c}=a,l=Au(a,eT),s=ey(o,n),f=ty(u),d=ry(c,n);if(!i||r==null)return null;var h=W(i,!1);return y.createElement(y.Fragment,null,r.map((p,v)=>{var{value:m,background:g,tooltipPosition:b}=p,x=Au(p,tT);if(!g)return null;var O=s(p,v),w=f(p,v),P=d(p,v),A=we(we(we(we(we({option:i,isActive:String(v)===t},x),{},{fill:"#eee"},g),h),mi(l,p,v)),{},{onMouseEnter:O,onMouseLeave:w,onClick:P,dataKey:n,index:v,className:"recharts-bar-background-rectangle"});return y.createElement(my,sn({key:"background-bar-".concat(v)},A))}))}function gy(e){var{data:t,props:r,showLabels:n}=e,i=W(r,!1),{shape:a,dataKey:o,activeBar:u}=r,c=D(sr),l=D(Nm),{onMouseEnter:s,onClick:f,onMouseLeave:d}=r,h=Au(r,rT),p=ey(s,o),v=ty(d),m=ry(f,o);return t?y.createElement(y.Fragment,null,t.map((g,b)=>{var x=u&&String(b)===c&&(l==null||o===l),O=x?u:a,w=we(we(we({},i),g),{},{isActive:x,option:O,index:b,dataKey:o});return y.createElement(ze,sn({className:"recharts-bar-rectangle"},mi(h,g,b),{onMouseEnter:p(g,b),onMouseLeave:v(g,b),onClick:m(g,b),key:"rectangle-".concat(g==null?void 0:g.x,"-").concat(g==null?void 0:g.y,"-").concat(g==null?void 0:g.value,"-").concat(b)}),y.createElement(my,w))}),n&&$t.renderCallByParent(r,t)):null}function cT(e){var{props:t,previousRectanglesRef:r}=e,{data:n,layout:i,isAnimationActive:a,animationBegin:o,animationDuration:u,animationEasing:c,onAnimationEnd:l,onAnimationStart:s}=t,f=r.current,d=oy(t,"recharts-bar-"),[h,p]=y.useState(!1),v=y.useCallback(()=>{typeof l=="function"&&l(),p(!1)},[l]),m=y.useCallback(()=>{typeof s=="function"&&s(),p(!0)},[s]);return y.createElement(or,{begin:o,duration:u,isActive:a,easing:c,from:{t:0},to:{t:1},onAnimationEnd:v,onAnimationStart:m,key:d},g=>{var{t:b}=g,x=b===1?n:n.map((O,w)=>{var P=f&&f[w];if(P){var A=nt(P.x,O.x),E=nt(P.y,O.y),j=nt(P.width,O.width),N=nt(P.height,O.height);return we(we({},O),{},{x:A(b),y:E(b),width:j(b),height:N(b)})}if(i==="horizontal"){var T=nt(0,O.height),C=T(b);return we(we({},O),{},{y:O.y+O.height-C,height:C})}var R=nt(0,O.width),L=R(b);return we(we({},O),{},{width:L})});return b>0&&(r.current=x),y.createElement(ze,null,y.createElement(gy,{props:t,data:x,showLabels:!h}))})}function lT(e){var{data:t,isAnimationActive:r}=e,n=y.useRef(null);return r&&t&&t.length&&(n.current==null||n.current!==t)?y.createElement(cT,{previousRectanglesRef:n,props:e}):y.createElement(gy,{props:e,data:t,showLabels:!0})}var by=0,fT=(e,t)=>{var r=Array.isArray(e.value)?e.value[1]:e.value;return{x:e.x,y:e.y,value:r,errorVal:Pe(e,t)}};class dT extends y.PureComponent{constructor(){super(...arguments),Xi(this,"id",xr("recharts-bar-"))}render(){var{hide:t,data:r,dataKey:n,className:i,xAxisId:a,yAxisId:o,needClip:u,background:c,id:l,layout:s}=this.props;if(t)return null;var f=H("recharts-bar",i),d=X(l)?this.id:l;return y.createElement(ze,{className:f},u&&y.createElement("defs",null,y.createElement(yy,{clipPathId:d,xAxisId:a,yAxisId:o})),y.createElement(ze,{className:"recharts-bar-rectangles",clipPath:u?"url(#clipPath-".concat(d,")"):null},y.createElement(sT,{data:r,dataKey:n,background:c,allOtherBarProps:this.props}),y.createElement(lT,this.props)),y.createElement(hy,{direction:s==="horizontal"?"y":"x"},this.props.children))}}var xy={activeBar:!1,animationBegin:0,animationDuration:400,animationEasing:"ease",hide:!1,isAnimationActive:!Tr.isSsr,legendType:"rect",minPointSize:by,xAxisId:0,yAxisId:0};function hT(e){var{xAxisId:t,yAxisId:r,hide:n,legendType:i,minPointSize:a,activeBar:o,animationBegin:u,animationDuration:c,animationEasing:l,isAnimationActive:s}=_t(e,xy),{needClip:f}=oc(t,r),d=ki(),h=Se(),p=y.useMemo(()=>({barSize:e.barSize,data:void 0,dataKey:e.dataKey,maxBarSize:e.maxBarSize,minPointSize:a,stackId:Ov(e.stackId)}),[e.barSize,e.dataKey,e.maxBarSize,a,e.stackId]),v=Nu(e.children,Um),m=D(x=>zT(x,t,r,h,p,v));if(d!=="vertical"&&d!=="horizontal")return null;var g,b=m==null?void 0:m[0];return b==null||b.height==null||b.width==null?g=0:g=d==="vertical"?b.height/2:b.width/2,y.createElement(cy,{xAxisId:t,yAxisId:r,data:m,dataPointFormatter:fT,errorBarOffset:g},y.createElement(dT,sn({},e,{layout:d,needClip:f,data:m,xAxisId:t,yAxisId:r,hide:n,legendType:i,minPointSize:a,activeBar:o,animationBegin:u,animationDuration:c,animationEasing:l,isAnimationActive:s})))}function vT(e){var{layout:t,barSettings:{dataKey:r,minPointSize:n},pos:i,bandSize:a,xAxis:o,yAxis:u,xAxisTicks:c,yAxisTicks:l,stackedData:s,displayedData:f,offset:d,cells:h}=e,p=t==="horizontal"?u:o,v=s?p.scale.domain():null,m=Ax({numericAxis:p});return f.map((g,b)=>{var x,O,w,P,A,E;s?x=bx(s[b],v):(x=Pe(g,r),Array.isArray(x)||(x=[m,x]));var j=Qj(n,by)(x[1],b);if(t==="horizontal"){var N,[T,C]=[u.scale(x[0]),u.scale(x[1])];O=Ll({axis:o,ticks:c,bandSize:a,offset:i.offset,entry:g,index:b}),w=(N=C??T)!==null&&N!==void 0?N:void 0,P=i.size;var R=T-C;if(A=qe(R)?0:R,E={x:O,y:d.top,width:P,height:d.height},Math.abs(j)>0&&Math.abs(A)0&&Math.abs(P)t,bT=(e,t,r)=>r,xT=(e,t,r,n)=>n,uc=(e,t,r,n,i)=>i,wT=(e,t,r,n,i)=>i.maxBarSize,OT=(e,t,r,n,i,a)=>a,Jd=(e,t,r)=>{var n=r??e;if(!X(n))return ut(n,t,0)},PT=S([F,Wi,gT,bT,xT],(e,t,r,n,i)=>t.filter(a=>e==="horizontal"?a.xAxisId===r:a.yAxisId===n).filter(a=>a.isPanorama===i).filter(a=>a.hide===!1).filter(a=>a.type==="bar")),AT=(e,t,r,n)=>{var i=F(e);return i==="horizontal"?pu(e,"yAxis",r,n):pu(e,"xAxis",t,n)},ST=(e,t,r)=>{var n=F(e);return n==="horizontal"?id(e,"xAxis",t):id(e,"yAxis",r)};function ET(e){return e.stackId!=null&&e.dataKey!=null}var _T=(e,t,r)=>{var n={},i=e.filter(ET),a=e.filter(l=>l.stackId==null),o=i.reduce((l,s)=>(l[s.stackId]||(l[s.stackId]=[]),l[s.stackId].push(s),l),n),u=Object.entries(o).map(l=>{var[s,f]=l,d=f.map(p=>p.dataKey),h=Jd(t,r,f[0].barSize);return{stackId:s,dataKeys:d,barSize:h}}),c=a.map(l=>{var s=[l.dataKey].filter(d=>d!=null),f=Jd(t,r,l.barSize);return{stackId:void 0,dataKeys:s,barSize:f}});return[...u,...c]},jT=S([PT,fA,ST],_T),TT=(e,t,r,n,i)=>{var a,o,u=F(e),c=Ip(e),{maxBarSize:l}=i,s=X(l)?c:l,f,d;return u==="horizontal"?(f=ht(e,"xAxis",t,n),d=Rt(e,"xAxis",t,n)):(f=ht(e,"yAxis",r,n),d=Rt(e,"yAxis",r,n)),(a=(o=Jr(f,d,!0))!==null&&o!==void 0?o:s)!==null&&a!==void 0?a:0},Oy=(e,t,r,n)=>{var i=F(e),a,o;return i==="horizontal"?(a=ht(e,"xAxis",t,n),o=Rt(e,"xAxis",t,n)):(a=ht(e,"yAxis",r,n),o=Rt(e,"yAxis",r,n)),Jr(a,o)};function CT(e,t,r,n,i){var a=n.length;if(!(a<1)){var o=ut(e,r,0,!0),u,c=[];if(Fe(n[0].barSize)){var l=!1,s=r/a,f=n.reduce((g,b)=>g+(b.barSize||0),0);f+=(a-1)*o,f>=r&&(f-=(a-1)*o,o=0),f>=r&&s>0&&(l=!0,s*=.9,f=a*s);var d=(r-f)/2>>0,h={offset:d-o,size:0};u=n.reduce((g,b)=>{var x,O={stackId:b.stackId,dataKeys:b.dataKeys,position:{offset:h.offset+h.size+o,size:l?s:(x=b.barSize)!==null&&x!==void 0?x:0}},w=[...g,O];return h=w[w.length-1].position,w},c)}else{var p=ut(t,r,0,!0);r-2*p-(a-1)*o<=0&&(o=0);var v=(r-2*p-(a-1)*o)/a;v>1&&(v>>=0);var m=Fe(i)?Math.min(v,i):v;u=n.reduce((g,b,x)=>[...g,{stackId:b.stackId,dataKeys:b.dataKeys,position:{offset:p+(v+o)*x+(v-m)/2,size:m}}],c)}return u}}var kT=(e,t,r,n,i,a,o)=>{var u=X(o)?t:o,c=CT(r,n,i!==a?i:a,e,u);return i!==a&&c!=null&&(c=c.map(l=>In(In({},l),{},{position:In(In({},l.position),{},{offset:l.position.offset-i/2})}))),c},MT=S([jT,Ip,lA,Dp,TT,Oy,wT],kT),IT=(e,t,r,n)=>ht(e,"xAxis",t,n),DT=(e,t,r,n)=>ht(e,"yAxis",r,n),NT=(e,t,r,n)=>Rt(e,"xAxis",t,n),$T=(e,t,r,n)=>Rt(e,"yAxis",r,n),RT=S([MT,uc],(e,t)=>{if(e!=null){var r=e.find(n=>n.stackId===t.stackId&&n.dataKeys.includes(t.dataKey));if(r!=null)return r.position}}),LT=(e,t)=>{if(!(!e||(t==null?void 0:t.dataKey)==null)){var{stackId:r}=t;if(r!=null){var n=e[r];if(n){var{stackedData:i}=n;if(i){var a=i.find(o=>o.key===t.dataKey);return a}}}}},BT=S([Wi,uc],(e,t)=>{if(e.some(r=>r.type==="bar"&&t.dataKey===r.dataKey&&t.stackId===r.stackId&&t.stackId===r.stackId))return t}),qT=S([AT,uc],LT),zT=S([ce,IT,DT,NT,$T,RT,F,ks,Oy,qT,BT,OT],(e,t,r,n,i,a,o,u,c,l,s,f)=>{var{chartData:d,dataStartIndex:h,dataEndIndex:p}=u;if(!(s==null||a==null||o!=="horizontal"&&o!=="vertical"||t==null||r==null||n==null||i==null||c==null)){var{data:v}=s,m;if(v!=null&&v.length>0?m=v:m=d==null?void 0:d.slice(h,p+1),m!=null)return vT({layout:o,barSettings:s,pos:a,bandSize:c,xAxis:t,yAxis:r,xAxisTicks:n,yAxisTicks:i,stackedData:l,displayedData:m,offset:e,cells:f})}}),KT=e=>{var{chartData:t}=e,r=re(),n=Se();return y.useEffect(()=>n?()=>{}:(r(vd(t)),()=>{r(vd(void 0))}),[t,r,n]),null},Qd={x:0,y:0,width:0,height:0,padding:{top:0,right:0,bottom:0,left:0}},Py=tt({name:"brush",initialState:Qd,reducers:{setBrushSettings(e,t){return t.payload==null?Qd:t.payload}}}),{setBrushSettings:qM}=Py.actions,WT=Py.reducer;function FT(e,t,r){return(t=UT(t))in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function UT(e){var t=HT(e,"string");return typeof t=="symbol"?t:t+""}function HT(e,t){if(typeof e!="object"||!e)return e;var r=e[Symbol.toPrimitive];if(r!==void 0){var n=r.call(e,t);if(typeof n!="object")return n;throw new TypeError("@@toPrimitive must return a primitive value.")}return(t==="string"?String:Number)(e)}class sc{static create(t){return new sc(t)}constructor(t){this.scale=t}get domain(){return this.scale.domain}get range(){return this.scale.range}get rangeMin(){return this.range()[0]}get rangeMax(){return this.range()[1]}get bandwidth(){return this.scale.bandwidth}apply(t){var{bandAware:r,position:n}=arguments.length>1&&arguments[1]!==void 0?arguments[1]:{};if(t!==void 0){if(n)switch(n){case"start":return this.scale(t);case"middle":{var i=this.bandwidth?this.bandwidth()/2:0;return this.scale(t)+i}case"end":{var a=this.bandwidth?this.bandwidth():0;return this.scale(t)+a}default:return this.scale(t)}if(r){var o=this.bandwidth?this.bandwidth()/2:0;return this.scale(t)+o}return this.scale(t)}}isInRange(t){var r=this.range(),n=r[0],i=r[r.length-1];return n<=i?t>=n&&t<=i:t>=i&&t<=n}}FT(sc,"EPS",1e-4);function YT(e){return(e%180+180)%180}var GT=function(t){var{width:r,height:n}=t,i=arguments.length>1&&arguments[1]!==void 0?arguments[1]:0,a=YT(i),o=a*Math.PI/180,u=Math.atan(n/r),c=o>u&&o{e.dots.push(t.payload)},removeDot:(e,t)=>{var r=gt(e).dots.findIndex(n=>n===t.payload);r!==-1&&e.dots.splice(r,1)},addArea:(e,t)=>{e.areas.push(t.payload)},removeArea:(e,t)=>{var r=gt(e).areas.findIndex(n=>n===t.payload);r!==-1&&e.areas.splice(r,1)},addLine:(e,t)=>{e.lines.push(t.payload)},removeLine:(e,t)=>{var r=gt(e).lines.findIndex(n=>n===t.payload);r!==-1&&e.lines.splice(r,1)}}}),{addDot:zM,removeDot:KM,addArea:WM,removeArea:FM,addLine:UM,removeLine:HM}=Ay.actions,XT=Ay.reducer,ZT=y.createContext(void 0),JT=e=>{var{children:t}=e,[r]=y.useState("".concat(xr("recharts"),"-clip")),n=ac();if(n==null)return null;var{x:i,y:a,width:o,height:u}=n;return y.createElement(ZT.Provider,{value:r},y.createElement("defs",null,y.createElement("clipPath",{id:r},y.createElement("rect",{x:i,y:a,height:u,width:o}))),t)};function Co(e,t){for(var r in e)if({}.hasOwnProperty.call(e,r)&&(!{}.hasOwnProperty.call(t,r)||e[r]!==t[r]))return!1;for(var n in t)if({}.hasOwnProperty.call(t,n)&&!{}.hasOwnProperty.call(e,n))return!1;return!0}function Sy(e,t,r){if(t<1)return[];if(t===1&&r===void 0)return e;for(var n=[],i=0;ie*i)return!1;var a=r();return e*(t-e*a/2-n)>=0&&e*(t+e*a/2-i)<=0}function tC(e,t){return Sy(e,t+1)}function rC(e,t,r,n,i){for(var a=(n||[]).slice(),{start:o,end:u}=t,c=0,l=1,s=o,f=function(){var p=n==null?void 0:n[c];if(p===void 0)return{v:Sy(n,l)};var v=c,m,g=()=>(m===void 0&&(m=r(p,v)),m),b=p.coordinate,x=c===0||hi(e,b,g,s,u);x||(c=0,s=o,l+=1),x&&(s=b+e*(g()/2+i),c+=l)},d;l<=a.length;)if(d=f(),d)return d.v;return[]}function eh(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function xe(e){for(var t=1;t(p===void 0&&(p=r(h,d)),p);if(d===o-1){var m=e*(h.coordinate+e*v()/2-c);a[d]=h=xe(xe({},h),{},{tickCoord:m>0?h.coordinate-m*e:h.coordinate})}else a[d]=h=xe(xe({},h),{},{tickCoord:h.coordinate});var g=hi(e,h.tickCoord,v,u,c);g&&(c=h.tickCoord-e*(v()/2+i),a[d]=xe(xe({},h),{},{isShow:!0}))},s=o-1;s>=0;s--)l(s);return a}function uC(e,t,r,n,i,a){var o=(n||[]).slice(),u=o.length,{start:c,end:l}=t;if(a){var s=n[u-1],f=r(s,u-1),d=e*(s.coordinate+e*f/2-l);o[u-1]=s=xe(xe({},s),{},{tickCoord:d>0?s.coordinate-d*e:s.coordinate});var h=hi(e,s.tickCoord,()=>f,c,l);h&&(l=s.tickCoord-e*(f/2+i),o[u-1]=xe(xe({},s),{},{isShow:!0}))}for(var p=a?u-1:u,v=function(b){var x=o[b],O,w=()=>(O===void 0&&(O=r(x,b)),O);if(b===0){var P=e*(x.coordinate-e*w()/2-c);o[b]=x=xe(xe({},x),{},{tickCoord:P<0?x.coordinate-P*e:x.coordinate})}else o[b]=x=xe(xe({},x),{},{tickCoord:x.coordinate});var A=hi(e,x.tickCoord,w,c,l);A&&(c=x.tickCoord+e*(w()/2+i),o[b]=xe(xe({},x),{},{isShow:!0}))},m=0;m{var w=typeof l=="function"?l(x.value,O):x.value;return p==="width"?QT(Vr(w,{fontSize:t,letterSpacing:r}),v,f):Vr(w,{fontSize:t,letterSpacing:r})[p]},g=i.length>=2?je(i[1].coordinate-i[0].coordinate):1,b=eC(a,g,p);return c==="equidistantPreserveStart"?rC(g,b,m,i,o):(c==="preserveStart"||c==="preserveStartEnd"?h=uC(g,b,m,i,o,c==="preserveStartEnd"):h=oC(g,b,m,i,o),h.filter(x=>x.isShow))}var sC=["viewBox"],cC=["viewBox"];function yr(){return yr=Object.assign?Object.assign.bind():function(e){for(var t=1;t2&&arguments[2]!==void 0?arguments[2]:[],{tickLine:i,stroke:a,tick:o,tickFormatter:u,unit:c}=this.props,l=cc(oe(oe({},this.props),{},{ticks:n}),t,r),s=this.getTickTextAnchor(),f=this.getTickVerticalAnchor(),d=W(this.props,!1),h=W(o,!1),p=oe(oe({},d),{},{fill:"none"},W(i,!1)),v=l.map((m,g)=>{var{line:b,tick:x}=this.getTickLineCoord(m),O=oe(oe(oe(oe({textAnchor:s,verticalAnchor:f},d),{},{stroke:"none",fill:a},h),x),{},{index:g,payload:m,visibleTicksCount:l.length,tickFormatter:u});return y.createElement(ze,yr({className:"recharts-cartesian-axis-tick",key:"tick-".concat(m.value,"-").concat(m.coordinate,"-").concat(m.tickCoord)},mi(this.props,m,g)),i&&y.createElement("line",yr({},p,b,{className:H("recharts-cartesian-axis-tick-line",nr(i,"className"))})),o&&Kt.renderTickItem(o,O,"".concat(typeof u=="function"?u(m.value,g):m.value).concat(c||"")))});return v.length>0?y.createElement("g",{className:"recharts-cartesian-axis-ticks"},v):null}render(){var{axisLine:t,width:r,height:n,className:i,hide:a}=this.props;if(a)return null;var{ticks:o}=this.props;return r!=null&&r<=0||n!=null&&n<=0?null:y.createElement(ze,{className:H("recharts-cartesian-axis",i),ref:u=>{if(u){var c=u.getElementsByClassName("recharts-cartesian-axis-tick-value");this.tickRefs.current=Array.from(c);var l=c[0];if(l){var s=window.getComputedStyle(l).fontSize,f=window.getComputedStyle(l).letterSpacing;(s!==this.state.fontSize||f!==this.state.letterSpacing)&&this.setState({fontSize:window.getComputedStyle(l).fontSize,letterSpacing:window.getComputedStyle(l).letterSpacing})}}}},t&&this.renderAxisLine(),this.renderTicks(this.state.fontSize,this.state.letterSpacing,o),Re.renderCallByParent(this.props))}}lc(Kt,"displayName","CartesianAxis");lc(Kt,"defaultProps",{x:0,y:0,width:0,height:0,viewBox:{x:0,y:0,width:0,height:0},orientation:"bottom",ticks:[],stroke:"#666",tickLine:!0,axisLine:!0,tick:!0,mirror:!1,minTickGap:5,tickSize:6,tickMargin:2,interval:"preserveEnd"});var hC=["x1","y1","x2","y2","key"],vC=["offset"],pC=["xAxisId","yAxisId"],mC=["xAxisId","yAxisId"];function nh(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function Oe(e){for(var t=1;t{var{fill:t}=e;if(!t||t==="none")return null;var{fillOpacity:r,x:n,y:i,width:a,height:o,ry:u}=e;return y.createElement("rect",{x:n,y:i,ry:u,width:a,height:o,stroke:"none",fill:t,fillOpacity:r,className:"recharts-cartesian-grid-bg"})};function Ey(e,t){var r;if(y.isValidElement(e))r=y.cloneElement(e,t);else if(typeof e=="function")r=e(t);else{var{x1:n,y1:i,x2:a,y2:o,key:u}=t,c=vi(t,hC),l=W(c,!1),{offset:s}=l,f=vi(l,vC);r=y.createElement("line",Jt({},f,{x1:n,y1:i,x2:a,y2:o,fill:"none",key:u}))}return r}function OC(e){var{x:t,width:r,horizontal:n=!0,horizontalPoints:i}=e;if(!n||!i||!i.length)return null;var{xAxisId:a,yAxisId:o}=e,u=vi(e,pC),c=i.map((l,s)=>{var f=Oe(Oe({},u),{},{x1:t,y1:l,x2:t+r,y2:l,key:"line-".concat(s),index:s});return Ey(n,f)});return y.createElement("g",{className:"recharts-cartesian-grid-horizontal"},c)}function PC(e){var{y:t,height:r,vertical:n=!0,verticalPoints:i}=e;if(!n||!i||!i.length)return null;var{xAxisId:a,yAxisId:o}=e,u=vi(e,mC),c=i.map((l,s)=>{var f=Oe(Oe({},u),{},{x1:l,y1:t,x2:l,y2:t+r,key:"line-".concat(s),index:s});return Ey(n,f)});return y.createElement("g",{className:"recharts-cartesian-grid-vertical"},c)}function AC(e){var{horizontalFill:t,fillOpacity:r,x:n,y:i,width:a,height:o,horizontalPoints:u,horizontal:c=!0}=e;if(!c||!t||!t.length)return null;var l=u.map(f=>Math.round(f+i-i)).sort((f,d)=>f-d);i!==l[0]&&l.unshift(0);var s=l.map((f,d)=>{var h=!l[d+1],p=h?i+o-f:l[d+1]-f;if(p<=0)return null;var v=d%t.length;return y.createElement("rect",{key:"react-".concat(d),y:f,x:n,height:p,width:a,stroke:"none",fill:t[v],fillOpacity:r,className:"recharts-cartesian-grid-bg"})});return y.createElement("g",{className:"recharts-cartesian-gridstripes-horizontal"},s)}function SC(e){var{vertical:t=!0,verticalFill:r,fillOpacity:n,x:i,y:a,width:o,height:u,verticalPoints:c}=e;if(!t||!r||!r.length)return null;var l=c.map(f=>Math.round(f+i-i)).sort((f,d)=>f-d);i!==l[0]&&l.unshift(0);var s=l.map((f,d)=>{var h=!l[d+1],p=h?i+o-f:l[d+1]-f;if(p<=0)return null;var v=d%r.length;return y.createElement("rect",{key:"react-".concat(d),x:f,y:a,width:p,height:u,stroke:"none",fill:r[v],fillOpacity:n,className:"recharts-cartesian-grid-bg"})});return y.createElement("g",{className:"recharts-cartesian-gridstripes-vertical"},s)}var EC=(e,t)=>{var{xAxis:r,width:n,height:i,offset:a}=e;return xv(cc(Oe(Oe(Oe({},Kt.defaultProps),r),{},{ticks:wv(r),viewBox:{x:0,y:0,width:n,height:i}})),a.left,a.left+a.width,t)},_C=(e,t)=>{var{yAxis:r,width:n,height:i,offset:a}=e;return xv(cc(Oe(Oe(Oe({},Kt.defaultProps),r),{},{ticks:wv(r),viewBox:{x:0,y:0,width:n,height:i}})),a.top,a.top+a.height,t)},jC={horizontal:!0,vertical:!0,horizontalPoints:[],verticalPoints:[],stroke:"#ccc",fill:"none",verticalFill:[],horizontalFill:[],xAxisId:0,yAxisId:0};function TC(e){var t=ns(),r=is(),n=Sv(),i=Oe(Oe({},_t(e,jC)),{},{x:k(e.x)?e.x:n.left,y:k(e.y)?e.y:n.top,width:k(e.width)?e.width:n.width,height:k(e.height)?e.height:n.height}),{xAxisId:a,yAxisId:o,x:u,y:c,width:l,height:s,syncWithTicks:f,horizontalValues:d,verticalValues:h}=i,p=Se(),v=D(j=>ad(j,"xAxis",a,p)),m=D(j=>ad(j,"yAxis",o,p));if(!k(l)||l<=0||!k(s)||s<=0||!k(u)||u!==+u||!k(c)||c!==+c)return null;var g=i.verticalCoordinatesGenerator||EC,b=i.horizontalCoordinatesGenerator||_C,{horizontalPoints:x,verticalPoints:O}=i;if((!x||!x.length)&&typeof b=="function"){var w=d&&d.length,P=b({yAxis:m?Oe(Oe({},m),{},{ticks:w?d:m.ticks}):void 0,width:t,height:r,offset:n},w?!0:f);Gr(Array.isArray(P),"horizontalCoordinatesGenerator should return Array but instead it returned [".concat(typeof P,"]")),Array.isArray(P)&&(x=P)}if((!O||!O.length)&&typeof g=="function"){var A=h&&h.length,E=g({xAxis:v?Oe(Oe({},v),{},{ticks:A?h:v.ticks}):void 0,width:t,height:r,offset:n},A?!0:f);Gr(Array.isArray(E),"verticalCoordinatesGenerator should return Array but instead it returned [".concat(typeof E,"]")),Array.isArray(E)&&(O=E)}return y.createElement("g",{className:"recharts-cartesian-grid"},y.createElement(wC,{fill:i.fill,fillOpacity:i.fillOpacity,x:i.x,y:i.y,width:i.width,height:i.height,ry:i.ry}),y.createElement(AC,Jt({},i,{horizontalPoints:x})),y.createElement(SC,Jt({},i,{verticalPoints:O})),y.createElement(OC,Jt({},i,{offset:n,horizontalPoints:x,xAxis:v,yAxis:m})),y.createElement(PC,Jt({},i,{offset:n,verticalPoints:O,xAxis:v,yAxis:m})))}TC.displayName="CartesianGrid";var _y=(e,t,r,n)=>ht(e,"xAxis",t,n),jy=(e,t,r,n)=>Rt(e,"xAxis",t,n),Ty=(e,t,r,n)=>ht(e,"yAxis",r,n),Cy=(e,t,r,n)=>Rt(e,"yAxis",r,n),CC=S([F,_y,Ty,jy,Cy],(e,t,r,n,i)=>Lt(e,"xAxis")?Jr(t,n,!1):Jr(r,i,!1)),kC=(e,t,r,n,i)=>i,MC=S([Wi,kC],(e,t)=>{if(e.some(r=>r.type==="line"&&t.dataKey===r.dataKey&&t.data===r.data))return t}),IC=S([F,_y,Ty,jy,Cy,MC,CC,ks],(e,t,r,n,i,a,o,u)=>{var{chartData:c,dataStartIndex:l,dataEndIndex:s}=u;if(!(a==null||t==null||r==null||n==null||i==null||n.length===0||i.length===0||o==null)){var{dataKey:f,data:d}=a,h;if(d!=null&&d.length>0?h=d:h=c==null?void 0:c.slice(l,s+1),h!=null)return JC({layout:e,xAxis:t,yAxis:r,xAxisTicks:n,yAxisTicks:i,dataKey:f,bandSize:o,displayedData:h})}}),DC=["type","layout","connectNulls","needClip"],NC=["activeDot","animateNewValues","animationBegin","animationDuration","animationEasing","connectNulls","dot","hide","isAnimationActive","label","legendType","xAxisId","yAxisId"];function ky(e,t){if(e==null)return{};var r,n,i=$C(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n{var{dataKey:t,name:r,stroke:n,legendType:i,hide:a}=e;return[{inactive:a,dataKey:t,type:i,color:n,value:_i(r,t),payload:e}]};function qC(e){var{dataKey:t,data:r,stroke:n,strokeWidth:i,fill:a,name:o,hide:u,unit:c}=e;return{dataDefinedOnItem:r,positions:void 0,settings:{stroke:n,strokeWidth:i,fill:a,dataKey:t,nameKey:void 0,name:_i(o,t),hide:u,type:e.tooltipType,color:e.stroke,unit:c}}}var My=(e,t)=>"".concat(t,"px ").concat(e-t,"px");function zC(e,t){for(var r=e.length%2!==0?[...e,0]:e,n=[],i=0;i{var n=r.reduce((f,d)=>f+d);if(!n)return My(t,e);for(var i=Math.floor(e/n),a=e%n,o=t-e,u=[],c=0,l=0;ca){u=[...r.slice(0,c),a-l];break}var s=u.length%2===0?[0,o]:[o];return[...zC(r,i),...u,...s].map(f=>"".concat(f,"px")).join(", ")};function WC(e,t){var r;if(y.isValidElement(e))r=y.cloneElement(e,t);else if(typeof e=="function")r=e(t);else{var n=H("recharts-line-dot",typeof e!="boolean"?e.className:"");r=y.createElement(Zm,jr({},t,{className:n}))}return r}function FC(e,t){return e==null?!1:t?!0:e.length===1}function UC(e){var{clipPathId:t,points:r,props:n}=e,{dot:i,dataKey:a,needClip:o}=n;if(!FC(r,i))return null;var u=bh(i),c=W(n,!1),l=W(i,!0),s=r.map((d,h)=>{var p=it(it(it({key:"dot-".concat(h),r:3},c),l),{},{index:h,cx:d.x,cy:d.y,dataKey:a,value:d.value,payload:d.payload,points:r});return WC(i,p)}),f={clipPath:o?"url(#clipPath-".concat(u?"":"dots-").concat(t,")"):null};return y.createElement(ze,jr({className:"recharts-line-dots",key:"dots"},f),s)}function Su(e){var{clipPathId:t,pathRef:r,points:n,strokeDasharray:i,props:a,showLabels:o}=e,{type:u,layout:c,connectNulls:l,needClip:s}=a,f=ky(a,DC),d=it(it({},W(f,!0)),{},{fill:"none",className:"recharts-line-curve",clipPath:s?"url(#clipPath-".concat(t,")"):null,points:n,type:u,layout:c,connectNulls:l,strokeDasharray:i??a.strokeDasharray});return y.createElement(y.Fragment,null,(n==null?void 0:n.length)>1&&y.createElement(Cv,jr({},d,{pathRef:r})),y.createElement(UC,{points:n,clipPathId:t,props:a}),o&&$t.renderCallByParent(a,n))}function HC(e){try{return e&&e.getTotalLength&&e.getTotalLength()||0}catch{return 0}}function YC(e){var{clipPathId:t,props:r,pathRef:n,previousPointsRef:i,longestAnimatedLengthRef:a}=e,{points:o,strokeDasharray:u,isAnimationActive:c,animationBegin:l,animationDuration:s,animationEasing:f,animateNewValues:d,width:h,height:p,onAnimationEnd:v,onAnimationStart:m}=r,g=i.current,b=oy(r,"recharts-line-"),[x,O]=y.useState(!1),w=y.useCallback(()=>{typeof v=="function"&&v(),O(!1)},[v]),P=y.useCallback(()=>{typeof m=="function"&&m(),O(!0)},[m]),A=HC(n.current),E=a.current;return y.createElement(or,{begin:l,duration:s,isActive:c,easing:f,from:{t:0},to:{t:1},onAnimationEnd:w,onAnimationStart:P,key:b},j=>{var{t:N}=j,T=nt(E,A+E),C=Math.min(T(N),A),R;if(u){var L="".concat(u).split(/[,\s]+/gim).map(B=>parseFloat(B));R=KC(C,A,L)}else R=My(A,C);if(g){var Y=g.length/o.length,te=N===1?o:o.map((B,he)=>{var ne=Math.floor(he*Y);if(g[ne]){var ke=g[ne],Ue=nt(ke.x,B.x),$=nt(ke.y,B.y);return it(it({},B),{},{x:Ue(N),y:$(N)})}if(d){var ve=nt(h*2,B.x),Wt=nt(p/2,B.y);return it(it({},B),{},{x:ve(N),y:Wt(N)})}return it(it({},B),{},{x:B.x,y:B.y})});return i.current=te,y.createElement(Su,{props:r,points:te,clipPathId:t,pathRef:n,showLabels:!x,strokeDasharray:R})}return N>0&&A>0&&(i.current=o,a.current=C),y.createElement(Su,{props:r,points:o,clipPathId:t,pathRef:n,showLabels:!x,strokeDasharray:R})})}function GC(e){var{clipPathId:t,props:r}=e,{points:n,isAnimationActive:i}=r,a=y.useRef(null),o=y.useRef(0),u=y.useRef(null),c=a.current;return i&&n&&n.length&&c!==n?y.createElement(YC,{props:r,clipPathId:t,previousPointsRef:a,longestAnimatedLengthRef:o,pathRef:u}):y.createElement(Su,{props:r,points:n,clipPathId:t,pathRef:u,showLabels:!0})}var VC=(e,t)=>({x:e.x,y:e.y,value:e.value,errorVal:Pe(e.payload,t)});class XC extends y.Component{constructor(){super(...arguments),Zi(this,"id",xr("recharts-line-"))}render(){var t,{hide:r,dot:n,points:i,className:a,xAxisId:o,yAxisId:u,top:c,left:l,width:s,height:f,id:d,needClip:h,layout:p}=this.props;if(r)return null;var v=H("recharts-line",a),m=X(d)?this.id:d,{r:g=3,strokeWidth:b=2}=(t=W(n,!1))!==null&&t!==void 0?t:{r:3,strokeWidth:2},x=bh(n),O=g*2+b;return y.createElement(y.Fragment,null,y.createElement(ze,{className:v},h&&y.createElement("defs",null,y.createElement(yy,{clipPathId:m,xAxisId:o,yAxisId:u}),!x&&y.createElement("clipPath",{id:"clipPath-dots-".concat(m)},y.createElement("rect",{x:l-O/2,y:c-O/2,width:s+O,height:f+O}))),y.createElement(GC,{props:this.props,clipPathId:m}),y.createElement(hy,{direction:p==="horizontal"?"y":"x"},y.createElement(cy,{xAxisId:o,yAxisId:u,data:i,dataPointFormatter:VC,errorBarOffset:0},this.props.children))),y.createElement(jj,{activeDot:this.props.activeDot,points:i,mainColor:this.props.stroke,itemDataKey:this.props.dataKey}))}}var Iy={activeDot:!0,animateNewValues:!0,animationBegin:0,animationDuration:1500,animationEasing:"ease",connectNulls:!1,dot:!0,fill:"#fff",hide:!1,isAnimationActive:!Tr.isSsr,label:!1,legendType:"line",stroke:"#3182bd",strokeWidth:1,xAxisId:0,yAxisId:0};function ZC(e){var t=_t(e,Iy),{activeDot:r,animateNewValues:n,animationBegin:i,animationDuration:a,animationEasing:o,connectNulls:u,dot:c,hide:l,isAnimationActive:s,label:f,legendType:d,xAxisId:h,yAxisId:p}=t,v=ky(t,NC),{needClip:m}=oc(h,p),{height:g,width:b,x,y:O}=ac(),w=ki(),P=Se(),A=y.useMemo(()=>({dataKey:e.dataKey,data:e.data}),[e.dataKey,e.data]),E=D(j=>IC(j,h,p,P,A));return w!=="horizontal"&&w!=="vertical"?null:y.createElement(XC,jr({},v,{connectNulls:u,dot:c,activeDot:r,animateNewValues:n,animationBegin:i,animationDuration:a,animationEasing:o,isAnimationActive:s,hide:l,label:f,legendType:d,xAxisId:h,yAxisId:p,points:E,layout:w,height:g,width:b,left:x,top:O,needClip:m}))}function JC(e){var{layout:t,xAxis:r,yAxis:n,xAxisTicks:i,yAxisTicks:a,dataKey:o,bandSize:u,displayedData:c}=e;return c.map((l,s)=>{var f=Pe(l,o);return t==="horizontal"?{x:Rl({axis:r,ticks:i,bandSize:u,entry:l,index:s}),y:X(f)?null:n.scale(f),value:f,payload:l}:{x:X(f)?null:r.scale(f),y:Rl({axis:n,ticks:a,bandSize:u,entry:l,index:s}),value:f,payload:l}})}class Dy extends y.PureComponent{render(){return y.createElement(ly,{type:"line",data:this.props.data,xAxisId:this.props.xAxisId,yAxisId:this.props.yAxisId,zAxisId:0,dataKey:this.props.dataKey,stackId:void 0,hide:this.props.hide,barSize:void 0},y.createElement(ay,{legendPayload:BC(this.props)}),y.createElement(ny,{fn:qC,args:this.props}),y.createElement(ZC,this.props))}}Zi(Dy,"displayName","Line");Zi(Dy,"defaultProps",Iy);function ah(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function oh(e){for(var t=1;t{var{children:a}=e,o=Ry(e,ck);return o},[e]),n=D(a=>Tt(a,r.id)),i=r===n;return y.useEffect(()=>(t(nk(r)),()=>{t(ik(r))}),[r,t]),i?e.children:null}var pk=e=>{var{xAxisId:t,className:r}=e,n=D(Av),i=Se(),a="xAxis",o=D(h=>Dr(h,a,t,i)),u=D(h=>mm(h,a,t,i)),c=D(h=>dm(h,t)),l=D(h=>iS(h,t));if(c==null||l==null)return null;var{dangerouslySetInnerHTML:s,ticks:f}=e,d=Ry(e,lk);return y.createElement(Kt,Eu({},d,{scale:o,x:l.x,y:l.y,width:c.width,height:c.height,className:H("recharts-".concat(a," ").concat(a),r),viewBox:n,ticks:u}))},mk=e=>{var t,r,n,i,a;return y.createElement(vk,{interval:(t=e.interval)!==null&&t!==void 0?t:"preserveEnd",id:e.xAxisId,scale:e.scale,type:e.type,padding:e.padding,allowDataOverflow:e.allowDataOverflow,domain:e.domain,dataKey:e.dataKey,allowDuplicatedCategory:e.allowDuplicatedCategory,allowDecimals:e.allowDecimals,tickCount:e.tickCount,includeHidden:(r=e.includeHidden)!==null&&r!==void 0?r:!1,reversed:e.reversed,ticks:e.ticks,height:e.height,orientation:e.orientation,mirror:e.mirror,hide:e.hide,unit:e.unit,name:e.name,angle:(n=e.angle)!==null&&n!==void 0?n:0,minTickGap:(i=e.minTickGap)!==null&&i!==void 0?i:5,tick:(a=e.tick)!==null&&a!==void 0?a:!0,tickFormatter:e.tickFormatter},y.createElement(pk,e))};class Ly extends y.Component{render(){return y.createElement(mk,this.props)}}$y(Ly,"displayName","XAxis");$y(Ly,"defaultProps",{allowDataOverflow:De.allowDataOverflow,allowDecimals:De.allowDecimals,allowDuplicatedCategory:De.allowDuplicatedCategory,height:De.height,hide:!1,mirror:De.mirror,orientation:De.orientation,padding:De.padding,reversed:De.reversed,scale:De.scale,tickCount:De.tickCount,type:De.type,xAxisId:0});var yk=e=>{var{ticks:t,label:r,labelGapWithTick:n=5,tickSize:i=0,tickMargin:a=0}=e,o=0;if(t){t.forEach(s=>{if(s){var f=s.getBoundingClientRect();f.width>o&&(o=f.width)}});var u=r?r.getBoundingClientRect().width:0,c=i+a,l=o+c+u+(r?n:0);return Math.round(l)}return 0},gk=["dangerouslySetInnerHTML","ticks"];function By(e,t,r){return(t=bk(t))in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function bk(e){var t=xk(e,"string");return typeof t=="symbol"?t:t+""}function xk(e,t){if(typeof e!="object"||!e)return e;var r=e[Symbol.toPrimitive];if(r!==void 0){var n=r.call(e,t);if(typeof n!="object")return n;throw new TypeError("@@toPrimitive must return a primitive value.")}return(t==="string"?String:Number)(e)}function _u(){return _u=Object.assign?Object.assign.bind():function(e){for(var t=1;t(t(ak(e)),()=>{t(ok(e))}),[e,t]),null}var Ak=e=>{var t,{yAxisId:r,className:n,width:i,label:a}=e,o=y.useRef(null),u=y.useRef(null),c=D(Av),l=Se(),s=re(),f="yAxis",d=D(x=>Dr(x,f,r,l)),h=D(x=>hm(x,r)),p=D(x=>aS(x,r)),v=D(x=>mm(x,f,r,l));if(y.useLayoutEffect(()=>{var x;if(!(i!=="auto"||!h||ic(a)||y.isValidElement(a))){var O=o.current,w=O==null||(x=O.tickRefs)===null||x===void 0?void 0:x.current,{tickSize:P,tickMargin:A}=O.props,E=yk({ticks:w,label:u.current,labelGapWithTick:5,tickSize:P,tickMargin:A});Math.round(h.width)!==Math.round(E)&&s(uk({id:r,width:E}))}},[o,o==null||(t=o.current)===null||t===void 0||(t=t.tickRefs)===null||t===void 0?void 0:t.current,h==null?void 0:h.width,h,s,a,r,i]),h==null||p==null)return null;var{dangerouslySetInnerHTML:m,ticks:g}=e,b=wk(e,gk);return y.createElement(Kt,_u({},b,{ref:o,labelRef:u,scale:d,x:p.x,y:p.y,width:h.width,height:h.height,className:H("recharts-".concat(f," ").concat(f),n),viewBox:c,ticks:v}))},Sk=e=>{var t,r,n,i,a;return y.createElement(y.Fragment,null,y.createElement(Pk,{interval:(t=e.interval)!==null&&t!==void 0?t:"preserveEnd",id:e.yAxisId,scale:e.scale,type:e.type,domain:e.domain,allowDataOverflow:e.allowDataOverflow,dataKey:e.dataKey,allowDuplicatedCategory:e.allowDuplicatedCategory,allowDecimals:e.allowDecimals,tickCount:e.tickCount,padding:e.padding,includeHidden:(r=e.includeHidden)!==null&&r!==void 0?r:!1,reversed:e.reversed,ticks:e.ticks,width:e.width,orientation:e.orientation,mirror:e.mirror,hide:e.hide,unit:e.unit,name:e.name,angle:(n=e.angle)!==null&&n!==void 0?n:0,minTickGap:(i=e.minTickGap)!==null&&i!==void 0?i:5,tick:(a=e.tick)!==null&&a!==void 0?a:!0,tickFormatter:e.tickFormatter}),y.createElement(Ak,e))},Ek={allowDataOverflow:Ne.allowDataOverflow,allowDecimals:Ne.allowDecimals,allowDuplicatedCategory:Ne.allowDuplicatedCategory,hide:!1,mirror:Ne.mirror,orientation:Ne.orientation,padding:Ne.padding,reversed:Ne.reversed,scale:Ne.scale,tickCount:Ne.tickCount,type:Ne.type,width:Ne.width,yAxisId:0};class qy extends y.Component{render(){return y.createElement(Sk,this.props)}}By(qy,"displayName","YAxis");By(qy,"defaultProps",Ek);var ko={exports:{}},Mo={};/** + * @license React + * use-sync-external-store-with-selector.production.js + * + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */var uh;function _k(){if(uh)return Mo;uh=1;var e=Cu();function t(c,l){return c===l&&(c!==0||1/c===1/l)||c!==c&&l!==l}var r=typeof Object.is=="function"?Object.is:t,n=e.useSyncExternalStore,i=e.useRef,a=e.useEffect,o=e.useMemo,u=e.useDebugValue;return Mo.useSyncExternalStoreWithSelector=function(c,l,s,f,d){var h=i(null);if(h.current===null){var p={hasValue:!1,value:null};h.current=p}else p=h.current;h=o(function(){function m(w){if(!g){if(g=!0,b=w,w=f(w),d!==void 0&&p.hasValue){var P=p.value;if(d(P,w))return x=P}return x=w}if(P=x,r(b,w))return P;var A=f(w);return d!==void 0&&d(P,A)?(b=w,P):(b=w,x=A)}var g=!1,b,x,O=s===void 0?null:s;return[function(){return m(l())},O===null?void 0:function(){return m(O())}]},[l,s,f,d]);var v=n(c,h[0],h[1]);return a(function(){p.hasValue=!0,p.value=v},[v]),u(v),v},Mo}var sh;function jk(){return sh||(sh=1,ko.exports=_k()),ko.exports}jk();function Tk(e){e()}function Ck(){let e=null,t=null;return{clear(){e=null,t=null},notify(){Tk(()=>{let r=e;for(;r;)r.callback(),r=r.next})},get(){const r=[];let n=e;for(;n;)r.push(n),n=n.next;return r},subscribe(r){let n=!0;const i=t={callback:r,next:null,prev:t};return i.prev?i.prev.next=i:e=i,function(){!n||e===null||(n=!1,i.next?i.next.prev=i.prev:t=i.prev,i.prev?i.prev.next=i.next:e=i.next)}}}}var ch={notify(){},get:()=>[]};function kk(e,t){let r,n=ch,i=0,a=!1;function o(v){s();const m=n.subscribe(v);let g=!1;return()=>{g||(g=!0,m(),f())}}function u(){n.notify()}function c(){p.onStateChange&&p.onStateChange()}function l(){return a}function s(){i++,r||(r=e.subscribe(c),n=Ck())}function f(){i--,r&&i===0&&(r(),r=void 0,n.clear(),n=ch)}function d(){a||(a=!0,s())}function h(){a&&(a=!1,f())}const p={addNestedSub:o,notifyNestedSubs:u,handleChangeWrapper:c,isSubscribed:l,trySubscribe:d,tryUnsubscribe:h,getListeners:()=>n};return p}var Mk=()=>typeof window<"u"&&typeof window.document<"u"&&typeof window.document.createElement<"u",Ik=Mk(),Dk=()=>typeof navigator<"u"&&navigator.product==="ReactNative",Nk=Dk(),$k=()=>Ik||Nk?y.useLayoutEffect:y.useEffect,Rk=$k(),Io=Symbol.for("react-redux-context"),Do=typeof globalThis<"u"?globalThis:{};function Lk(){if(!y.createContext)return{};const e=Do[Io]??(Do[Io]=new Map);let t=e.get(y.createContext);return t||(t=y.createContext(null),e.set(y.createContext,t)),t}var Bk=Lk();function qk(e){const{children:t,context:r,serverState:n,store:i}=e,a=y.useMemo(()=>{const c=kk(i);return{store:i,subscription:c,getServerState:n?()=>n:void 0}},[i,n]),o=y.useMemo(()=>i.getState(),[i]);Rk(()=>{const{subscription:c}=a;return c.onStateChange=c.notifyNestedSubs,c.trySubscribe(),o!==i.getState()&&c.notifyNestedSubs(),()=>{c.tryUnsubscribe(),c.onStateChange=void 0}},[a,o]);const u=r||Bk;return y.createElement(u.Provider,{value:a},t)}var zk=qk,Kk=(e,t)=>t,fc=S([Kk,F,yA,de,km,Ct,sE,ce],hE),dc=e=>{var t=e.currentTarget.getBoundingClientRect(),r=t.width/e.currentTarget.offsetWidth,n=t.height/e.currentTarget.offsetHeight;return{chartX:Math.round((e.clientX-t.left)/r),chartY:Math.round((e.clientY-t.top)/n)}},zy=Qe("mouseClick"),Ky=ln();Ky.startListening({actionCreator:zy,effect:(e,t)=>{var r=e.payload,n=fc(t.getState(),dc(r));(n==null?void 0:n.activeIndex)!=null&&t.dispatch(gS({activeIndex:n.activeIndex,activeDataKey:void 0,activeCoordinate:n.activeCoordinate}))}});var ju=Qe("mouseMove"),Wy=ln();Wy.startListening({actionCreator:ju,effect:(e,t)=>{var r=e.payload,n=t.getState(),i=Gs(n,n.tooltip.settings.shared),a=fc(n,dc(r));i==="axis"&&((a==null?void 0:a.activeIndex)!=null?t.dispatch(Am({activeIndex:a.activeIndex,activeDataKey:void 0,activeCoordinate:a.activeCoordinate})):t.dispatch(Pm()))}});function Wk(e,t){return t instanceof HTMLElement?"HTMLElement <".concat(t.tagName,' class="').concat(t.className,'">'):t===window?"global.window":t}var lh={accessibilityLayer:!0,barCategoryGap:"10%",barGap:4,barSize:void 0,className:void 0,maxBarSize:void 0,stackOffset:"none",syncId:void 0,syncMethod:"index"},Fy=tt({name:"rootProps",initialState:lh,reducers:{updateOptions:(e,t)=>{var r;e.accessibilityLayer=t.payload.accessibilityLayer,e.barCategoryGap=t.payload.barCategoryGap,e.barGap=(r=t.payload.barGap)!==null&&r!==void 0?r:lh.barGap,e.barSize=t.payload.barSize,e.maxBarSize=t.payload.maxBarSize,e.stackOffset=t.payload.stackOffset,e.syncId=t.payload.syncId,e.syncMethod=t.payload.syncMethod,e.className=t.payload.className}}}),Fk=Fy.reducer,{updateOptions:Uk}=Fy.actions,Uy=tt({name:"polarOptions",initialState:null,reducers:{updatePolarOptions:(e,t)=>t.payload}}),{updatePolarOptions:VM}=Uy.actions,Hk=Uy.reducer,Hy=Qe("keyDown"),Yy=Qe("focus"),hc=ln();hc.startListening({actionCreator:Hy,effect:(e,t)=>{var r=t.getState(),n=r.rootProps.accessibilityLayer!==!1;if(n){var{keyboardInteraction:i}=r.tooltip,a=e.payload;if(!(a!=="ArrowRight"&&a!=="ArrowLeft"&&a!=="Enter")){var o=Number(Vs(i,hr(r))),u=Ct(r);if(a==="Enter"){var c=ci(r,"axis","hover",String(i.index));t.dispatch(yu({active:!i.active,activeIndex:i.index,activeDataKey:i.dataKey,activeCoordinate:c}));return}var l=cS(r),s=l==="left-to-right"?1:-1,f=a==="ArrowRight"?1:-1,d=o+f*s;if(!(u==null||d>=u.length||d<0)){var h=ci(r,"axis","hover",String(d));t.dispatch(yu({active:!0,activeIndex:d.toString(),activeDataKey:void 0,activeCoordinate:h}))}}}}});hc.startListening({actionCreator:Yy,effect:(e,t)=>{var r=t.getState(),n=r.rootProps.accessibilityLayer!==!1;if(n){var{keyboardInteraction:i}=r.tooltip;if(!i.active&&i.index==null){var a="0",o=ci(r,"axis","hover",String(a));t.dispatch(yu({activeDataKey:void 0,active:!0,activeIndex:a,activeCoordinate:o}))}}}});var Xe=Qe("externalEvent"),Gy=ln();Gy.startListening({actionCreator:Xe,effect:(e,t)=>{if(e.payload.handler!=null){var r=t.getState(),n={activeCoordinate:JS(r),activeDataKey:Nm(r),activeIndex:sr(r),activeLabel:Dm(r),activeTooltipIndex:sr(r),isTooltipActive:QS(r)};e.payload.handler(n,e.payload.reactEvent)}}});var Yk=S([Nr],e=>e.tooltipItemPayloads),Gk=S([Yk,bn,(e,t,r)=>t,(e,t,r)=>r],(e,t,r,n)=>{var i=e.find(u=>u.settings.dataKey===n);if(i!=null){var{positions:a}=i;if(a!=null){var o=t(a,r);return o}}}),Vy=Qe("touchMove"),Xy=ln();Xy.startListening({actionCreator:Vy,effect:(e,t)=>{var r=e.payload,n=t.getState(),i=Gs(n,n.tooltip.settings.shared);if(i==="axis"){var a=fc(n,dc({clientX:r.touches[0].clientX,clientY:r.touches[0].clientY,currentTarget:r.currentTarget}));(a==null?void 0:a.activeIndex)!=null&&t.dispatch(Am({activeIndex:a.activeIndex,activeDataKey:void 0,activeCoordinate:a.activeCoordinate}))}else if(i==="item"){var o,u=r.touches[0],c=document.elementFromPoint(u.clientX,u.clientY);if(!c||!c.getAttribute)return;var l=c.getAttribute(Mx),s=(o=c.getAttribute(Ix))!==null&&o!==void 0?o:void 0,f=Gk(t.getState(),l,s);t.dispatch(Om({activeDataKey:s,activeIndex:l,activeCoordinate:f}))}}});var Vk=Zh({brush:WT,cartesianAxis:sk,chartData:CE,graphicalItems:J_,layout:rx,legend:Ux,options:SE,polarAxis:W_,polarOptions:Hk,referenceElements:XT,rootProps:Fk,tooltip:bS}),Xk=function(t){var r=arguments.length>1&&arguments[1]!==void 0?arguments[1]:"Chart";return _b({reducer:Vk,preloadedState:t,middleware:n=>n({serializableCheck:!1}).concat([Ky.middleware,Wy.middleware,hc.middleware,Gy.middleware,Xy.middleware]),devTools:{serialize:{replacer:Wk},name:"recharts-".concat(r)}})};function Zk(e){var{preloadedState:t,children:r,reduxStoreName:n}=e,i=Se(),a=y.useRef(null);if(i)return r;a.current==null&&(a.current=Xk(t,n));var o=Gu;return y.createElement(zk,{context:o,store:a.current},r)}function Jk(e){var{layout:t,width:r,height:n,margin:i}=e,a=re(),o=Se();return y.useEffect(()=>{o||(a(Qb(t)),a(ex({width:r,height:n})),a(Jb(i)))},[a,o,t,r,n,i]),null}function Qk(e){var t=re();return y.useEffect(()=>{t(Uk(e))},[t,e]),null}var eM=["children"];function tM(e,t){if(e==null)return{};var r,n,i=rM(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n{var r=ns(),n=is(),i=Tv();if(!Gn(r)||!Gn(n))return null;var{children:a,otherAttributes:o,title:u,desc:c}=e,l,s;return typeof o.tabIndex=="number"?l=o.tabIndex:l=i?0:void 0,typeof o.role=="string"?s=o.role:s=i?"application":void 0,y.createElement($u,pi({},o,{title:u,desc:c,role:s,tabIndex:l,width:r,height:n,style:nM,ref:t}),a)}),aM=e=>{var{children:t}=e,r=D(Ci);if(!r)return null;var{width:n,height:i,y:a,x:o}=r;return y.createElement($u,{width:n,height:i,x:o,y:a},t)},fh=y.forwardRef((e,t)=>{var{children:r}=e,n=tM(e,eM),i=Se();return i?y.createElement(aM,null,r):y.createElement(iM,pi({ref:t},n),r)});function oM(){var e=re(),[t,r]=y.useState(null),n=D(kx);return y.useEffect(()=>{if(t!=null){var i=t.getBoundingClientRect(),a=i.width/t.offsetWidth;Fe(a)&&a!==n&&e(tx(a))}},[t,e,n]),r}function dh(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter(function(i){return Object.getOwnPropertyDescriptor(e,i).enumerable})),r.push.apply(r,n)}return r}function uM(e){for(var t=1;t{var{children:r,className:n,height:i,onClick:a,onContextMenu:o,onDoubleClick:u,onMouseDown:c,onMouseEnter:l,onMouseLeave:s,onMouseMove:f,onMouseUp:d,onTouchEnd:h,onTouchMove:p,onTouchStart:v,style:m,width:g}=e,b=re(),[x,O]=y.useState(null),[w,P]=y.useState(null);IE();var A=oM(),E=y.useCallback($=>{A($),typeof t=="function"&&t($),O($),P($)},[A,t,O,P]),j=y.useCallback($=>{b(zy($)),b(Xe({handler:a,reactEvent:$}))},[b,a]),N=y.useCallback($=>{b(ju($)),b(Xe({handler:l,reactEvent:$}))},[b,l]),T=y.useCallback($=>{b(Pm()),b(Xe({handler:s,reactEvent:$}))},[b,s]),C=y.useCallback($=>{b(ju($)),b(Xe({handler:f,reactEvent:$}))},[b,f]),R=y.useCallback(()=>{b(Yy())},[b]),L=y.useCallback($=>{b(Hy($.key))},[b]),Y=y.useCallback($=>{b(Xe({handler:o,reactEvent:$}))},[b,o]),te=y.useCallback($=>{b(Xe({handler:u,reactEvent:$}))},[b,u]),B=y.useCallback($=>{b(Xe({handler:c,reactEvent:$}))},[b,c]),he=y.useCallback($=>{b(Xe({handler:d,reactEvent:$}))},[b,d]),ne=y.useCallback($=>{b(Xe({handler:v,reactEvent:$}))},[b,v]),ke=y.useCallback($=>{b(Vy($)),b(Xe({handler:p,reactEvent:$}))},[b,p]),Ue=y.useCallback($=>{b(Xe({handler:h,reactEvent:$}))},[b,h]);return y.createElement(zm.Provider,{value:x},y.createElement(xh.Provider,{value:w},y.createElement("div",{className:H("recharts-wrapper",n),style:uM({position:"relative",cursor:"default",width:g,height:i},m),onClick:j,onContextMenu:Y,onDoubleClick:te,onFocus:R,onKeyDown:L,onMouseDown:B,onMouseEnter:N,onMouseLeave:T,onMouseMove:C,onMouseUp:he,onTouchEnd:Ue,onTouchMove:ke,onTouchStart:ne,ref:E},r)))}),dM=["children","className","width","height","style","compact","title","desc"];function hM(e,t){if(e==null)return{};var r,n,i=vM(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n{var{children:r,className:n,width:i,height:a,style:o,compact:u,title:c,desc:l}=e,s=hM(e,dM),f=W(s,!1);return u?y.createElement(fh,{otherAttributes:f,title:c,desc:l},r):y.createElement(fM,{className:n,style:o,width:i,height:a,onClick:e.onClick,onMouseLeave:e.onMouseLeave,onMouseEnter:e.onMouseEnter,onMouseMove:e.onMouseMove,onMouseDown:e.onMouseDown,onMouseUp:e.onMouseUp,onContextMenu:e.onContextMenu,onDoubleClick:e.onDoubleClick,onTouchStart:e.onTouchStart,onTouchMove:e.onTouchMove,onTouchEnd:e.onTouchEnd},y.createElement(fh,{otherAttributes:f,title:c,desc:l,ref:t},y.createElement(JT,null,r)))}),mM=["width","height"];function Tu(){return Tu=Object.assign?Object.assign.bind():function(e){for(var t=1;ty.createElement(wM,{chartName:"LineChart",defaultTooltipEventType:"axis",validateTooltipEventTypes:OM,tooltipPayloadSearcher:PE,categoricalChartProps:e,ref:t}));const ZM=({icon:e,title:t,value:r,subtitle:n,trend:i,trendDirection:a,color:o="green",size:u="normal"})=>{const c={green:"bg-green-100 dark:bg-green-900/30 text-green-600 dark:text-green-400",blue:"bg-blue-100 dark:bg-blue-900/30 text-blue-600 dark:text-blue-400",yellow:"bg-yellow-100 dark:bg-yellow-900/30 text-yellow-600 dark:text-yellow-400",purple:"bg-purple-100 dark:bg-purple-900/30 text-purple-600 dark:text-purple-400",red:"bg-red-100 dark:bg-red-900/30 text-red-600 dark:text-red-400",indigo:"bg-indigo-100 dark:bg-indigo-900/30 text-indigo-600 dark:text-indigo-400",orange:"bg-orange-100 dark:bg-orange-900/30 text-orange-600 dark:text-orange-400"},l={up:"text-green-600 dark:text-green-400",down:"text-red-600 dark:text-red-400",neutral:"text-gray-600 dark:text-gray-400"},s={small:"p-4",normal:"p-6",large:"p-8"},f=()=>{if(!a)return null;const d=`w-4 h-4 ${l[a]}`;switch(a){case"up":return be.jsx("svg",{className:d,fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:be.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M7 17l9.2-9.2M17 17V7H7"})});case"down":return be.jsx("svg",{className:d,fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:be.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M17 7l-9.2 9.2M7 7v10h10"})});case"neutral":return be.jsx("svg",{className:d,fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:be.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M20 12H4"})});default:return null}};return be.jsx("div",{className:`bg-white dark:bg-gray-800 ${s[u]} rounded-lg shadow-sm border border-gray-200 dark:border-gray-700`,children:be.jsx("div",{className:"flex items-start justify-between",children:be.jsxs("div",{className:"flex items-center flex-1",children:[be.jsx("div",{className:`p-2 rounded-lg ${c[o]} flex-shrink-0`,children:e}),be.jsxs("div",{className:"ml-4 flex-1",children:[be.jsx("p",{className:"text-sm font-medium text-gray-600 dark:text-gray-300",children:t}),be.jsx("p",{className:"text-2xl font-bold text-gray-900 dark:text-white",children:r}),n&&be.jsx("p",{className:"text-xs text-gray-500 dark:text-gray-400 mt-1",children:n}),i&&a&&be.jsxs("div",{className:"flex items-center mt-2",children:[f(),be.jsx("span",{className:`text-sm ml-1 ${l[a]}`,children:i})]})]})]})})})};export{$t as $,X as A,wy as B,Um as C,ze as D,ZM as E,yy as F,Tr as G,cy as H,Pe as I,Rl as J,oy as K,XM as L,or as M,nt as N,W as O,sr as P,ey as Q,IM as R,gj as S,MM as T,ty as U,ry as V,Ix as W,Ly as X,qy as Y,Mx as Z,mi as _,TC as a,TM as a0,Cv as a1,wM as a2,PE as a3,fe as a4,Ki as a5,zp as a6,Ce as a7,Kp as a8,Fp as a9,Qk as aA,pM as aB,pu as aC,Lt as aD,Jr as aE,Ov as aF,ki as aG,uE as aH,ac as aI,bh as aJ,jj as aK,jM as aL,qe as aM,Fe as aN,Zm as aO,XP as aa,Up as ab,Hp as ac,BA as ad,nm as ae,F as af,Bi as ag,im as ah,yn as ai,om as aj,um as ak,ce as al,LM as am,BM as an,k as ao,je as ap,ye as aq,ux as ar,ut as as,nr as at,nc as au,VM as av,Gn as aw,Zk as ax,KT as ay,Jk as az,os as b,_M as c,Dy as d,EM as e,YM as f,qu as g,S as h,wA as i,ht as j,Rt as k,CM as l,Wi as m,ly as n,ay as o,_i as p,_t as q,GM as r,ks as s,oc as t,re as u,Nu as v,Se as w,D as x,ny as y,xr as z}; diff --git a/frontend/dist/assets/ErrorPages-DbEIbd0-.js b/frontend/dist/assets/ErrorPages-DbEIbd0-.js new file mode 100644 index 0000000000..e5b3088162 --- /dev/null +++ b/frontend/dist/assets/ErrorPages-DbEIbd0-.js @@ -0,0 +1 @@ +import{a as g,j as e,B as t}from"./index-tEqMizXb.js";import"./IconButton-DmJ5sToe.js";import"./Pagination-EPc_5dQ6.js";import"./TreeDetailModal-PvfFs-NG.js";import"./ChartComponents-BKyCVLLP.js";import"./vendor-BtP0CW_r.js";import"./useKeyboardNavigation-DT9lybaQ.js";const j=()=>{const o=g();return{goBack:()=>{o(-1)},goHome:()=>{o("/")},reload:()=>{window.location.reload()}}},v=({className:o})=>e.jsx("svg",{className:o,fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M10 19l-7-7m0 0l7-7m-7 7h18"})}),k=({className:o})=>e.jsx("svg",{className:o,fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M3 12l2-2m0 0l7-7 7 7M5 10v10a1 1 0 001 1h3m10-11l2 2m-2-2v10a1 1 0 01-1 1h-3m-6 0a1 1 0 001-1v-4a1 1 0 011-1h2a1 1 0 011 1v4a1 1 0 001 1m-6 0h6"})}),f=({className:o})=>e.jsx("svg",{className:o,fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"})}),a=({code:o,title:r,message:s,illustration:n,showBackButton:i=!0,showHomeButton:l=!0,showReloadButton:c=!1,className:m,...d})=>{const{goBack:x,goHome:h,reload:u}=j();return e.jsx("div",{className:`min-h-screen flex items-center justify-center bg-gray-50 px-4 ${m}`,...d,children:e.jsxs("div",{className:"max-w-md w-full text-center",children:[e.jsx("div",{className:"mb-8",children:n}),e.jsx("div",{className:"text-6xl font-bold text-gray-800 mb-4",children:o}),e.jsx("h1",{className:"text-2xl font-semibold text-gray-900 mb-2",children:r}),e.jsx("p",{className:"text-gray-600 mb-8",children:s}),e.jsxs("div",{className:"flex justify-center gap-4 flex-wrap mt-8",children:[i&&e.jsxs(t,{onClick:x,variant:"primary",className:"flex items-center gap-2",children:[e.jsx(v,{className:"w-4 h-4"}),"Go Back"]}),c&&e.jsxs(t,{onClick:u,variant:"primary",className:"flex items-center gap-2",children:[e.jsx(f,{className:"w-4 h-4"}),"Try Again"]}),l&&e.jsxs(t,{onClick:h,variant:"secondary",className:"flex items-center gap-2",children:[e.jsx(k,{className:"w-4 h-4"}),"Go Home"]})]})]})})},p=()=>e.jsx("div",{className:"flex items-center justify-center w-32 h-32 mx-auto bg-gray-100 rounded-full",children:e.jsx("svg",{className:"w-16 h-16 text-gray-400",fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1,d:"M9.172 16.172a4 4 0 015.656 0M9 12h6m-6-4h6m2 5.291A7.962 7.962 0 0112 15c-2.34 0-4.291-1.1-5.5-2.709"})})}),w=()=>e.jsx("div",{className:"flex items-center justify-center w-32 h-32 mx-auto bg-red-100 rounded-full",children:e.jsx("svg",{className:"w-16 h-16 text-red-400",fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1,d:"M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-2.5L13.732 4c-.77-.833-1.994-.833-2.464 0L3.34 16.5c-.77.833.192 2.5 1.732 2.5z"})})}),I=()=>e.jsx(a,{code:"404",title:"Page Not Found",message:"The page you're looking for doesn't exist. It might have been moved, deleted, or you entered the wrong URL.",illustration:e.jsx(p,{}),showBackButton:!0,showHomeButton:!0,showReloadButton:!1}),M=()=>e.jsx(a,{code:"500",title:"Internal Server Error",message:"Something went wrong on our end. We're working to fix it. Please try again in a few moments.",illustration:e.jsx(w,{}),showBackButton:!1,showHomeButton:!0,showReloadButton:!0});export{I as NotFoundPage,M as ServerErrorPage}; diff --git a/frontend/dist/assets/ExportPage-DZ0UV3sF.js b/frontend/dist/assets/ExportPage-DZ0UV3sF.js new file mode 100644 index 0000000000..5074196078 --- /dev/null +++ b/frontend/dist/assets/ExportPage-DZ0UV3sF.js @@ -0,0 +1 @@ +import{A as I,f as P,r as c,j as e,L as O,b as L}from"./index-tEqMizXb.js";import{u as T,D as M,a as A}from"./useSidebarState-CJJaSDFr.js";import{G as B,t as R}from"./GlobalFilters-B5C4gkva.js";import"./vendor-BtP0CW_r.js";import"./DarkModeToggle-ZfYhBz0e.js";import"./IconButton-DmJ5sToe.js";const m=new I,V={exportTreesCSV:(r={},a="trees_export.csv")=>m.downloadFile("/exports/trees/csv",r,a),exportTreesXLSX:(r={},a="trees_export.xlsx")=>m.downloadFile("/exports/trees/xlsx",r,a),exportForestAnalytics:(r={},a="forest_analytics.xlsx")=>m.downloadFile("/exports/forest-analytics",r,a),exportTrees:(r={})=>{const{format:a="csv",...g}=r,o=`trees_export_${new Date().toISOString().split("T")[0]}.${a}`,l=a==="xlsx"?"/exports/trees/xlsx":"/exports/trees/csv";return m.downloadFile(l,g,o)}},W=()=>{const{sidebarOpen:r,toggleSidebar:a,closeSidebar:g}=T(),{showToast:o}=P(),[l,w]=c.useState({}),[i,u]=c.useState("csv"),[n,f]=c.useState("filtered"),[d,k]=c.useState({basicInfo:!0,location:!0,health:!0,measurements:!0,images:!1,genetics:!1,growthModel:!1,maintenance:!1,economicValue:!1,canopy:!1,ecologicalBenefits:!1,environmentalFactors:!1,metadata:!1}),[h,j]=c.useState(!1),C=c.useCallback(t=>{w(t)},[]),y=t=>{k(s=>({...s,[t]:!s[t]}))},E=async()=>{j(!0);try{const t={format:i,scope:n,...d};if(n==="filtered"){const s=R(l);Object.assign(t,s)}await V.exportTrees(t),o("Export completed successfully!","success")}catch(t){console.error("Export failed:",t),o("Export failed. Please try again.","error")}finally{j(!1)}},S=()=>{var b,N,v;const t=Object.values(d).filter(p=>p).length;let s="all records";if(n==="filtered"&&(((b=l.selectedForests)==null?void 0:b.length)>0&&(s=`records from ${l.selectedForests.length} forest(s)`),(N=l.dateRange)!=null&&N.startDate&&((v=l.dateRange)!=null&&v.endDate))){const p=l.dateRange.startDate.toLocaleDateString(),D=l.dateRange.endDate.toLocaleDateString();s+=` between ${p} - ${D}`}return{includeCount:t,recordEstimate:s}},{includeCount:x,recordEstimate:F}=S();return e.jsxs("div",{className:"min-h-screen bg-gray-50 dark:bg-gray-900 flex flex-col",children:[e.jsx(M,{onToggleSidebar:a}),e.jsx(A,{isOpen:r,onClose:g}),e.jsx("main",{className:"flex-1 p-4 md:p-6 lg:p-8 lg:ml-64",children:e.jsxs("div",{className:"max-w-7xl mx-auto",children:[e.jsx("div",{className:"mb-8",children:e.jsx("div",{className:"flex justify-between items-center mb-4",children:e.jsxs("div",{children:[e.jsx("nav",{className:"flex mb-2","aria-label":"Breadcrumb",children:e.jsxs("ol",{className:"flex items-center space-x-4",children:[e.jsx("li",{children:e.jsx(O,{to:"/dashboard",className:"text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-300",children:"Overview"})}),e.jsx("li",{children:e.jsx("svg",{className:"flex-shrink-0 h-4 w-4 text-gray-400",fill:"currentColor",viewBox:"0 0 20 20","aria-hidden":"true",children:e.jsx("path",{d:"M5.555 17.776l8-16 .894.448-8 16-.894-.448z"})})}),e.jsx("li",{children:e.jsx("span",{className:"text-gray-900 dark:text-white font-medium",children:"Data Export"})})]})}),e.jsx("h2",{className:"text-3xl font-bold text-gray-900 dark:text-white mb-2",children:"Data Export"}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300",children:"Export your tree data in various formats with customizable options"})]})})}),e.jsxs("div",{className:"grid grid-cols-1 lg:grid-cols-3 gap-6",children:[e.jsxs("div",{className:"lg:col-span-2 space-y-6",children:[e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow p-6",children:[e.jsx("h2",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Export Scope"}),e.jsxs("div",{className:"space-y-3",children:[e.jsxs("label",{className:"flex items-center",children:[e.jsx("input",{type:"radio",value:"all",checked:n==="all",onChange:t=>f(t.target.value),className:"w-4 h-4 text-green-600 focus:ring-green-500"}),e.jsxs("span",{className:"ml-3 text-gray-700 dark:text-gray-300",children:[e.jsx("span",{className:"font-medium",children:"All Data"}),e.jsx("span",{className:"block text-sm text-gray-500 dark:text-gray-400",children:"Export entire database without any filters"})]})]}),e.jsxs("label",{className:"flex items-center",children:[e.jsx("input",{type:"radio",value:"filtered",checked:n==="filtered",onChange:t=>f(t.target.value),className:"w-4 h-4 text-green-600 focus:ring-green-500"}),e.jsxs("span",{className:"ml-3 text-gray-700 dark:text-gray-300",children:[e.jsx("span",{className:"font-medium",children:"Filtered Data"}),e.jsx("span",{className:"block text-sm text-gray-500 dark:text-gray-400",children:"Export only data matching the filters below"})]})]})]})]}),n==="filtered"&&e.jsx("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow",children:e.jsx(B,{onFiltersChange:C})}),e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow p-6",children:[e.jsx("h2",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Data Fields to Include"}),e.jsxs("div",{className:"mb-6",children:[e.jsx("h3",{className:"text-md font-medium text-gray-800 dark:text-gray-200 mb-3",children:"Basic Information"}),e.jsx("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-3",children:Object.entries({basicInfo:"Basic Info (ID, Species, Forest)",location:"Location (Coordinates)",health:"Health Status",measurements:"Measurements (Height, Diameter, CO₂)",images:"Image URLs"}).map(([t,s])=>e.jsxs("label",{className:"flex items-center",children:[e.jsx("input",{type:"checkbox",checked:d[t],onChange:()=>y(t),className:"w-4 h-4 text-green-600 rounded focus:ring-green-500"}),e.jsx("span",{className:"ml-3 text-gray-700 dark:text-gray-300",children:s})]},t))})]}),e.jsxs("div",{children:[e.jsx("h3",{className:"text-md font-medium text-gray-800 dark:text-gray-200 mb-3",children:"Advanced Data"}),e.jsx("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-3",children:Object.entries({genetics:"Genetics (Seed Source, Cultivar, Diversity)",growthModel:"Growth Model (Predictions, Rate, Index)",maintenance:"Maintenance (Fertilization, Pruning, Damage)",economicValue:"Economic Value (Timber, Carbon Credits)",canopy:"Canopy (Area, Density, Condition)",ecologicalBenefits:"Ecological Benefits (CO₂, Pollutants, Wildlife)",environmentalFactors:"Environmental (Climate, Site, Position)",metadata:"Metadata (Planting Method, Timestamps)"}).map(([t,s])=>e.jsxs("label",{className:"flex items-center",children:[e.jsx("input",{type:"checkbox",checked:d[t],onChange:()=>y(t),className:"w-4 h-4 text-green-600 rounded focus:ring-green-500"}),e.jsx("span",{className:"ml-3 text-gray-700 dark:text-gray-300",children:s})]},t))})]})]}),e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow p-6",children:[e.jsx("h2",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Export Format"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-4",children:[e.jsxs("label",{className:"flex items-center p-4 border-2 rounded-lg cursor-pointer transition-colors hover:bg-gray-50 dark:hover:bg-gray-700",style:{borderColor:i==="csv"?"#10b981":"#e5e7eb"},children:[e.jsx("input",{type:"radio",value:"csv",checked:i==="csv",onChange:t=>u(t.target.value),className:"w-4 h-4 text-green-600 focus:ring-green-500"}),e.jsxs("div",{className:"ml-3",children:[e.jsx("span",{className:"font-medium text-gray-900 dark:text-white",children:"CSV"}),e.jsx("span",{className:"block text-sm text-gray-500 dark:text-gray-400",children:"Comma-separated values, compatible with Excel"})]})]}),e.jsxs("label",{className:"flex items-center p-4 border-2 rounded-lg cursor-pointer transition-colors hover:bg-gray-50 dark:hover:bg-gray-700",style:{borderColor:i==="xlsx"?"#10b981":"#e5e7eb"},children:[e.jsx("input",{type:"radio",value:"xlsx",checked:i==="xlsx",onChange:t=>u(t.target.value),className:"w-4 h-4 text-green-600 focus:ring-green-500"}),e.jsxs("div",{className:"ml-3",children:[e.jsx("span",{className:"font-medium text-gray-900 dark:text-white",children:"Excel (XLSX)"}),e.jsx("span",{className:"block text-sm text-gray-500 dark:text-gray-400",children:"Native Excel format with formatting"})]})]})]})]})]}),e.jsx("div",{className:"lg:col-span-1",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow p-6 sticky top-6",children:[e.jsx("h2",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Export Preview"}),e.jsxs("div",{className:"space-y-4 mb-6",children:[e.jsxs("div",{className:"flex justify-between text-sm",children:[e.jsx("span",{className:"text-gray-600 dark:text-gray-400",children:"Format:"}),e.jsx("span",{className:"font-medium text-gray-900 dark:text-white uppercase",children:i})]}),e.jsxs("div",{className:"flex justify-between text-sm",children:[e.jsx("span",{className:"text-gray-600 dark:text-gray-400",children:"Scope:"}),e.jsx("span",{className:"font-medium text-gray-900 dark:text-white capitalize",children:n})]}),e.jsxs("div",{className:"flex justify-between text-sm",children:[e.jsx("span",{className:"text-gray-600 dark:text-gray-400",children:"Fields:"}),e.jsxs("span",{className:"font-medium text-gray-900 dark:text-white",children:[x," selected"]})]}),e.jsxs("div",{className:"text-sm",children:[e.jsx("span",{className:"text-gray-600 dark:text-gray-400",children:"Exporting:"}),e.jsx("span",{className:"block font-medium text-gray-900 dark:text-white mt-1",children:F})]})]}),e.jsx("button",{onClick:E,disabled:h||x===0,className:`w-full py-3 px-4 rounded-lg font-medium transition-colors flex items-center justify-center ${h||x===0?"bg-gray-300 dark:bg-gray-700 text-gray-500 dark:text-gray-400 cursor-not-allowed":"bg-green-600 hover:bg-green-700 text-white"}`,children:h?e.jsxs(e.Fragment,{children:[e.jsx(L,{size:"small",className:"mr-2"}),"Exporting..."]}):e.jsxs(e.Fragment,{children:[e.jsx("svg",{className:"w-5 h-5 mr-2",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"})}),"Export Data"]})}),x===0&&e.jsx("p",{className:"mt-3 text-sm text-red-600 dark:text-red-400",children:"Please select at least one field to export"})]})})]})]})})]})};export{W as ExportPage}; diff --git a/frontend/dist/assets/FinancialDashboardPage-UxW0hI6p.js b/frontend/dist/assets/FinancialDashboardPage-UxW0hI6p.js new file mode 100644 index 0000000000..3b6843712c --- /dev/null +++ b/frontend/dist/assets/FinancialDashboardPage-UxW0hI6p.js @@ -0,0 +1 @@ +import{r as l,c as Me,j as e,b as ie,L as ne}from"./index-tEqMizXb.js";import{C as I,a as C,b as E}from"./ChartComponents-BKyCVLLP.js";import{i as B,u as Oe,f as Te,r as De,S as ue,g as Ke,h as ve,s as Le,j as be,k as je,l as _e,m as Re,n as ze,o as Be,p as ke,q as Fe,t as He,v as Ve,w as $e,x as we,y as We,G as Ge,C as se,z as Ze,A as V,D as J,F as Ye,H as Ue,I as re,J as pe,K as Je,M as Qe,N as te,O as oe,P as Xe,Q as qe,U as et,V as tt,W as at,Z as rt,_ as st,$ as it,a0 as nt,a1 as ot,a2 as lt,a3 as ct,R as $,T as W,a as Q,X,Y as q,B as le,e as de,b as ce,d as H,L as dt,E as ae}from"./EnhancedStatCard-CVgqadrW.js";import{P as ht,a as mt,B as Ne}from"./PieChart-DRbmPJpB.js";import{A as xt,a as ge,u as ut}from"./useDashboardStats-3Mt6zhuV.js";import{G as pt}from"./GlobalFilters-B5C4gkva.js";import{u as gt,D as ft,a as yt}from"./useSidebarState-CJJaSDFr.js";import"./vendor-BtP0CW_r.js";import"./DarkModeToggle-ZfYhBz0e.js";import"./IconButton-DmJ5sToe.js";function Se(t,r,a){return(r=vt(r))in t?Object.defineProperty(t,r,{value:a,enumerable:!0,configurable:!0,writable:!0}):t[r]=a,t}function vt(t){var r=bt(t,"string");return typeof r=="symbol"?r:r+""}function bt(t,r){if(typeof t!="object"||!t)return t;var a=t[Symbol.toPrimitive];if(a!==void 0){var s=a.call(t,r);if(typeof s!="object")return s;throw new TypeError("@@toPrimitive must return a primitive value.")}return(r==="string"?String:Number)(t)}function jt(t){var r=Oe();return l.useEffect(()=>(r(Te(t)),()=>{r(De(t))}),[t,r]),null}class he extends l.Component{render(){return l.createElement(jt,{domain:this.props.domain,id:this.props.zAxisId,dataKey:this.props.dataKey,name:this.props.name,unit:this.props.unit,range:this.props.range,scale:this.props.scale,type:this.props.type,allowDuplicatedCategory:B.allowDuplicatedCategory,allowDataOverflow:B.allowDataOverflow,reversed:B.reversed,includeHidden:B.includeHidden})}}Se(he,"displayName","ZAxis");Se(he,"defaultProps",{zAxisId:0,range:B.range,scale:B.scale,type:B.type});var kt=["option","isActive"];function U(){return U=Object.assign?Object.assign.bind():function(t){for(var r=1;rbe(t,"xAxis",r,m),Pt=(t,r,a,s,c,i,m)=>je(t,"xAxis",r,m),It=(t,r,a,s,c,i,m)=>be(t,"yAxis",a,m),Ct=(t,r,a,s,c,i,m)=>je(t,"yAxis",a,m),Et=(t,r,a,s)=>_e(t,"zAxis",s,!1),Mt=(t,r,a,s,c)=>c,Ot=(t,r,a,s,c,i)=>i,Tt=(t,r,a,s,c,i,m)=>Le(t,r,a,m),Dt=ve([Re,Mt],(t,r)=>{if(t.some(a=>a.type==="scatter"&&r.dataKey===a.dataKey&&r.data===a.data))return r}),Kt=ve([Tt,At,Pt,It,Ct,Et,Dt,Ot],(t,r,a,s,c,i,m,y)=>{var{chartData:u,dataStartIndex:g,dataEndIndex:j}=t;if(m!=null){var p;if((m==null?void 0:m.data)!=null&&m.data.length>0?p=m.data:p=u==null?void 0:u.slice(g,j+1),!(p==null||r==null||s==null||a==null||c==null||(a==null?void 0:a.length)===0||(c==null?void 0:c.length)===0))return Gt({displayedData:p,xAxis:r,yAxis:s,zAxis:i,scatterSettings:m,xAxisTicks:a,yAxisTicks:c,cells:y})}}),Lt=["onMouseEnter","onClick","onMouseLeave"],_t=["animationBegin","animationDuration","animationEasing","hide","isAnimationActive","legendType","lineJointType","lineType","shape","xAxisId","yAxisId","zAxisId"];function Ae(t,r){if(t==null)return{};var a,s,c=Rt(t,r);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(t);for(s=0;s{var{dataKey:r,name:a,fill:s,legendType:c,hide:i}=t;return[{inactive:i,dataKey:r,type:c,color:s,value:ke(a,r),payload:t}]};function Ht(t){var{points:r,props:a}=t,{line:s,lineType:c,lineJointType:i}=a;if(!s)return null;var m=oe(a,!1),y=oe(s,!1),u,g;if(c==="joint")u=r.map(o=>({x:o.cx,y:o.cy}));else if(c==="fitting"){var{xmin:j,xmax:p,a:v,b:x}=nt(r),n=o=>v*o+x;u=[{x:j,y:n(j)},{x:p,y:n(p)}]}var d=M(M(M({},m),{},{fill:"none",stroke:m&&m.fill},y),{},{points:u});return l.isValidElement(s)?g=l.cloneElement(s,d):typeof s=="function"?g=s(d):g=l.createElement(ot,G({},d,{type:i})),l.createElement(J,{className:"recharts-scatter-line",key:"recharts-scatter-line"},g)}function Pe(t){var{points:r,showLabels:a,allOtherScatterProps:s}=t,{shape:c,activeShape:i,dataKey:m}=s,y=oe(s,!1),u=we(Xe),{onMouseEnter:g,onClick:j,onMouseLeave:p}=s,v=Ae(s,Lt),x=qe(g,s.dataKey),n=et(p),d=tt(j,s.dataKey);return r==null?null:l.createElement(l.Fragment,null,l.createElement(Ht,{points:r,props:s}),r.map((o,h)=>{var f=i&&u===String(h),b=f?i:c,k=M(M(M({key:"symbol-".concat(h)},y),o),{},{[rt]:h,[at]:String(m)});return l.createElement(J,G({className:"recharts-scatter-symbol"},st(v,o,h),{onMouseEnter:x(o,h),onMouseLeave:n(o,h),onClick:d(o,h),key:"symbol-".concat(o==null?void 0:o.cx,"-").concat(o==null?void 0:o.cy,"-").concat(o==null?void 0:o.size,"-").concat(h)}),l.createElement(St,G({option:b,isActive:f},k)))}),a&&it.renderCallByParent(s,r))}function Vt(t){var{previousPointsRef:r,props:a}=t,{points:s,isAnimationActive:c,animationBegin:i,animationDuration:m,animationEasing:y}=a,u=r.current,g=Je(a,"recharts-scatter-"),[j,p]=l.useState(!1),v=l.useCallback(()=>{p(!1)},[]),x=l.useCallback(()=>{p(!0)},[]);return l.createElement(Qe,{begin:i,duration:m,isActive:c,easing:y,from:{t:0},to:{t:1},onAnimationEnd:v,onAnimationStart:x,key:g},n=>{var{t:d}=n,o=d===1?s:s.map((h,f)=>{var b=u&&u[f];if(b){var k=te(b.cx,h.cx),S=te(b.cy,h.cy),w=te(b.size,h.size);return M(M({},h),{},{cx:k(d),cy:S(d),size:w(d)})}var A=te(0,h.size);return M(M({},h),{},{size:A(d)})});return d>0&&(r.current=o),l.createElement(J,null,l.createElement(Pe,{points:o,allOtherScatterProps:a,showLabels:!j}))})}function $t(t){var{points:r,isAnimationActive:a}=t,s=l.useRef(null),c=s.current;return a&&r&&r.length&&(!c||c!==r)?l.createElement(Vt,{props:t,previousPointsRef:s}):l.createElement(Pe,{points:r,allOtherScatterProps:t,showLabels:!0})}function Wt(t){var{dataKey:r,points:a,stroke:s,strokeWidth:c,fill:i,name:m,hide:y,tooltipType:u}=t;return{dataDefinedOnItem:a==null?void 0:a.map(g=>g.tooltipPayload),positions:a==null?void 0:a.map(g=>g.tooltipPosition),settings:{stroke:s,strokeWidth:c,fill:i,nameKey:void 0,dataKey:r,name:ke(m,r),hide:y,type:u,color:i,unit:""}}}function Gt(t){var{displayedData:r,xAxis:a,yAxis:s,zAxis:c,scatterSettings:i,xAxisTicks:m,yAxisTicks:y,cells:u}=t,g=V(a.dataKey)?i.dataKey:a.dataKey,j=V(s.dataKey)?i.dataKey:s.dataKey,p=c&&c.dataKey,v=c?c.range:he.defaultProps.range,x=v&&v[0],n=a.scale.bandwidth?a.scale.bandwidth():0,d=s.scale.bandwidth?s.scale.bandwidth():0;return r.map((o,h)=>{var f=re(o,g),b=re(o,j),k=!V(p)&&re(o,p)||"-",S=[{name:V(a.dataKey)?i.name:a.name||a.dataKey,unit:a.unit||"",value:f,payload:o,dataKey:g,type:i.tooltipType},{name:V(s.dataKey)?i.name:s.name||s.dataKey,unit:s.unit||"",value:b,payload:o,dataKey:j,type:i.tooltipType}];k!=="-"&&S.push({name:c.name||c.dataKey,unit:c.unit||"",value:k,payload:o,dataKey:p,type:i.tooltipType});var w=pe({axis:a,ticks:m,bandSize:n,entry:o,index:h,dataKey:g}),A=pe({axis:s,ticks:y,bandSize:d,entry:o,index:h,dataKey:j}),D=k!=="-"?c.scale(k):x,P=Math.sqrt(Math.max(D,0)/Math.PI);return M(M({},o),{},{cx:w,cy:A,x:w-P,y:A-P,width:2*P,height:2*P,size:D,node:{x:f,y:b,z:k},tooltipPayload:S,tooltipPosition:{x:w,y:A},payload:o},u&&u[h]&&u[h].props)})}var Zt=(t,r,a)=>({x:t.cx,y:t.cy,value:a==="x"?+t.node.x:+t.node.y,errorVal:re(t,r)});function Yt(t){var r=l.useRef(Ze("recharts-scatter-")),{hide:a,points:s,className:c,needClip:i,xAxisId:m,yAxisId:y,id:u,children:g}=t;if(a)return null;var j=Me("recharts-scatter",c),p=V(u)?r.current:u;return l.createElement(J,{className:j,clipPath:i?"url(#clipPath-".concat(p,")"):null},i&&l.createElement("defs",null,l.createElement(Ye,{clipPathId:p,xAxisId:m,yAxisId:y})),l.createElement(Ue,{xAxisId:m,yAxisId:y,data:s,dataPointFormatter:Zt,errorBarOffset:0},g),l.createElement(J,{key:"recharts-scatter-symbols"},l.createElement($t,t)))}var Ie={xAxisId:0,yAxisId:0,zAxisId:0,legendType:"circle",lineType:"joint",lineJointType:"linear",data:[],shape:"circle",hide:!1,isAnimationActive:!Ge.isSsr,animationBegin:0,animationDuration:400,animationEasing:"linear"};function Ut(t){var r=Fe(t,Ie),{animationBegin:a,animationDuration:s,animationEasing:c,hide:i,isAnimationActive:m,legendType:y,lineJointType:u,lineType:g,shape:j,xAxisId:p,yAxisId:v,zAxisId:x}=r,n=Ae(r,_t),{needClip:d}=He(p,v),o=l.useMemo(()=>Ve(t.children,se),[t.children]),h=l.useMemo(()=>({name:t.name,tooltipType:t.tooltipType,data:t.data,dataKey:t.dataKey}),[t.data,t.dataKey,t.name,t.tooltipType]),f=$e(),b=we(k=>Kt(k,p,v,x,h,o,f));return d==null?null:l.createElement(l.Fragment,null,l.createElement(We,{fn:Wt,args:M(M({},t),{},{points:b})}),l.createElement(Yt,G({},n,{xAxisId:p,yAxisId:v,zAxisId:x,lineType:g,lineJointType:u,legendType:y,shape:j,hide:i,isAnimationActive:m,animationBegin:a,animationDuration:s,animationEasing:c,points:b,needClip:d})))}class xe extends l.Component{render(){return l.createElement(ze,{type:"scatter",data:this.props.data,xAxisId:this.props.xAxisId,yAxisId:this.props.yAxisId,zAxisId:this.props.zAxisId,dataKey:this.props.dataKey,stackId:void 0,hide:this.props.hide,barSize:void 0},l.createElement(Be,{legendPayload:Ft(this.props)}),l.createElement(Ut,this.props))}}me(xe,"displayName","Scatter");me(xe,"defaultProps",Ie);var Jt=["item"],Qt=l.forwardRef((t,r)=>l.createElement(lt,{chartName:"ScatterChart",defaultTooltipEventType:"item",validateTooltipEventTypes:Jt,tooltipPayloadSearcher:ct,categoricalChartProps:t,ref:r}));const Xt=({filters:t={},chartType:r="bar"})=>{const[a,s]=l.useState([]),[c,i]=l.useState(!0),[m,y]=l.useState(null),u=l.useMemo(()=>({excellent:"#10b981",good:"#3b82f6",fair:"#f59e0b",poor:"#ef4444",critical:"#dc2626"}),[]);l.useEffect(()=>{(async()=>{var d,o;try{i(!0),y(null);const b=(((o=(d=(await de.getStats(t)).data)==null?void 0:d.distributions)==null?void 0:o.health)||[]).map(S=>{var w;return{status:S._id||"Unknown",count:S.count||0,fill:u[(w=S._id)==null?void 0:w.toLowerCase()]||"#6b7280"}}),k=["excellent","good","fair","poor","critical"];b.sort((S,w)=>{const A=k.indexOf(S.status.toLowerCase()),D=k.indexOf(w.status.toLowerCase());return A-D}),s(b)}catch(h){console.error("Error fetching health distribution data:",h),y(h.message||"Failed to load health distribution data")}finally{i(!1)}})()},[t,u]);const g=n=>n.charAt(0).toUpperCase()+n.slice(1),j=({active:n,payload:d,label:o})=>{if(n&&d&&d.length){const h=d[0];return e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white",children:`${g(o||h.payload.status)}: ${h.value} trees`}),e.jsx("p",{className:"text-xs text-gray-500 dark:text-gray-400",children:`${(h.value/h.payload.total*100).toFixed(1)}%`})]})}return null},p=({active:n,payload:d})=>{var o;if(n&&d&&d.length){const h=d[0],f=h.payload.totalTrees||((o=h.payload.data)==null?void 0:o.reduce((b,k)=>b+k.count,0))||0;return e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white",children:`${g(h.payload.status)}: ${h.value} trees`}),e.jsx("p",{className:"text-xs text-gray-500 dark:text-gray-400",children:f>0?`${(h.value/f*100).toFixed(1)}%`:"0%"})]})}return null};if(c)return e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Tree Health Status Distribution"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx(ie,{size:"32px",text:"Loading health data..."})})]});if(m)return e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Tree Health Status Distribution"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsxs("div",{className:"text-center",children:[e.jsx("p",{className:"text-red-600 mb-2",children:"Error loading chart data"}),e.jsx("p",{className:"text-sm text-gray-500",children:m})]})})]});if(!a||a.length===0)return e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Tree Health Status Distribution"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx("p",{className:"text-gray-500",children:"No health status data available"})})]});const v=a.reduce((n,d)=>n+d.count,0),x=a.map(n=>({...n,total:v}));return r==="pie"?e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Tree Health Status Distribution"})}),e.jsx($,{width:"100%",height:300,children:e.jsxs(ht,{children:[e.jsx(mt,{data:x,cx:"50%",cy:"50%",labelLine:!1,label:({status:n,percent:d})=>`${g(n)}: ${(d*100).toFixed(0)}%`,outerRadius:80,fill:"#8884d8",dataKey:"count",children:x.map((n,d)=>e.jsx(se,{fill:n.fill},`cell-${d}`))}),e.jsx(W,{content:e.jsx(p,{})})]})})]}):e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Tree Health Status Distribution"})}),e.jsx($,{width:"100%",height:300,children:e.jsxs(Ne,{data:x,margin:{top:5,right:30,left:20,bottom:5},children:[e.jsx(Q,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(X,{dataKey:"status",className:"text-xs",tick:{fontSize:12},tickFormatter:g}),e.jsx(q,{className:"text-xs",tick:{fontSize:12}}),e.jsx(W,{content:e.jsx(j,{})}),e.jsx(le,{dataKey:"count",radius:[4,4,0,0],children:x.map((n,d)=>e.jsx(se,{fill:n.fill},`cell-${d}`))})]})}),e.jsx("div",{className:"flex flex-wrap justify-center mt-4 gap-4",children:x.map((n,d)=>e.jsxs("div",{className:"flex items-center",children:[e.jsx("div",{className:"w-3 h-3 rounded mr-2",style:{backgroundColor:n.fill}}),e.jsxs("span",{className:"text-xs text-gray-600 dark:text-gray-400",children:[g(n.status)," (",n.count,")"]})]},d))})]})},qt=({filters:t={},chartType:r="bar"})=>{const[a,s]=l.useState([]),[c,i]=l.useState(!0),[m,y]=l.useState(null);l.useEffect(()=>{(async()=>{var n,d;try{i(!0),y(null);const h=(await de.getStats(t)).data,f=(n=h==null?void 0:h.investor)==null?void 0:n.timber,b=(d=h==null?void 0:h.investor)==null?void 0:d.carbonCredits,S=["Pine","Oak","Birch","Spruce","Maple","Beech","Fir","Cedar","Willow","Ash"].map((w,A)=>{const D=((f==null?void 0:f.averageValuePerTree)||5e3)*(.8+Math.random()*.4),P=((b==null?void 0:b.averagePrice)||500)*(10+Math.random()*20),K={Pine:{timber:1,carbon:.9,growth:1.2},Oak:{timber:1.4,carbon:1.3,growth:.8},Birch:{timber:.8,carbon:1,growth:1.5},Spruce:{timber:1.1,carbon:1.1,growth:1},Maple:{timber:1.3,carbon:1.2,growth:.9},Beech:{timber:1.2,carbon:1.1,growth:.85},Fir:{timber:1,carbon:1,growth:1},Cedar:{timber:1.5,carbon:1.4,growth:.7},Willow:{timber:.6,carbon:.8,growth:1.8},Ash:{timber:1.25,carbon:1.15,growth:.95}}[w]||{timber:1,carbon:1,growth:1},L=Math.round(D*K.timber),_=Math.round(P*K.carbon),O=L+_,R=Math.round((O/3e3-1)*100),F=Math.round(K.growth*100)/10;return{species:w,timberValue:L,carbonValue:_,totalValue:O,roi:R,growthRate:F,riskLevel:{Pine:2,Oak:1,Birch:3,Spruce:2,Maple:1,Beech:1,Fir:2,Cedar:1,Willow:4,Ash:2}[w]||2,treeCount:Math.floor(Math.random()*500)+100,averageAge:Math.floor(Math.random()*30)+10,timberValuePerYear:Math.round(L/25),carbonValuePerYear:Math.round(_/10),totalROI:R}}).sort((w,A)=>A.totalValue-w.totalValue);s(S)}catch(o){console.error("Error fetching species performance data:",o),y(o.message||"Failed to load species data")}finally{i(!1)}})()},[t,r]);const u=x=>x?new Intl.NumberFormat("sv-SE",{style:"currency",currency:"SEK",minimumFractionDigits:0,maximumFractionDigits:0}).format(x):"0 SEK",g=x=>`${x}%`,j=x=>x>=15?"#10b981":x>=5?"#f59e0b":"#ef4444",p=({active:x,payload:n,label:d})=>{if(x&&n&&n.length){const o=n[0].payload;return e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white mb-2",children:d}),e.jsxs("div",{className:"space-y-1",children:[e.jsxs("p",{className:"text-sm text-green-600 dark:text-green-400",children:["Total Value: ",u(o.totalValue)]}),e.jsxs("p",{className:"text-sm text-blue-600 dark:text-blue-400",children:["Timber Value: ",u(o.timberValue)]}),e.jsxs("p",{className:"text-sm text-purple-600 dark:text-purple-400",children:["Carbon Value: ",u(o.carbonValue)]}),e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["ROI: ",g(o.roi)]}),e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["Growth Rate: ",o.growthRate,"%/year"]}),e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["Trees: ",o.treeCount.toLocaleString()]}),e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["Risk Level: ",o.riskLevel,"/5"]})]})]})}return null},v=({active:x,payload:n,label:d})=>{if(x&&n&&n.length){const o=n[0].payload;return e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white mb-2",children:o.species}),e.jsxs("div",{className:"space-y-1",children:[e.jsxs("p",{className:"text-sm text-blue-600 dark:text-blue-400",children:["ROI: ",g(o.roi)]}),e.jsxs("p",{className:"text-sm text-green-600 dark:text-green-400",children:["Growth Rate: ",o.growthRate,"%/year"]}),e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["Total Value: ",u(o.totalValue)]})]})]})}return null};return c?e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Species Economic Performance"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx(ie,{size:"32px",text:"Loading species data..."})})]}):m?e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Species Economic Performance"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsxs("div",{className:"text-center",children:[e.jsx("p",{className:"text-red-600 mb-2",children:"Error loading chart data"}),e.jsx("p",{className:"text-sm text-gray-500",children:m})]})})]}):!a||a.length===0?e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Species Economic Performance"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx("p",{className:"text-gray-500",children:"No species data available"})})]}):r==="scatter"?e.jsxs(I,{children:[e.jsxs(C,{children:[e.jsx(E,{children:"ROI vs Growth Rate by Species"}),e.jsx("p",{className:"text-sm text-gray-500 dark:text-gray-400",children:"Investment return vs growth performance analysis"})]}),e.jsx($,{width:"100%",height:300,children:e.jsxs(Qt,{margin:{top:20,right:20,bottom:20,left:20},children:[e.jsx(Q,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(X,{type:"number",dataKey:"growthRate",name:"Growth Rate",className:"text-xs",tick:{fontSize:12},label:{value:"Growth Rate (%/year)",position:"insideBottom",offset:-10}}),e.jsx(q,{type:"number",dataKey:"roi",name:"ROI",className:"text-xs",tick:{fontSize:12},label:{value:"ROI (%)",angle:-90,position:"insideLeft"}}),e.jsx(W,{content:e.jsx(v,{})}),e.jsx(xe,{name:"Species",data:a,fill:"#3b82f6",children:a.map((x,n)=>e.jsx(se,{fill:j(x.roi)},`cell-${n}`))})]})})]}):e.jsxs(I,{children:[e.jsxs(C,{children:[e.jsx(E,{children:"Species Economic Performance"}),e.jsx("p",{className:"text-sm text-gray-500 dark:text-gray-400",children:"Timber and carbon value potential by tree species"})]}),e.jsx($,{width:"100%",height:300,children:e.jsxs(Ne,{data:a,margin:{top:5,right:30,left:20,bottom:5},children:[e.jsx(Q,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(X,{dataKey:"species",className:"text-xs",tick:{fontSize:11},angle:-45,textAnchor:"end",height:60}),e.jsx(q,{className:"text-xs",tick:{fontSize:12},tickFormatter:x=>`${(x/1e3).toFixed(0)}k`}),e.jsx(W,{content:e.jsx(p,{})}),e.jsx(ce,{}),e.jsx(le,{dataKey:"timberValue",stackId:"value",fill:"#8b5cf6",name:"Timber Value (SEK)",radius:[0,0,0,0]}),e.jsx(le,{dataKey:"carbonValue",stackId:"value",fill:"#10b981",name:"Carbon Value (SEK)",radius:[4,4,0,0]})]})})]})},ye=({filters:t={},chartType:r="line"})=>{const[a,s]=l.useState([]),[c,i]=l.useState(!0),[m,y]=l.useState(null);l.useEffect(()=>{(async()=>{try{i(!0),y(null);const x=(await de.getStats(t)).data,n=x==null?void 0:x.height,d=(n==null?void 0:n.average)||15,o=new Date().getFullYear(),h=[];for(let f=-5;f<=5;f++){const b=o+f,k=f<=0,S=f===0,w=.8,A=.6,D=.4;let P=null,T=null;if(k||S){const N=d+f*A,z=Math.sin(f*Math.PI/3)*.2,Ce=(Math.random()-.5)*.4;P=Math.max(0,N+z+Ce);const Ee=(Math.random()-.5)*.3;T=N+Ee}else T=d+f*A;const K=Math.abs(f)*.1,L=T+K,_=Math.max(0,T-K),O={temperature:18+Math.sin(f*.5)*2,rainfall:800+Math.cos(f*.3)*100,soilQuality:7.5+(Math.random()-.5)*.5,co2Level:420+f*2},R=Math.max(0,1-Math.abs(O.temperature-18)*.05),F=Math.max(0,1-Math.abs(O.rainfall-800)*.001),Z=Math.max(0,O.soilQuality/10),Y=(R+F+Z)/3,ee=A*Y;h.push({year:b,yearLabel:b.toString(),actualHeight:P?Math.round(P*10)/10:null,predictedHeight:Math.round(T*10)/10,upperBound:Math.round(L*10)/10,lowerBound:Math.round(_*10)/10,growthRate:Math.round(ee*100)/100,isHistorical:k,isFuture:f>0,isCurrent:S,environmentalScore:Math.round(Y*100),temperature:Math.round(O.temperature*10)/10,rainfall:Math.round(O.rainfall),soilQuality:Math.round(O.soilQuality*10)/10,accuracy:P&&T?Math.round((1-Math.abs(P-T)/P)*100):null})}s(h)}catch(v){console.error("Error fetching growth prediction data:",v),y(v.message||"Failed to load growth data")}finally{i(!1)}})()},[t,r]);const u=p=>p?`${p}m`:"0m",g=p=>p?`${p}m/yr`:"0m/yr",j=({active:p,payload:v,label:x})=>{if(p&&v&&v.length){const n=v[0].payload;return e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white mb-2",children:x}),e.jsxs("div",{className:"space-y-1",children:[n.actualHeight&&e.jsxs("p",{className:"text-sm text-blue-600 dark:text-blue-400",children:["Actual Height: ",u(n.actualHeight)]}),e.jsxs("p",{className:"text-sm text-green-600 dark:text-green-400",children:["Predicted Height: ",u(n.predictedHeight)]}),n.upperBound&&n.lowerBound&&e.jsxs("p",{className:"text-sm text-purple-600 dark:text-purple-400",children:["Range: ",u(n.lowerBound)," - ",u(n.upperBound)]}),e.jsxs("p",{className:"text-sm text-orange-600 dark:text-orange-400",children:["Growth Rate: ",g(n.growthRate)]}),n.accuracy&&e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["Prediction Accuracy: ",n.accuracy,"%"]}),e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["Environmental Score: ",n.environmentalScore,"%"]}),e.jsxs("div",{className:"text-xs text-gray-500 dark:text-gray-400 border-t pt-1 mt-1",children:[e.jsxs("p",{children:["Temp: ",n.temperature,"°C | Rainfall: ",n.rainfall,"mm"]}),e.jsxs("p",{children:["Soil pH: ",n.soilQuality]})]})]})]})}return null};return c?e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Growth Performance vs Predictions"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx(ie,{size:"32px",text:"Loading growth data..."})})]}):m?e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Growth Performance vs Predictions"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsxs("div",{className:"text-center",children:[e.jsx("p",{className:"text-red-600 mb-2",children:"Error loading chart data"}),e.jsx("p",{className:"text-sm text-gray-500",children:m})]})})]}):!a||a.length===0?e.jsxs(I,{children:[e.jsx(C,{children:e.jsx(E,{children:"Growth Performance vs Predictions"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx("p",{className:"text-gray-500",children:"No growth data available"})})]}):r==="area"?e.jsxs(I,{children:[e.jsxs(C,{children:[e.jsx(E,{children:"Growth Prediction Confidence Range"}),e.jsx("p",{className:"text-sm text-gray-500 dark:text-gray-400",children:"Predicted height range with confidence intervals"})]}),e.jsx($,{width:"100%",height:300,children:e.jsxs(xt,{data:a,margin:{top:5,right:30,left:20,bottom:5},children:[e.jsx(Q,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(X,{dataKey:"yearLabel",className:"text-xs",tick:{fontSize:12}}),e.jsx(q,{className:"text-xs",tick:{fontSize:12},tickFormatter:u}),e.jsx(W,{content:e.jsx(j,{})}),e.jsx(ce,{}),e.jsx(ge,{dataKey:"upperBound",stackId:"1",stroke:"#e5e7eb",fill:"#f3f4f6",name:"Upper Confidence",fillOpacity:.3}),e.jsx(ge,{dataKey:"lowerBound",stackId:"2",stroke:"#e5e7eb",fill:"#ffffff",name:"Lower Confidence",fillOpacity:.3}),e.jsx(H,{type:"monotone",dataKey:"predictedHeight",stroke:"#10b981",strokeWidth:2,name:"Predicted Height",dot:{r:3,fill:"#10b981"},strokeDasharray:"0"}),e.jsx(H,{type:"monotone",dataKey:"actualHeight",stroke:"#3b82f6",strokeWidth:3,name:"Actual Height",dot:{r:4,fill:"#3b82f6"},connectNulls:!1})]})})]}):e.jsxs(I,{children:[e.jsxs(C,{children:[e.jsx(E,{children:"Growth Performance vs Predictions"}),e.jsx("p",{className:"text-sm text-gray-500 dark:text-gray-400",children:"Historical actual vs predicted growth with future forecasts"})]}),e.jsx($,{width:"100%",height:300,children:e.jsxs(dt,{data:a,margin:{top:5,right:30,left:20,bottom:5},children:[e.jsx(Q,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(X,{dataKey:"yearLabel",className:"text-xs",tick:{fontSize:12}}),e.jsx(q,{className:"text-xs",tick:{fontSize:12},tickFormatter:u}),e.jsx(W,{content:e.jsx(j,{})}),e.jsx(ce,{}),e.jsx(H,{type:"monotone",dataKey:"actualHeight",stroke:"#3b82f6",strokeWidth:3,name:"Actual Height",dot:{r:4,fill:"#3b82f6"},connectNulls:!1}),e.jsx(H,{type:"monotone",dataKey:"predictedHeight",stroke:"#10b981",strokeWidth:2,name:"Predicted Height",dot:{r:3,fill:"#10b981"},strokeDasharray:"5 5"}),e.jsx(H,{type:"monotone",dataKey:"upperBound",stroke:"#94a3b8",strokeWidth:1,name:"Upper Bound",dot:!1,strokeDasharray:"2 2"}),e.jsx(H,{type:"monotone",dataKey:"lowerBound",stroke:"#94a3b8",strokeWidth:1,name:"Lower Bound",dot:!1,strokeDasharray:"2 2"})]})})]})},da=()=>{var n,d,o,h,f,b,k,S,w,A,D,P,T,K,L,_,O,R,F,Z,Y,ee;const{sidebarOpen:t,toggleSidebar:r,closeSidebar:a}=gt(),[s,c]=l.useState({}),{stats:i,loading:m,error:y,refresh:u}=ut(s),g=l.useCallback(N=>{c(N)},[]),j=N=>N?N.toLocaleString():"0",p=N=>N?`${N.toFixed(1)}%`:"0%",v=(N,z="SEK")=>N?new Intl.NumberFormat("sv-SE",{style:"currency",currency:z,minimumFractionDigits:0,maximumFractionDigits:0}).format(N):"0",x=(N,z="SEK")=>N?N>=1e6?`${(N/1e6).toFixed(1)}M ${z}`:N>=1e3?`${(N/1e3).toFixed(1)}K ${z}`:v(N,z):"0";return e.jsxs("div",{className:"min-h-screen bg-gray-50 dark:bg-gray-900 flex flex-col",children:[e.jsx(ft,{onToggleSidebar:r}),e.jsx(yt,{isOpen:t,onClose:a}),e.jsx("main",{className:"flex-1 p-4 md:p-6 lg:p-8 lg:ml-64",children:e.jsxs("div",{className:"max-w-7xl mx-auto",children:[e.jsx("div",{className:"mb-8",children:e.jsx("div",{className:"flex justify-between items-center mb-4",children:e.jsxs("div",{children:[e.jsx("nav",{className:"flex mb-2","aria-label":"Breadcrumb",children:e.jsxs("ol",{className:"flex items-center space-x-4",children:[e.jsx("li",{children:e.jsx(ne,{to:"/dashboard",className:"text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-300",children:"Overview"})}),e.jsx("li",{children:e.jsx("svg",{className:"flex-shrink-0 h-4 w-4 text-gray-400",fill:"currentColor",viewBox:"0 0 20 20","aria-hidden":"true",children:e.jsx("path",{d:"M5.555 17.776l8-16 .894.448-8 16-.894-.448z"})})}),e.jsx("li",{children:e.jsx("span",{className:"text-gray-900 dark:text-white font-medium",children:"Financial Dashboard"})})]})}),e.jsx("h2",{className:"text-3xl font-bold text-gray-900 dark:text-white mb-2",children:"Financial Dashboard"}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300",children:"Investment performance, revenue analysis, and financial risk assessment for your forest portfolio."})]})})}),e.jsx(pt,{onFiltersChange:g}),m?e.jsx("div",{className:"flex justify-center items-center py-12",children:e.jsx(ie,{text:"Loading financial data..."})}):y?e.jsxs("div",{className:"bg-red-50 border border-red-200 rounded-lg p-4 mb-8",children:[e.jsxs("p",{className:"text-red-600",children:["Error loading financial statistics: ",y]}),e.jsx("button",{onClick:u,className:"mt-2 text-red-600 hover:text-red-800 underline",children:"Try again"})]}):e.jsx(e.Fragment,{children:e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Investment Performance"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6",children:[e.jsx(ae,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 8c-1.657 0-3 .895-3 2s1.343 2 3 2 3 .895 3 2-1.343 2-3 2m0-8c1.11 0 2.08.402 2.599 1M12 8V7m0 1v8m0 0v1m0-1c-1.11 0-2.08-.402-2.599-1"})}),title:"Portfolio Value",value:x((d=(n=i==null?void 0:i.investor)==null?void 0:n.portfolio)==null?void 0:d.totalCurrentValue),subtitle:`${((h=(o=i==null?void 0:i.investor)==null?void 0:o.portfolio)==null?void 0:h.forestCount)||0} forests | Avg: ${x((b=(f=i==null?void 0:i.investor)==null?void 0:f.portfolio)==null?void 0:b.averageValue)}`,color:"green"}),e.jsx(ae,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M13 7h8m0 0v8m0-8l-8 8-4-4-6 6"})}),title:"Average ROI",value:p((S=(k=i==null?void 0:i.investor)==null?void 0:k.roi)==null?void 0:S.averageROI),subtitle:`Range: ${p((A=(w=i==null?void 0:i.investor)==null?void 0:w.roi)==null?void 0:A.minROI)} - ${p((P=(D=i==null?void 0:i.investor)==null?void 0:D.roi)==null?void 0:P.maxROI)}`,color:"blue"}),e.jsx(ae,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M3 6l3 1m0 0l-3 9a5.002 5.002 0 006.001 0M6 7l3 9M6 7l6-2m6 2l3-1m-3 1l-3 9a5.002 5.002 0 006.001 0M18 7l3 9m-3-9l-6-2m0-2v2m0 16V5m0 16H9m3 0h3"})}),title:"Carbon Credits",value:j((K=(T=i==null?void 0:i.investor)==null?void 0:T.carbonCredits)==null?void 0:K.totalAvailable),subtitle:`${j((_=(L=i==null?void 0:i.investor)==null?void 0:L.carbonCredits)==null?void 0:_.totalSold)} sold | ${v((R=(O=i==null?void 0:i.investor)==null?void 0:O.carbonCredits)==null?void 0:R.averagePrice)}/credit`,color:"green"}),e.jsx(ae,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M19 21V5a2 2 0 00-2-2H7a2 2 0 00-2 2v16m14 0h2m-2 0h-5m-9 0H3m2 0h5M9 7h1m-1 4h1m4-4h1m-1 4h1m-5 10v-5a1 1 0 011-1h2a1 1 0 011 1v5m-4 0h4"})}),title:"Timber Value",value:x((Z=(F=i==null?void 0:i.investor)==null?void 0:F.timber)==null?void 0:Z.totalValue),subtitle:`${v((ee=(Y=i==null?void 0:i.investor)==null?void 0:Y.timber)==null?void 0:ee.averageValuePerTree)}/tree avg`,color:"orange"})]})]})}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Revenue Analysis"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-6",children:[e.jsx("div",{children:e.jsx(qt,{filters:s,chartType:"bar"})}),e.jsx("div",{children:e.jsx(Xt,{filters:s,chartType:"bar"})})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Growth & Returns"}),e.jsxs("div",{className:"grid grid-cols-1 lg:grid-cols-2 gap-6",children:[e.jsx("div",{className:"col-span-1 lg:col-span-2",children:e.jsx(ye,{filters:s,chartType:"line"})}),e.jsx("div",{className:"col-span-1 lg:col-span-2",children:e.jsx(ye,{filters:s,chartType:"area"})})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Financial Risk Assessment"}),e.jsx("div",{className:"grid grid-cols-1 lg:grid-cols-2 gap-6"})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Related Analysis"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-6",children:[e.jsx(ne,{to:"/dashboard",className:"block",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow-md hover:shadow-lg transition-shadow duration-200 p-6 border border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600",children:[e.jsxs("div",{className:"flex items-center mb-3",children:[e.jsx("div",{className:"p-2 bg-gray-100 dark:bg-gray-700 rounded-lg mr-4",children:e.jsx("svg",{className:"h-5 w-5 text-gray-600 dark:text-gray-400",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"})})}),e.jsx("h4",{className:"text-lg font-medium text-gray-900 dark:text-white",children:"Overview Dashboard"})]}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300 text-sm",children:"Return to the main overview for quick insights across all metrics."})]})}),e.jsx(ne,{to:"/dashboard/ecological",className:"block",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow-md hover:shadow-lg transition-shadow duration-200 p-6 border border-gray-200 dark:border-gray-700 hover:border-green-300 dark:hover:border-green-600",children:[e.jsxs("div",{className:"flex items-center mb-3",children:[e.jsx("div",{className:"p-2 bg-green-100 dark:bg-green-900 rounded-lg mr-4",children:e.jsx("svg",{className:"h-5 w-5 text-green-600 dark:text-green-400",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4.318 6.318a4.5 4.5 0 000 6.364L12 20.364l7.682-7.682a4.5 4.5 0 00-6.364-6.364L12 7.636l-1.318-1.318a4.5 4.5 0 00-6.364 0z"})})}),e.jsx("h4",{className:"text-lg font-medium text-gray-900 dark:text-white",children:"Ecological Dashboard"})]}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300 text-sm",children:"Explore environmental impact and biodiversity metrics that affect long-term value."})]})})]})]})]})})]})};export{da as FinancialDashboardPage}; diff --git a/frontend/dist/assets/GlobalFilters-B5C4gkva.js b/frontend/dist/assets/GlobalFilters-B5C4gkva.js new file mode 100644 index 0000000000..cae019b911 --- /dev/null +++ b/frontend/dist/assets/GlobalFilters-B5C4gkva.js @@ -0,0 +1,311 @@ +var aa=Object.defineProperty;var oa=(n,a,t)=>a in n?aa(n,a,{enumerable:!0,configurable:!0,writable:!0,value:t}):n[a]=t;var S=(n,a,t)=>oa(n,typeof a!="symbol"?a+"":a,t);import{A as sa,r as v,g as bn,e as ia,j as x,R as g,c as re,h as ca,d as j,i as la,k as Or}from"./index-tEqMizXb.js";const Ke=new sa,ua={getAll:(n={})=>Ke.get("/forests",n),getById:n=>Ke.get(`/forests/${n}`),getAnalytics:n=>Ke.get(`/forests/${n}/analytics`),create:n=>Ke.post("/forests",n),update:(n,a)=>Ke.put(`/forests/${n}`,a),delete:n=>Ke.delete(`/forests/${n}`)},Rr=n=>{if(!n)return null;try{const a=n instanceof Date?n:new Date(n);return isNaN(a.getTime())?null:a.toISOString().split("T")[0]}catch(a){return console.error("Error formatting date:",a),null}},jl=(n={})=>{const a={};if(n.dateRange){const{startDate:o,endDate:s}=n.dateRange;o&&(a.startDate=Rr(o)),s&&(a.endDate=Rr(s))}n.selectedForests&&Array.isArray(n.selectedForests)&&n.selectedForests.length>0&&(a.forestIds=n.selectedForests.join(","));const{dateRange:t,selectedForests:e,...r}=n;return Object.assign(a,r),a},da=(n,a,t)=>{const[e,r]=v.useState(a||null),[o,s]=v.useState(t||null),i=v.useRef(n),c=v.useRef(!1);return v.useEffect(()=>{i.current=n},[n]),v.useEffect(()=>{if(!c.current){c.current=!0;return}const f=setTimeout(()=>{i.current&&e&&o&&i.current({startDate:e,endDate:o})},1e3);return()=>clearTimeout(f)},[e,o]),{startDate:e,endDate:o,handleStartDateChange:f=>{r(f),f&&o&&f>o&&s(f)},handleEndDateChange:f=>{s(f),f&&e&&f{r(null),s(null),i.current&&i.current({startDate:null,endDate:null})}}},kn=6048e5,fa=864e5,qt=6e4,Kt=36e5,pa=1e3,Nr=Symbol.for("constructDateFrom");function U(n,a){return typeof n=="function"?n(a):n&&typeof n=="object"&&Nr in n?n[Nr](a):n instanceof Date?new n.constructor(a):new Date(a)}function P(n,a){return U(a||n,n)}function pe(n,a,t){const e=P(n,t==null?void 0:t.in);return isNaN(a)?U((t==null?void 0:t.in)||n,NaN):(a&&e.setDate(e.getDate()+a),e)}function he(n,a,t){const e=P(n,t==null?void 0:t.in);if(isNaN(a))return U(n,NaN);if(!a)return e;const r=e.getDate(),o=U(n,e.getTime());o.setMonth(e.getMonth()+a+1,0);const s=o.getDate();return r>=s?o:(e.setFullYear(o.getFullYear(),o.getMonth(),r),e)}function xn(n,a,t){return U(n,+P(n)+a)}function ha(n,a,t){return xn(n,a*Kt)}let ma={};function qe(){return ma}function Ee(n,a){var i,c,l,u;const t=qe(),e=(a==null?void 0:a.weekStartsOn)??((c=(i=a==null?void 0:a.locale)==null?void 0:i.options)==null?void 0:c.weekStartsOn)??t.weekStartsOn??((u=(l=t.locale)==null?void 0:l.options)==null?void 0:u.weekStartsOn)??0,r=P(n,a==null?void 0:a.in),o=r.getDay(),s=(o=o.getTime()?e+1:t.getTime()>=i.getTime()?e:e-1}function Nt(n){const a=P(n),t=new Date(Date.UTC(a.getFullYear(),a.getMonth(),a.getDate(),a.getHours(),a.getMinutes(),a.getSeconds(),a.getMilliseconds()));return t.setUTCFullYear(a.getFullYear()),+n-+t}function Pe(n,...a){const t=U.bind(null,a.find(e=>typeof e=="object"));return a.map(t)}function Qe(n,a){const t=P(n,a==null?void 0:a.in);return t.setHours(0,0,0,0),t}function Je(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a),o=Qe(e),s=Qe(r),i=+o-Nt(o),c=+s-Nt(s);return Math.round((i-c)/fa)}function ga(n,a){const t=_n(n,a),e=U(n,0);return e.setFullYear(t,0,4),e.setHours(0,0,0,0),Ze(e)}function ur(n,a,t){const e=P(n,t==null?void 0:t.in);return e.setTime(e.getTime()+a*qt),e}function vr(n,a,t){return he(n,a*3,t)}function va(n,a,t){return xn(n,a*1e3)}function Yt(n,a,t){return pe(n,a*7,t)}function xe(n,a,t){return he(n,a*12,t)}function Yr(n,a){let t,e=a==null?void 0:a.in;return n.forEach(r=>{!e&&typeof r=="object"&&(e=U.bind(null,r));const o=P(r,e);(!t||t{!e&&typeof r=="object"&&(e=U.bind(null,r));const o=P(r,e);(!t||t>o||isNaN(+o))&&(t=o)}),U(e,t||NaN)}function wa(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a);return+Qe(e)==+Qe(r)}function _e(n){return n instanceof Date||typeof n=="object"&&Object.prototype.toString.call(n)==="[object Date]"}function Lt(n){return!(!_e(n)&&typeof n!="number"||isNaN(+P(n)))}function Ft(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a),o=e.getFullYear()-r.getFullYear(),s=e.getMonth()-r.getMonth();return o*12+s}function He(n,a){const t=P(n,a==null?void 0:a.in);return Math.trunc(t.getMonth()/3)+1}function It(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a),o=e.getFullYear()-r.getFullYear(),s=He(e)-He(r);return o*4+s}function At(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a);return e.getFullYear()-r.getFullYear()}function ya(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a),o=Fr(e,r),s=Math.abs(Je(e,r));e.setDate(e.getDate()-o*s);const i=+(Fr(e,r)===-o),c=o*(s-i);return c===0?0:c}function Fr(n,a){const t=n.getFullYear()-a.getFullYear()||n.getMonth()-a.getMonth()||n.getDate()-a.getDate()||n.getHours()-a.getHours()||n.getMinutes()-a.getMinutes()||n.getSeconds()-a.getSeconds()||n.getMilliseconds()-a.getMilliseconds();return t<0?-1:t>0?1:t}function Mn(n,a){const t=P(n,a==null?void 0:a.in);return t.setHours(23,59,59,999),t}function Cn(n,a){const t=P(n,a==null?void 0:a.in),e=t.getMonth();return t.setFullYear(t.getFullYear(),e+1,0),t.setHours(23,59,59,999),t}function dr(n,a){const t=P(n,a==null?void 0:a.in),e=t.getMonth(),r=e-e%3;return t.setMonth(r,1),t.setHours(0,0,0,0),t}function Sn(n,a){const t=P(n,a==null?void 0:a.in);return t.setDate(1),t.setHours(0,0,0,0),t}function En(n,a){const t=P(n,a==null?void 0:a.in),e=t.getFullYear();return t.setFullYear(e+1,0,0),t.setHours(23,59,59,999),t}function Ut(n,a){const t=P(n,a==null?void 0:a.in);return t.setFullYear(t.getFullYear(),0,1),t.setHours(0,0,0,0),t}function Da(n,a){var i,c;const t=qe(),e=t.weekStartsOn??((c=(i=t.locale)==null?void 0:i.options)==null?void 0:c.weekStartsOn)??0,r=P(n,a==null?void 0:a.in),o=r.getDay(),s=(o{let e;const r=ba[n];return typeof r=="string"?e=r:a===1?e=r.one:e=r.other.replace("{{count}}",a.toString()),t!=null&&t.addSuffix?t.comparison&&t.comparison>0?"in "+e:e+" ago":e};function tr(n){return(a={})=>{const t=a.width?String(a.width):n.defaultWidth;return n.formats[t]||n.formats[n.defaultWidth]}}const xa={full:"EEEE, MMMM do, y",long:"MMMM do, y",medium:"MMM d, y",short:"MM/dd/yyyy"},_a={full:"h:mm:ss a zzzz",long:"h:mm:ss a z",medium:"h:mm:ss a",short:"h:mm a"},Ma={full:"{{date}} 'at' {{time}}",long:"{{date}} 'at' {{time}}",medium:"{{date}}, {{time}}",short:"{{date}}, {{time}}"},Ca={date:tr({formats:xa,defaultWidth:"full"}),time:tr({formats:_a,defaultWidth:"full"}),dateTime:tr({formats:Ma,defaultWidth:"full"})},Sa={lastWeek:"'last' eeee 'at' p",yesterday:"'yesterday at' p",today:"'today at' p",tomorrow:"'tomorrow at' p",nextWeek:"eeee 'at' p",other:"P"},Ea=(n,a,t,e)=>Sa[n];function st(n){return(a,t)=>{const e=t!=null&&t.context?String(t.context):"standalone";let r;if(e==="formatting"&&n.formattingValues){const s=n.defaultFormattingWidth||n.defaultWidth,i=t!=null&&t.width?String(t.width):s;r=n.formattingValues[i]||n.formattingValues[s]}else{const s=n.defaultWidth,i=t!=null&&t.width?String(t.width):n.defaultWidth;r=n.values[i]||n.values[s]}const o=n.argumentCallback?n.argumentCallback(a):a;return r[o]}}const Pa={narrow:["B","A"],abbreviated:["BC","AD"],wide:["Before Christ","Anno Domini"]},Ta={narrow:["1","2","3","4"],abbreviated:["Q1","Q2","Q3","Q4"],wide:["1st quarter","2nd quarter","3rd quarter","4th quarter"]},Oa={narrow:["J","F","M","A","M","J","J","A","S","O","N","D"],abbreviated:["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"],wide:["January","February","March","April","May","June","July","August","September","October","November","December"]},Ra={narrow:["S","M","T","W","T","F","S"],short:["Su","Mo","Tu","We","Th","Fr","Sa"],abbreviated:["Sun","Mon","Tue","Wed","Thu","Fri","Sat"],wide:["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"]},Na={narrow:{am:"a",pm:"p",midnight:"mi",noon:"n",morning:"morning",afternoon:"afternoon",evening:"evening",night:"night"},abbreviated:{am:"AM",pm:"PM",midnight:"midnight",noon:"noon",morning:"morning",afternoon:"afternoon",evening:"evening",night:"night"},wide:{am:"a.m.",pm:"p.m.",midnight:"midnight",noon:"noon",morning:"morning",afternoon:"afternoon",evening:"evening",night:"night"}},Ya={narrow:{am:"a",pm:"p",midnight:"mi",noon:"n",morning:"in the morning",afternoon:"in the afternoon",evening:"in the evening",night:"at night"},abbreviated:{am:"AM",pm:"PM",midnight:"midnight",noon:"noon",morning:"in the morning",afternoon:"in the afternoon",evening:"in the evening",night:"at night"},wide:{am:"a.m.",pm:"p.m.",midnight:"midnight",noon:"noon",morning:"in the morning",afternoon:"in the afternoon",evening:"in the evening",night:"at night"}},La=(n,a)=>{const t=Number(n),e=t%100;if(e>20||e<10)switch(e%10){case 1:return t+"st";case 2:return t+"nd";case 3:return t+"rd"}return t+"th"},Fa={ordinalNumber:La,era:st({values:Pa,defaultWidth:"wide"}),quarter:st({values:Ta,defaultWidth:"wide",argumentCallback:n=>n-1}),month:st({values:Oa,defaultWidth:"wide"}),day:st({values:Ra,defaultWidth:"wide"}),dayPeriod:st({values:Na,defaultWidth:"wide",formattingValues:Ya,defaultFormattingWidth:"wide"})};function it(n){return(a,t={})=>{const e=t.width,r=e&&n.matchPatterns[e]||n.matchPatterns[n.defaultMatchWidth],o=a.match(r);if(!o)return null;const s=o[0],i=e&&n.parsePatterns[e]||n.parsePatterns[n.defaultParseWidth],c=Array.isArray(i)?Aa(i,d=>d.test(s)):Ia(i,d=>d.test(s));let l;l=n.valueCallback?n.valueCallback(c):c,l=t.valueCallback?t.valueCallback(l):l;const u=a.slice(s.length);return{value:l,rest:u}}}function Ia(n,a){for(const t in n)if(Object.prototype.hasOwnProperty.call(n,t)&&a(n[t]))return t}function Aa(n,a){for(let t=0;t{const e=a.match(n.matchPattern);if(!e)return null;const r=e[0],o=a.match(n.parsePattern);if(!o)return null;let s=n.valueCallback?n.valueCallback(o[0]):o[0];s=t.valueCallback?t.valueCallback(s):s;const i=a.slice(r.length);return{value:s,rest:i}}}const Ha=/^(\d+)(th|st|nd|rd)?/i,ja=/\d+/i,Ba={narrow:/^(b|a)/i,abbreviated:/^(b\.?\s?c\.?|b\.?\s?c\.?\s?e\.?|a\.?\s?d\.?|c\.?\s?e\.?)/i,wide:/^(before christ|before common era|anno domini|common era)/i},Qa={any:[/^b/i,/^(a|c)/i]},$a={narrow:/^[1234]/i,abbreviated:/^q[1234]/i,wide:/^[1234](th|st|nd|rd)? quarter/i},Va={any:[/1/i,/2/i,/3/i,/4/i]},qa={narrow:/^[jfmasond]/i,abbreviated:/^(jan|feb|mar|apr|may|jun|jul|aug|sep|oct|nov|dec)/i,wide:/^(january|february|march|april|may|june|july|august|september|october|november|december)/i},Ka={narrow:[/^j/i,/^f/i,/^m/i,/^a/i,/^m/i,/^j/i,/^j/i,/^a/i,/^s/i,/^o/i,/^n/i,/^d/i],any:[/^ja/i,/^f/i,/^mar/i,/^ap/i,/^may/i,/^jun/i,/^jul/i,/^au/i,/^s/i,/^o/i,/^n/i,/^d/i]},Ua={narrow:/^[smtwf]/i,short:/^(su|mo|tu|we|th|fr|sa)/i,abbreviated:/^(sun|mon|tue|wed|thu|fri|sat)/i,wide:/^(sunday|monday|tuesday|wednesday|thursday|friday|saturday)/i},Ga={narrow:[/^s/i,/^m/i,/^t/i,/^w/i,/^t/i,/^f/i,/^s/i],any:[/^su/i,/^m/i,/^tu/i,/^w/i,/^th/i,/^f/i,/^sa/i]},za={narrow:/^(a|p|mi|n|(in the|at) (morning|afternoon|evening|night))/i,any:/^([ap]\.?\s?m\.?|midnight|noon|(in the|at) (morning|afternoon|evening|night))/i},Xa={any:{am:/^a/i,pm:/^p/i,midnight:/^mi/i,noon:/^no/i,morning:/morning/i,afternoon:/afternoon/i,evening:/evening/i,night:/night/i}},Za={ordinalNumber:Wa({matchPattern:Ha,parsePattern:ja,valueCallback:n=>parseInt(n,10)}),era:it({matchPatterns:Ba,defaultMatchWidth:"wide",parsePatterns:Qa,defaultParseWidth:"any"}),quarter:it({matchPatterns:$a,defaultMatchWidth:"wide",parsePatterns:Va,defaultParseWidth:"any",valueCallback:n=>n+1}),month:it({matchPatterns:qa,defaultMatchWidth:"wide",parsePatterns:Ka,defaultParseWidth:"any"}),day:it({matchPatterns:Ua,defaultMatchWidth:"wide",parsePatterns:Ga,defaultParseWidth:"any"}),dayPeriod:it({matchPatterns:za,defaultMatchWidth:"any",parsePatterns:Xa,defaultParseWidth:"any"})},Pn={code:"en-US",formatDistance:ka,formatLong:Ca,formatRelative:Ea,localize:Fa,match:Za,options:{weekStartsOn:0,firstWeekContainsDate:1}};function Ja(n,a){const t=P(n,a==null?void 0:a.in);return Je(t,Ut(t))+1}function wr(n,a){const t=P(n,a==null?void 0:a.in),e=+Ze(t)-+ga(t);return Math.round(e/kn)+1}function yr(n,a){var u,d,f,p;const t=P(n,a==null?void 0:a.in),e=t.getFullYear(),r=qe(),o=(a==null?void 0:a.firstWeekContainsDate)??((d=(u=a==null?void 0:a.locale)==null?void 0:u.options)==null?void 0:d.firstWeekContainsDate)??r.firstWeekContainsDate??((p=(f=r.locale)==null?void 0:f.options)==null?void 0:p.firstWeekContainsDate)??1,s=U((a==null?void 0:a.in)||n,0);s.setFullYear(e+1,0,o),s.setHours(0,0,0,0);const i=Ee(s,a),c=U((a==null?void 0:a.in)||n,0);c.setFullYear(e,0,o),c.setHours(0,0,0,0);const l=Ee(c,a);return+t>=+i?e+1:+t>=+l?e:e-1}function eo(n,a){var i,c,l,u;const t=qe(),e=(a==null?void 0:a.firstWeekContainsDate)??((c=(i=a==null?void 0:a.locale)==null?void 0:i.options)==null?void 0:c.firstWeekContainsDate)??t.firstWeekContainsDate??((u=(l=t.locale)==null?void 0:l.options)==null?void 0:u.firstWeekContainsDate)??1,r=yr(n,a),o=U((a==null?void 0:a.in)||n,0);return o.setFullYear(r,0,e),o.setHours(0,0,0,0),Ee(o,a)}function Tn(n,a){const t=P(n,a==null?void 0:a.in),e=+Ee(t,a)-+eo(t,a);return Math.round(e/kn)+1}function K(n,a){const t=n<0?"-":"",e=Math.abs(n).toString().padStart(a,"0");return t+e}const Te={y(n,a){const t=n.getFullYear(),e=t>0?t:1-t;return K(a==="yy"?e%100:e,a.length)},M(n,a){const t=n.getMonth();return a==="M"?String(t+1):K(t+1,2)},d(n,a){return K(n.getDate(),a.length)},a(n,a){const t=n.getHours()/12>=1?"pm":"am";switch(a){case"a":case"aa":return t.toUpperCase();case"aaa":return t;case"aaaaa":return t[0];case"aaaa":default:return t==="am"?"a.m.":"p.m."}},h(n,a){return K(n.getHours()%12||12,a.length)},H(n,a){return K(n.getHours(),a.length)},m(n,a){return K(n.getMinutes(),a.length)},s(n,a){return K(n.getSeconds(),a.length)},S(n,a){const t=a.length,e=n.getMilliseconds(),r=Math.trunc(e*Math.pow(10,t-3));return K(r,a.length)}},Ue={midnight:"midnight",noon:"noon",morning:"morning",afternoon:"afternoon",evening:"evening",night:"night"},Ir={G:function(n,a,t){const e=n.getFullYear()>0?1:0;switch(a){case"G":case"GG":case"GGG":return t.era(e,{width:"abbreviated"});case"GGGGG":return t.era(e,{width:"narrow"});case"GGGG":default:return t.era(e,{width:"wide"})}},y:function(n,a,t){if(a==="yo"){const e=n.getFullYear(),r=e>0?e:1-e;return t.ordinalNumber(r,{unit:"year"})}return Te.y(n,a)},Y:function(n,a,t,e){const r=yr(n,e),o=r>0?r:1-r;if(a==="YY"){const s=o%100;return K(s,2)}return a==="Yo"?t.ordinalNumber(o,{unit:"year"}):K(o,a.length)},R:function(n,a){const t=_n(n);return K(t,a.length)},u:function(n,a){const t=n.getFullYear();return K(t,a.length)},Q:function(n,a,t){const e=Math.ceil((n.getMonth()+1)/3);switch(a){case"Q":return String(e);case"QQ":return K(e,2);case"Qo":return t.ordinalNumber(e,{unit:"quarter"});case"QQQ":return t.quarter(e,{width:"abbreviated",context:"formatting"});case"QQQQQ":return t.quarter(e,{width:"narrow",context:"formatting"});case"QQQQ":default:return t.quarter(e,{width:"wide",context:"formatting"})}},q:function(n,a,t){const e=Math.ceil((n.getMonth()+1)/3);switch(a){case"q":return String(e);case"qq":return K(e,2);case"qo":return t.ordinalNumber(e,{unit:"quarter"});case"qqq":return t.quarter(e,{width:"abbreviated",context:"standalone"});case"qqqqq":return t.quarter(e,{width:"narrow",context:"standalone"});case"qqqq":default:return t.quarter(e,{width:"wide",context:"standalone"})}},M:function(n,a,t){const e=n.getMonth();switch(a){case"M":case"MM":return Te.M(n,a);case"Mo":return t.ordinalNumber(e+1,{unit:"month"});case"MMM":return t.month(e,{width:"abbreviated",context:"formatting"});case"MMMMM":return t.month(e,{width:"narrow",context:"formatting"});case"MMMM":default:return t.month(e,{width:"wide",context:"formatting"})}},L:function(n,a,t){const e=n.getMonth();switch(a){case"L":return String(e+1);case"LL":return K(e+1,2);case"Lo":return t.ordinalNumber(e+1,{unit:"month"});case"LLL":return t.month(e,{width:"abbreviated",context:"standalone"});case"LLLLL":return t.month(e,{width:"narrow",context:"standalone"});case"LLLL":default:return t.month(e,{width:"wide",context:"standalone"})}},w:function(n,a,t,e){const r=Tn(n,e);return a==="wo"?t.ordinalNumber(r,{unit:"week"}):K(r,a.length)},I:function(n,a,t){const e=wr(n);return a==="Io"?t.ordinalNumber(e,{unit:"week"}):K(e,a.length)},d:function(n,a,t){return a==="do"?t.ordinalNumber(n.getDate(),{unit:"date"}):Te.d(n,a)},D:function(n,a,t){const e=Ja(n);return a==="Do"?t.ordinalNumber(e,{unit:"dayOfYear"}):K(e,a.length)},E:function(n,a,t){const e=n.getDay();switch(a){case"E":case"EE":case"EEE":return t.day(e,{width:"abbreviated",context:"formatting"});case"EEEEE":return t.day(e,{width:"narrow",context:"formatting"});case"EEEEEE":return t.day(e,{width:"short",context:"formatting"});case"EEEE":default:return t.day(e,{width:"wide",context:"formatting"})}},e:function(n,a,t,e){const r=n.getDay(),o=(r-e.weekStartsOn+8)%7||7;switch(a){case"e":return String(o);case"ee":return K(o,2);case"eo":return t.ordinalNumber(o,{unit:"day"});case"eee":return t.day(r,{width:"abbreviated",context:"formatting"});case"eeeee":return t.day(r,{width:"narrow",context:"formatting"});case"eeeeee":return t.day(r,{width:"short",context:"formatting"});case"eeee":default:return t.day(r,{width:"wide",context:"formatting"})}},c:function(n,a,t,e){const r=n.getDay(),o=(r-e.weekStartsOn+8)%7||7;switch(a){case"c":return String(o);case"cc":return K(o,a.length);case"co":return t.ordinalNumber(o,{unit:"day"});case"ccc":return t.day(r,{width:"abbreviated",context:"standalone"});case"ccccc":return t.day(r,{width:"narrow",context:"standalone"});case"cccccc":return t.day(r,{width:"short",context:"standalone"});case"cccc":default:return t.day(r,{width:"wide",context:"standalone"})}},i:function(n,a,t){const e=n.getDay(),r=e===0?7:e;switch(a){case"i":return String(r);case"ii":return K(r,a.length);case"io":return t.ordinalNumber(r,{unit:"day"});case"iii":return t.day(e,{width:"abbreviated",context:"formatting"});case"iiiii":return t.day(e,{width:"narrow",context:"formatting"});case"iiiiii":return t.day(e,{width:"short",context:"formatting"});case"iiii":default:return t.day(e,{width:"wide",context:"formatting"})}},a:function(n,a,t){const r=n.getHours()/12>=1?"pm":"am";switch(a){case"a":case"aa":return t.dayPeriod(r,{width:"abbreviated",context:"formatting"});case"aaa":return t.dayPeriod(r,{width:"abbreviated",context:"formatting"}).toLowerCase();case"aaaaa":return t.dayPeriod(r,{width:"narrow",context:"formatting"});case"aaaa":default:return t.dayPeriod(r,{width:"wide",context:"formatting"})}},b:function(n,a,t){const e=n.getHours();let r;switch(e===12?r=Ue.noon:e===0?r=Ue.midnight:r=e/12>=1?"pm":"am",a){case"b":case"bb":return t.dayPeriod(r,{width:"abbreviated",context:"formatting"});case"bbb":return t.dayPeriod(r,{width:"abbreviated",context:"formatting"}).toLowerCase();case"bbbbb":return t.dayPeriod(r,{width:"narrow",context:"formatting"});case"bbbb":default:return t.dayPeriod(r,{width:"wide",context:"formatting"})}},B:function(n,a,t){const e=n.getHours();let r;switch(e>=17?r=Ue.evening:e>=12?r=Ue.afternoon:e>=4?r=Ue.morning:r=Ue.night,a){case"B":case"BB":case"BBB":return t.dayPeriod(r,{width:"abbreviated",context:"formatting"});case"BBBBB":return t.dayPeriod(r,{width:"narrow",context:"formatting"});case"BBBB":default:return t.dayPeriod(r,{width:"wide",context:"formatting"})}},h:function(n,a,t){if(a==="ho"){let e=n.getHours()%12;return e===0&&(e=12),t.ordinalNumber(e,{unit:"hour"})}return Te.h(n,a)},H:function(n,a,t){return a==="Ho"?t.ordinalNumber(n.getHours(),{unit:"hour"}):Te.H(n,a)},K:function(n,a,t){const e=n.getHours()%12;return a==="Ko"?t.ordinalNumber(e,{unit:"hour"}):K(e,a.length)},k:function(n,a,t){let e=n.getHours();return e===0&&(e=24),a==="ko"?t.ordinalNumber(e,{unit:"hour"}):K(e,a.length)},m:function(n,a,t){return a==="mo"?t.ordinalNumber(n.getMinutes(),{unit:"minute"}):Te.m(n,a)},s:function(n,a,t){return a==="so"?t.ordinalNumber(n.getSeconds(),{unit:"second"}):Te.s(n,a)},S:function(n,a){return Te.S(n,a)},X:function(n,a,t){const e=n.getTimezoneOffset();if(e===0)return"Z";switch(a){case"X":return Wr(e);case"XXXX":case"XX":return Ie(e);case"XXXXX":case"XXX":default:return Ie(e,":")}},x:function(n,a,t){const e=n.getTimezoneOffset();switch(a){case"x":return Wr(e);case"xxxx":case"xx":return Ie(e);case"xxxxx":case"xxx":default:return Ie(e,":")}},O:function(n,a,t){const e=n.getTimezoneOffset();switch(a){case"O":case"OO":case"OOO":return"GMT"+Ar(e,":");case"OOOO":default:return"GMT"+Ie(e,":")}},z:function(n,a,t){const e=n.getTimezoneOffset();switch(a){case"z":case"zz":case"zzz":return"GMT"+Ar(e,":");case"zzzz":default:return"GMT"+Ie(e,":")}},t:function(n,a,t){const e=Math.trunc(+n/1e3);return K(e,a.length)},T:function(n,a,t){return K(+n,a.length)}};function Ar(n,a=""){const t=n>0?"-":"+",e=Math.abs(n),r=Math.trunc(e/60),o=e%60;return o===0?t+String(r):t+String(r)+a+K(o,2)}function Wr(n,a){return n%60===0?(n>0?"-":"+")+K(Math.abs(n)/60,2):Ie(n,a)}function Ie(n,a=""){const t=n>0?"-":"+",e=Math.abs(n),r=K(Math.trunc(e/60),2),o=K(e%60,2);return t+r+a+o}const Hr=(n,a)=>{switch(n){case"P":return a.date({width:"short"});case"PP":return a.date({width:"medium"});case"PPP":return a.date({width:"long"});case"PPPP":default:return a.date({width:"full"})}},On=(n,a)=>{switch(n){case"p":return a.time({width:"short"});case"pp":return a.time({width:"medium"});case"ppp":return a.time({width:"long"});case"pppp":default:return a.time({width:"full"})}},to=(n,a)=>{const t=n.match(/(P+)(p+)?/)||[],e=t[1],r=t[2];if(!r)return Hr(n,a);let o;switch(e){case"P":o=a.dateTime({width:"short"});break;case"PP":o=a.dateTime({width:"medium"});break;case"PPP":o=a.dateTime({width:"long"});break;case"PPPP":default:o=a.dateTime({width:"full"});break}return o.replace("{{date}}",Hr(e,a)).replace("{{time}}",On(r,a))},fr={p:On,P:to},ro=/^D+$/,no=/^Y+$/,ao=["D","DD","YY","YYYY"];function oo(n){return ro.test(n)}function so(n){return no.test(n)}function io(n,a,t){const e=co(n,a,t);if(console.warn(e),ao.includes(n))throw new RangeError(e)}function co(n,a,t){const e=n[0]==="Y"?"years":"days of the month";return`Use \`${n.toLowerCase()}\` instead of \`${n}\` (in \`${a}\`) for formatting ${e} to the input \`${t}\`; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md`}const lo=/[yYQqMLwIdDecihHKkms]o|(\w)\1*|''|'(''|[^'])+('|$)|./g,uo=/P+p+|P+|p+|''|'(''|[^'])+('|$)|./g,fo=/^'([^]*?)'?$/,po=/''/g,ho=/[a-zA-Z]/;function jr(n,a,t){var u,d,f,p,h,w,m,y;const e=qe(),r=(t==null?void 0:t.locale)??e.locale??Pn,o=(t==null?void 0:t.firstWeekContainsDate)??((d=(u=t==null?void 0:t.locale)==null?void 0:u.options)==null?void 0:d.firstWeekContainsDate)??e.firstWeekContainsDate??((p=(f=e.locale)==null?void 0:f.options)==null?void 0:p.firstWeekContainsDate)??1,s=(t==null?void 0:t.weekStartsOn)??((w=(h=t==null?void 0:t.locale)==null?void 0:h.options)==null?void 0:w.weekStartsOn)??e.weekStartsOn??((y=(m=e.locale)==null?void 0:m.options)==null?void 0:y.weekStartsOn)??0,i=P(n,t==null?void 0:t.in);if(!Lt(i))throw new RangeError("Invalid time value");let c=a.match(uo).map(_=>{const k=_[0];if(k==="p"||k==="P"){const E=fr[k];return E(_,r.formatLong)}return _}).join("").match(lo).map(_=>{if(_==="''")return{isToken:!1,value:"'"};const k=_[0];if(k==="'")return{isToken:!1,value:mo(_)};if(Ir[k])return{isToken:!0,value:_};if(k.match(ho))throw new RangeError("Format string contains an unescaped latin alphabet character `"+k+"`");return{isToken:!1,value:_}});r.localize.preprocessor&&(c=r.localize.preprocessor(i,c));const l={firstWeekContainsDate:o,weekStartsOn:s,locale:r};return c.map(_=>{if(!_.isToken)return _.value;const k=_.value;(!(t!=null&&t.useAdditionalWeekYearTokens)&&so(k)||!(t!=null&&t.useAdditionalDayOfYearTokens)&&oo(k))&&io(k,a,String(n));const E=Ir[k[0]];return E(i,k,r.localize,l)}).join("")}function mo(n){const a=n.match(fo);return a?a[1].replace(po,"'"):n}function Br(n,a){return P(n,a==null?void 0:a.in).getDate()}function go(n,a){return P(n,a==null?void 0:a.in).getDay()}function vo(n,a){const t=P(n,a==null?void 0:a.in),e=t.getFullYear(),r=t.getMonth(),o=U(t,0);return o.setFullYear(e,r+1,0),o.setHours(0,0,0,0),o.getDate()}function wo(){return Object.assign({},qe())}function Me(n,a){return P(n,a==null?void 0:a.in).getHours()}function yo(n,a){const t=P(n,a==null?void 0:a.in).getDay();return t===0?7:t}function Ce(n,a){return P(n,a==null?void 0:a.in).getMinutes()}function ne(n,a){return P(n,a==null?void 0:a.in).getMonth()}function Ne(n){return P(n).getSeconds()}function pr(n){return+P(n)}function I(n,a){return P(n,a==null?void 0:a.in).getFullYear()}function Le(n,a){return+P(n)>+P(a)}function $e(n,a){return+P(n)<+P(a)}function Do(n,a){return+P(n)==+P(a)}function bo(n,a){const t=ko(a)?new a(0):U(a,0);return t.setFullYear(n.getFullYear(),n.getMonth(),n.getDate()),t.setHours(n.getHours(),n.getMinutes(),n.getSeconds(),n.getMilliseconds()),t}function ko(n){var a;return typeof n=="function"&&((a=n.prototype)==null?void 0:a.constructor)===n}const xo=10;class Rn{constructor(){S(this,"subPriority",0)}validate(a,t){return!0}}class _o extends Rn{constructor(a,t,e,r,o){super(),this.value=a,this.validateValue=t,this.setValue=e,this.priority=r,o&&(this.subPriority=o)}validate(a,t){return this.validateValue(a,this.value,t)}set(a,t,e){return this.setValue(a,t,this.value,e)}}class Mo extends Rn{constructor(t,e){super();S(this,"priority",xo);S(this,"subPriority",-1);this.context=t||(r=>U(e,r))}set(t,e){return e.timestampIsSet?t:U(t,bo(t,this.context))}}class ${run(a,t,e,r){const o=this.parse(a,t,e,r);return o?{setter:new _o(o.value,this.validate,this.set,this.priority,this.subPriority),rest:o.rest}:null}validate(a,t,e){return!0}}class Co extends ${constructor(){super(...arguments);S(this,"priority",140);S(this,"incompatibleTokens",["R","u","t","T"])}parse(t,e,r){switch(e){case"G":case"GG":case"GGG":return r.era(t,{width:"abbreviated"})||r.era(t,{width:"narrow"});case"GGGGG":return r.era(t,{width:"narrow"});case"GGGG":default:return r.era(t,{width:"wide"})||r.era(t,{width:"abbreviated"})||r.era(t,{width:"narrow"})}}set(t,e,r){return e.era=r,t.setFullYear(r,0,1),t.setHours(0,0,0,0),t}}const Z={month:/^(1[0-2]|0?\d)/,date:/^(3[0-1]|[0-2]?\d)/,dayOfYear:/^(36[0-6]|3[0-5]\d|[0-2]?\d?\d)/,week:/^(5[0-3]|[0-4]?\d)/,hour23h:/^(2[0-3]|[0-1]?\d)/,hour24h:/^(2[0-4]|[0-1]?\d)/,hour11h:/^(1[0-1]|0?\d)/,hour12h:/^(1[0-2]|0?\d)/,minute:/^[0-5]?\d/,second:/^[0-5]?\d/,singleDigit:/^\d/,twoDigits:/^\d{1,2}/,threeDigits:/^\d{1,3}/,fourDigits:/^\d{1,4}/,anyDigitsSigned:/^-?\d+/,singleDigitSigned:/^-?\d/,twoDigitsSigned:/^-?\d{1,2}/,threeDigitsSigned:/^-?\d{1,3}/,fourDigitsSigned:/^-?\d{1,4}/},ve={basicOptionalMinutes:/^([+-])(\d{2})(\d{2})?|Z/,basic:/^([+-])(\d{2})(\d{2})|Z/,basicOptionalSeconds:/^([+-])(\d{2})(\d{2})((\d{2}))?|Z/,extended:/^([+-])(\d{2}):(\d{2})|Z/,extendedOptionalSeconds:/^([+-])(\d{2}):(\d{2})(:(\d{2}))?|Z/};function J(n,a){return n&&{value:a(n.value),rest:n.rest}}function z(n,a){const t=a.match(n);return t?{value:parseInt(t[0],10),rest:a.slice(t[0].length)}:null}function we(n,a){const t=a.match(n);if(!t)return null;if(t[0]==="Z")return{value:0,rest:a.slice(1)};const e=t[1]==="+"?1:-1,r=t[2]?parseInt(t[2],10):0,o=t[3]?parseInt(t[3],10):0,s=t[5]?parseInt(t[5],10):0;return{value:e*(r*Kt+o*qt+s*pa),rest:a.slice(t[0].length)}}function Nn(n){return z(Z.anyDigitsSigned,n)}function X(n,a){switch(n){case 1:return z(Z.singleDigit,a);case 2:return z(Z.twoDigits,a);case 3:return z(Z.threeDigits,a);case 4:return z(Z.fourDigits,a);default:return z(new RegExp("^\\d{1,"+n+"}"),a)}}function Wt(n,a){switch(n){case 1:return z(Z.singleDigitSigned,a);case 2:return z(Z.twoDigitsSigned,a);case 3:return z(Z.threeDigitsSigned,a);case 4:return z(Z.fourDigitsSigned,a);default:return z(new RegExp("^-?\\d{1,"+n+"}"),a)}}function Dr(n){switch(n){case"morning":return 4;case"evening":return 17;case"pm":case"noon":case"afternoon":return 12;case"am":case"midnight":case"night":default:return 0}}function Yn(n,a){const t=a>0,e=t?a:1-a;let r;if(e<=50)r=n||100;else{const o=e+50,s=Math.trunc(o/100)*100,i=n>=o%100;r=n+s-(i?100:0)}return t?r:1-r}function Ln(n){return n%400===0||n%4===0&&n%100!==0}class So extends ${constructor(){super(...arguments);S(this,"priority",130);S(this,"incompatibleTokens",["Y","R","u","w","I","i","e","c","t","T"])}parse(t,e,r){const o=s=>({year:s,isTwoDigitYear:e==="yy"});switch(e){case"y":return J(X(4,t),o);case"yo":return J(r.ordinalNumber(t,{unit:"year"}),o);default:return J(X(e.length,t),o)}}validate(t,e){return e.isTwoDigitYear||e.year>0}set(t,e,r){const o=t.getFullYear();if(r.isTwoDigitYear){const i=Yn(r.year,o);return t.setFullYear(i,0,1),t.setHours(0,0,0,0),t}const s=!("era"in e)||e.era===1?r.year:1-r.year;return t.setFullYear(s,0,1),t.setHours(0,0,0,0),t}}class Eo extends ${constructor(){super(...arguments);S(this,"priority",130);S(this,"incompatibleTokens",["y","R","u","Q","q","M","L","I","d","D","i","t","T"])}parse(t,e,r){const o=s=>({year:s,isTwoDigitYear:e==="YY"});switch(e){case"Y":return J(X(4,t),o);case"Yo":return J(r.ordinalNumber(t,{unit:"year"}),o);default:return J(X(e.length,t),o)}}validate(t,e){return e.isTwoDigitYear||e.year>0}set(t,e,r,o){const s=yr(t,o);if(r.isTwoDigitYear){const c=Yn(r.year,s);return t.setFullYear(c,0,o.firstWeekContainsDate),t.setHours(0,0,0,0),Ee(t,o)}const i=!("era"in e)||e.era===1?r.year:1-r.year;return t.setFullYear(i,0,o.firstWeekContainsDate),t.setHours(0,0,0,0),Ee(t,o)}}class Po extends ${constructor(){super(...arguments);S(this,"priority",130);S(this,"incompatibleTokens",["G","y","Y","u","Q","q","M","L","w","d","D","e","c","t","T"])}parse(t,e){return Wt(e==="R"?4:e.length,t)}set(t,e,r){const o=U(t,0);return o.setFullYear(r,0,4),o.setHours(0,0,0,0),Ze(o)}}class To extends ${constructor(){super(...arguments);S(this,"priority",130);S(this,"incompatibleTokens",["G","y","Y","R","w","I","i","e","c","t","T"])}parse(t,e){return Wt(e==="u"?4:e.length,t)}set(t,e,r){return t.setFullYear(r,0,1),t.setHours(0,0,0,0),t}}class Oo extends ${constructor(){super(...arguments);S(this,"priority",120);S(this,"incompatibleTokens",["Y","R","q","M","L","w","I","d","D","i","e","c","t","T"])}parse(t,e,r){switch(e){case"Q":case"QQ":return X(e.length,t);case"Qo":return r.ordinalNumber(t,{unit:"quarter"});case"QQQ":return r.quarter(t,{width:"abbreviated",context:"formatting"})||r.quarter(t,{width:"narrow",context:"formatting"});case"QQQQQ":return r.quarter(t,{width:"narrow",context:"formatting"});case"QQQQ":default:return r.quarter(t,{width:"wide",context:"formatting"})||r.quarter(t,{width:"abbreviated",context:"formatting"})||r.quarter(t,{width:"narrow",context:"formatting"})}}validate(t,e){return e>=1&&e<=4}set(t,e,r){return t.setMonth((r-1)*3,1),t.setHours(0,0,0,0),t}}class Ro extends ${constructor(){super(...arguments);S(this,"priority",120);S(this,"incompatibleTokens",["Y","R","Q","M","L","w","I","d","D","i","e","c","t","T"])}parse(t,e,r){switch(e){case"q":case"qq":return X(e.length,t);case"qo":return r.ordinalNumber(t,{unit:"quarter"});case"qqq":return r.quarter(t,{width:"abbreviated",context:"standalone"})||r.quarter(t,{width:"narrow",context:"standalone"});case"qqqqq":return r.quarter(t,{width:"narrow",context:"standalone"});case"qqqq":default:return r.quarter(t,{width:"wide",context:"standalone"})||r.quarter(t,{width:"abbreviated",context:"standalone"})||r.quarter(t,{width:"narrow",context:"standalone"})}}validate(t,e){return e>=1&&e<=4}set(t,e,r){return t.setMonth((r-1)*3,1),t.setHours(0,0,0,0),t}}class No extends ${constructor(){super(...arguments);S(this,"incompatibleTokens",["Y","R","q","Q","L","w","I","D","i","e","c","t","T"]);S(this,"priority",110)}parse(t,e,r){const o=s=>s-1;switch(e){case"M":return J(z(Z.month,t),o);case"MM":return J(X(2,t),o);case"Mo":return J(r.ordinalNumber(t,{unit:"month"}),o);case"MMM":return r.month(t,{width:"abbreviated",context:"formatting"})||r.month(t,{width:"narrow",context:"formatting"});case"MMMMM":return r.month(t,{width:"narrow",context:"formatting"});case"MMMM":default:return r.month(t,{width:"wide",context:"formatting"})||r.month(t,{width:"abbreviated",context:"formatting"})||r.month(t,{width:"narrow",context:"formatting"})}}validate(t,e){return e>=0&&e<=11}set(t,e,r){return t.setMonth(r,1),t.setHours(0,0,0,0),t}}class Yo extends ${constructor(){super(...arguments);S(this,"priority",110);S(this,"incompatibleTokens",["Y","R","q","Q","M","w","I","D","i","e","c","t","T"])}parse(t,e,r){const o=s=>s-1;switch(e){case"L":return J(z(Z.month,t),o);case"LL":return J(X(2,t),o);case"Lo":return J(r.ordinalNumber(t,{unit:"month"}),o);case"LLL":return r.month(t,{width:"abbreviated",context:"standalone"})||r.month(t,{width:"narrow",context:"standalone"});case"LLLLL":return r.month(t,{width:"narrow",context:"standalone"});case"LLLL":default:return r.month(t,{width:"wide",context:"standalone"})||r.month(t,{width:"abbreviated",context:"standalone"})||r.month(t,{width:"narrow",context:"standalone"})}}validate(t,e){return e>=0&&e<=11}set(t,e,r){return t.setMonth(r,1),t.setHours(0,0,0,0),t}}function Lo(n,a,t){const e=P(n,t==null?void 0:t.in),r=Tn(e,t)-a;return e.setDate(e.getDate()-r*7),P(e,t==null?void 0:t.in)}class Fo extends ${constructor(){super(...arguments);S(this,"priority",100);S(this,"incompatibleTokens",["y","R","u","q","Q","M","L","I","d","D","i","t","T"])}parse(t,e,r){switch(e){case"w":return z(Z.week,t);case"wo":return r.ordinalNumber(t,{unit:"week"});default:return X(e.length,t)}}validate(t,e){return e>=1&&e<=53}set(t,e,r,o){return Ee(Lo(t,r,o),o)}}function Io(n,a,t){const e=P(n,t==null?void 0:t.in),r=wr(e,t)-a;return e.setDate(e.getDate()-r*7),e}class Ao extends ${constructor(){super(...arguments);S(this,"priority",100);S(this,"incompatibleTokens",["y","Y","u","q","Q","M","L","w","d","D","e","c","t","T"])}parse(t,e,r){switch(e){case"I":return z(Z.week,t);case"Io":return r.ordinalNumber(t,{unit:"week"});default:return X(e.length,t)}}validate(t,e){return e>=1&&e<=53}set(t,e,r){return Ze(Io(t,r))}}const Wo=[31,28,31,30,31,30,31,31,30,31,30,31],Ho=[31,29,31,30,31,30,31,31,30,31,30,31];class jo extends ${constructor(){super(...arguments);S(this,"priority",90);S(this,"subPriority",1);S(this,"incompatibleTokens",["Y","R","q","Q","w","I","D","i","e","c","t","T"])}parse(t,e,r){switch(e){case"d":return z(Z.date,t);case"do":return r.ordinalNumber(t,{unit:"date"});default:return X(e.length,t)}}validate(t,e){const r=t.getFullYear(),o=Ln(r),s=t.getMonth();return o?e>=1&&e<=Ho[s]:e>=1&&e<=Wo[s]}set(t,e,r){return t.setDate(r),t.setHours(0,0,0,0),t}}class Bo extends ${constructor(){super(...arguments);S(this,"priority",90);S(this,"subpriority",1);S(this,"incompatibleTokens",["Y","R","q","Q","M","L","w","I","d","E","i","e","c","t","T"])}parse(t,e,r){switch(e){case"D":case"DD":return z(Z.dayOfYear,t);case"Do":return r.ordinalNumber(t,{unit:"date"});default:return X(e.length,t)}}validate(t,e){const r=t.getFullYear();return Ln(r)?e>=1&&e<=366:e>=1&&e<=365}set(t,e,r){return t.setMonth(0,r),t.setHours(0,0,0,0),t}}function br(n,a,t){var d,f,p,h;const e=qe(),r=(t==null?void 0:t.weekStartsOn)??((f=(d=t==null?void 0:t.locale)==null?void 0:d.options)==null?void 0:f.weekStartsOn)??e.weekStartsOn??((h=(p=e.locale)==null?void 0:p.options)==null?void 0:h.weekStartsOn)??0,o=P(n,t==null?void 0:t.in),s=o.getDay(),c=(a%7+7)%7,l=7-r,u=a<0||a>6?a-(s+l)%7:(c+l)%7-(s+l)%7;return pe(o,u,t)}class Qo extends ${constructor(){super(...arguments);S(this,"priority",90);S(this,"incompatibleTokens",["D","i","e","c","t","T"])}parse(t,e,r){switch(e){case"E":case"EE":case"EEE":return r.day(t,{width:"abbreviated",context:"formatting"})||r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"});case"EEEEE":return r.day(t,{width:"narrow",context:"formatting"});case"EEEEEE":return r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"});case"EEEE":default:return r.day(t,{width:"wide",context:"formatting"})||r.day(t,{width:"abbreviated",context:"formatting"})||r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"})}}validate(t,e){return e>=0&&e<=6}set(t,e,r,o){return t=br(t,r,o),t.setHours(0,0,0,0),t}}class $o extends ${constructor(){super(...arguments);S(this,"priority",90);S(this,"incompatibleTokens",["y","R","u","q","Q","M","L","I","d","D","E","i","c","t","T"])}parse(t,e,r,o){const s=i=>{const c=Math.floor((i-1)/7)*7;return(i+o.weekStartsOn+6)%7+c};switch(e){case"e":case"ee":return J(X(e.length,t),s);case"eo":return J(r.ordinalNumber(t,{unit:"day"}),s);case"eee":return r.day(t,{width:"abbreviated",context:"formatting"})||r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"});case"eeeee":return r.day(t,{width:"narrow",context:"formatting"});case"eeeeee":return r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"});case"eeee":default:return r.day(t,{width:"wide",context:"formatting"})||r.day(t,{width:"abbreviated",context:"formatting"})||r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"})}}validate(t,e){return e>=0&&e<=6}set(t,e,r,o){return t=br(t,r,o),t.setHours(0,0,0,0),t}}class Vo extends ${constructor(){super(...arguments);S(this,"priority",90);S(this,"incompatibleTokens",["y","R","u","q","Q","M","L","I","d","D","E","i","e","t","T"])}parse(t,e,r,o){const s=i=>{const c=Math.floor((i-1)/7)*7;return(i+o.weekStartsOn+6)%7+c};switch(e){case"c":case"cc":return J(X(e.length,t),s);case"co":return J(r.ordinalNumber(t,{unit:"day"}),s);case"ccc":return r.day(t,{width:"abbreviated",context:"standalone"})||r.day(t,{width:"short",context:"standalone"})||r.day(t,{width:"narrow",context:"standalone"});case"ccccc":return r.day(t,{width:"narrow",context:"standalone"});case"cccccc":return r.day(t,{width:"short",context:"standalone"})||r.day(t,{width:"narrow",context:"standalone"});case"cccc":default:return r.day(t,{width:"wide",context:"standalone"})||r.day(t,{width:"abbreviated",context:"standalone"})||r.day(t,{width:"short",context:"standalone"})||r.day(t,{width:"narrow",context:"standalone"})}}validate(t,e){return e>=0&&e<=6}set(t,e,r,o){return t=br(t,r,o),t.setHours(0,0,0,0),t}}function qo(n,a,t){const e=P(n,t==null?void 0:t.in),r=yo(e,t),o=a-r;return pe(e,o,t)}class Ko extends ${constructor(){super(...arguments);S(this,"priority",90);S(this,"incompatibleTokens",["y","Y","u","q","Q","M","L","w","d","D","E","e","c","t","T"])}parse(t,e,r){const o=s=>s===0?7:s;switch(e){case"i":case"ii":return X(e.length,t);case"io":return r.ordinalNumber(t,{unit:"day"});case"iii":return J(r.day(t,{width:"abbreviated",context:"formatting"})||r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"}),o);case"iiiii":return J(r.day(t,{width:"narrow",context:"formatting"}),o);case"iiiiii":return J(r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"}),o);case"iiii":default:return J(r.day(t,{width:"wide",context:"formatting"})||r.day(t,{width:"abbreviated",context:"formatting"})||r.day(t,{width:"short",context:"formatting"})||r.day(t,{width:"narrow",context:"formatting"}),o)}}validate(t,e){return e>=1&&e<=7}set(t,e,r){return t=qo(t,r),t.setHours(0,0,0,0),t}}class Uo extends ${constructor(){super(...arguments);S(this,"priority",80);S(this,"incompatibleTokens",["b","B","H","k","t","T"])}parse(t,e,r){switch(e){case"a":case"aa":case"aaa":return r.dayPeriod(t,{width:"abbreviated",context:"formatting"})||r.dayPeriod(t,{width:"narrow",context:"formatting"});case"aaaaa":return r.dayPeriod(t,{width:"narrow",context:"formatting"});case"aaaa":default:return r.dayPeriod(t,{width:"wide",context:"formatting"})||r.dayPeriod(t,{width:"abbreviated",context:"formatting"})||r.dayPeriod(t,{width:"narrow",context:"formatting"})}}set(t,e,r){return t.setHours(Dr(r),0,0,0),t}}class Go extends ${constructor(){super(...arguments);S(this,"priority",80);S(this,"incompatibleTokens",["a","B","H","k","t","T"])}parse(t,e,r){switch(e){case"b":case"bb":case"bbb":return r.dayPeriod(t,{width:"abbreviated",context:"formatting"})||r.dayPeriod(t,{width:"narrow",context:"formatting"});case"bbbbb":return r.dayPeriod(t,{width:"narrow",context:"formatting"});case"bbbb":default:return r.dayPeriod(t,{width:"wide",context:"formatting"})||r.dayPeriod(t,{width:"abbreviated",context:"formatting"})||r.dayPeriod(t,{width:"narrow",context:"formatting"})}}set(t,e,r){return t.setHours(Dr(r),0,0,0),t}}class zo extends ${constructor(){super(...arguments);S(this,"priority",80);S(this,"incompatibleTokens",["a","b","t","T"])}parse(t,e,r){switch(e){case"B":case"BB":case"BBB":return r.dayPeriod(t,{width:"abbreviated",context:"formatting"})||r.dayPeriod(t,{width:"narrow",context:"formatting"});case"BBBBB":return r.dayPeriod(t,{width:"narrow",context:"formatting"});case"BBBB":default:return r.dayPeriod(t,{width:"wide",context:"formatting"})||r.dayPeriod(t,{width:"abbreviated",context:"formatting"})||r.dayPeriod(t,{width:"narrow",context:"formatting"})}}set(t,e,r){return t.setHours(Dr(r),0,0,0),t}}class Xo extends ${constructor(){super(...arguments);S(this,"priority",70);S(this,"incompatibleTokens",["H","K","k","t","T"])}parse(t,e,r){switch(e){case"h":return z(Z.hour12h,t);case"ho":return r.ordinalNumber(t,{unit:"hour"});default:return X(e.length,t)}}validate(t,e){return e>=1&&e<=12}set(t,e,r){const o=t.getHours()>=12;return o&&r<12?t.setHours(r+12,0,0,0):!o&&r===12?t.setHours(0,0,0,0):t.setHours(r,0,0,0),t}}class Zo extends ${constructor(){super(...arguments);S(this,"priority",70);S(this,"incompatibleTokens",["a","b","h","K","k","t","T"])}parse(t,e,r){switch(e){case"H":return z(Z.hour23h,t);case"Ho":return r.ordinalNumber(t,{unit:"hour"});default:return X(e.length,t)}}validate(t,e){return e>=0&&e<=23}set(t,e,r){return t.setHours(r,0,0,0),t}}class Jo extends ${constructor(){super(...arguments);S(this,"priority",70);S(this,"incompatibleTokens",["h","H","k","t","T"])}parse(t,e,r){switch(e){case"K":return z(Z.hour11h,t);case"Ko":return r.ordinalNumber(t,{unit:"hour"});default:return X(e.length,t)}}validate(t,e){return e>=0&&e<=11}set(t,e,r){return t.getHours()>=12&&r<12?t.setHours(r+12,0,0,0):t.setHours(r,0,0,0),t}}class es extends ${constructor(){super(...arguments);S(this,"priority",70);S(this,"incompatibleTokens",["a","b","h","H","K","t","T"])}parse(t,e,r){switch(e){case"k":return z(Z.hour24h,t);case"ko":return r.ordinalNumber(t,{unit:"hour"});default:return X(e.length,t)}}validate(t,e){return e>=1&&e<=24}set(t,e,r){const o=r<=24?r%24:r;return t.setHours(o,0,0,0),t}}class ts extends ${constructor(){super(...arguments);S(this,"priority",60);S(this,"incompatibleTokens",["t","T"])}parse(t,e,r){switch(e){case"m":return z(Z.minute,t);case"mo":return r.ordinalNumber(t,{unit:"minute"});default:return X(e.length,t)}}validate(t,e){return e>=0&&e<=59}set(t,e,r){return t.setMinutes(r,0,0),t}}class rs extends ${constructor(){super(...arguments);S(this,"priority",50);S(this,"incompatibleTokens",["t","T"])}parse(t,e,r){switch(e){case"s":return z(Z.second,t);case"so":return r.ordinalNumber(t,{unit:"second"});default:return X(e.length,t)}}validate(t,e){return e>=0&&e<=59}set(t,e,r){return t.setSeconds(r,0),t}}class ns extends ${constructor(){super(...arguments);S(this,"priority",30);S(this,"incompatibleTokens",["t","T"])}parse(t,e){const r=o=>Math.trunc(o*Math.pow(10,-e.length+3));return J(X(e.length,t),r)}set(t,e,r){return t.setMilliseconds(r),t}}class as extends ${constructor(){super(...arguments);S(this,"priority",10);S(this,"incompatibleTokens",["t","T","x"])}parse(t,e){switch(e){case"X":return we(ve.basicOptionalMinutes,t);case"XX":return we(ve.basic,t);case"XXXX":return we(ve.basicOptionalSeconds,t);case"XXXXX":return we(ve.extendedOptionalSeconds,t);case"XXX":default:return we(ve.extended,t)}}set(t,e,r){return e.timestampIsSet?t:U(t,t.getTime()-Nt(t)-r)}}class os extends ${constructor(){super(...arguments);S(this,"priority",10);S(this,"incompatibleTokens",["t","T","X"])}parse(t,e){switch(e){case"x":return we(ve.basicOptionalMinutes,t);case"xx":return we(ve.basic,t);case"xxxx":return we(ve.basicOptionalSeconds,t);case"xxxxx":return we(ve.extendedOptionalSeconds,t);case"xxx":default:return we(ve.extended,t)}}set(t,e,r){return e.timestampIsSet?t:U(t,t.getTime()-Nt(t)-r)}}class ss extends ${constructor(){super(...arguments);S(this,"priority",40);S(this,"incompatibleTokens","*")}parse(t){return Nn(t)}set(t,e,r){return[U(t,r*1e3),{timestampIsSet:!0}]}}class is extends ${constructor(){super(...arguments);S(this,"priority",20);S(this,"incompatibleTokens","*")}parse(t){return Nn(t)}set(t,e,r){return[U(t,r),{timestampIsSet:!0}]}}const cs={G:new Co,y:new So,Y:new Eo,R:new Po,u:new To,Q:new Oo,q:new Ro,M:new No,L:new Yo,w:new Fo,I:new Ao,d:new jo,D:new Bo,E:new Qo,e:new $o,c:new Vo,i:new Ko,a:new Uo,b:new Go,B:new zo,h:new Xo,H:new Zo,K:new Jo,k:new es,m:new ts,s:new rs,S:new ns,X:new as,x:new os,t:new ss,T:new is},ls=/[yYQqMLwIdDecihHKkms]o|(\w)\1*|''|'(''|[^'])+('|$)|./g,us=/P+p+|P+|p+|''|'(''|[^'])+('|$)|./g,ds=/^'([^]*?)'?$/,fs=/''/g,ps=/\S/,hs=/[a-zA-Z]/;function ms(n,a,t,e){var m,y,_,k,E,O,L,T;const r=()=>U((e==null?void 0:e.in)||t,NaN),o=wo(),s=(e==null?void 0:e.locale)??o.locale??Pn,i=(e==null?void 0:e.firstWeekContainsDate)??((y=(m=e==null?void 0:e.locale)==null?void 0:m.options)==null?void 0:y.firstWeekContainsDate)??o.firstWeekContainsDate??((k=(_=o.locale)==null?void 0:_.options)==null?void 0:k.firstWeekContainsDate)??1,c=(e==null?void 0:e.weekStartsOn)??((O=(E=e==null?void 0:e.locale)==null?void 0:E.options)==null?void 0:O.weekStartsOn)??o.weekStartsOn??((T=(L=o.locale)==null?void 0:L.options)==null?void 0:T.weekStartsOn)??0;if(!a)return n?r():P(t,e==null?void 0:e.in);const l={firstWeekContainsDate:i,weekStartsOn:c,locale:s},u=[new Mo(e==null?void 0:e.in,t)],d=a.match(us).map(C=>{const D=C[0];if(D in fr){const b=fr[D];return b(C,s.formatLong)}return C}).join("").match(ls),f=[];for(let C of d){const D=C[0],b=cs[D];if(b){const{incompatibleTokens:N}=b;if(Array.isArray(N)){const A=f.find(W=>N.includes(W.token)||W.token===D);if(A)throw new RangeError(`The format string mustn't contain \`${A.fullToken}\` and \`${C}\` at the same time`)}else if(b.incompatibleTokens==="*"&&f.length>0)throw new RangeError(`The format string mustn't contain \`${C}\` and any other token at the same time`);f.push({token:D,fullToken:C});const R=b.run(n,C,s.match,l);if(!R)return r();u.push(R.setter),n=R.rest}else{if(D.match(hs))throw new RangeError("Format string contains an unescaped latin alphabet character `"+D+"`");if(C==="''"?C="'":D==="'"&&(C=gs(C)),n.indexOf(C)===0)n=n.slice(C.length);else return r()}}if(n.length>0&&ps.test(n))return r();const p=u.map(C=>C.priority).sort((C,D)=>D-C).filter((C,D,b)=>b.indexOf(C)===D).map(C=>u.filter(D=>D.priority===C).sort((D,b)=>b.subPriority-D.subPriority)).map(C=>C[0]);let h=P(t,e==null?void 0:e.in);if(isNaN(+h))return r();const w={};for(const C of p){if(!C.validate(h,l))return r();const D=C.set(h,w,l);Array.isArray(D)?(h=D[0],Object.assign(w,D[1])):h=D}return h}function gs(n){return n.match(ds)[1].replace(fs,"'")}function vs(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a);return e.getFullYear()===r.getFullYear()&&e.getMonth()===r.getMonth()}function ws(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a);return+dr(e)==+dr(r)}function ys(n,a,t){const[e,r]=Pe(t==null?void 0:t.in,n,a);return e.getFullYear()===r.getFullYear()}function ft(n,a,t){const e=+P(n,t==null?void 0:t.in),[r,o]=[+P(a.start,t==null?void 0:t.in),+P(a.end,t==null?void 0:t.in)].sort((s,i)=>s-i);return e>=r&&e<=o}function Ds(n,a,t){return pe(n,-1,t)}function bs(n,a){const t=()=>U(a==null?void 0:a.in,NaN),r=Ms(n);let o;if(r.date){const l=Cs(r.date,2);o=Ss(l.restDateString,l.year)}if(!o||isNaN(+o))return t();const s=+o;let i=0,c;if(r.time&&(i=Es(r.time),isNaN(i)))return t();if(r.timezone){if(c=Ps(r.timezone),isNaN(c))return t()}else{const l=new Date(s+i),u=P(0,a==null?void 0:a.in);return u.setFullYear(l.getUTCFullYear(),l.getUTCMonth(),l.getUTCDate()),u.setHours(l.getUTCHours(),l.getUTCMinutes(),l.getUTCSeconds(),l.getUTCMilliseconds()),u}return P(s+i+c,a==null?void 0:a.in)}const yt={dateTimeDelimiter:/[T ]/,timeZoneDelimiter:/[Z ]/i,timezone:/([Z+-].*)$/},ks=/^-?(?:(\d{3})|(\d{2})(?:-?(\d{2}))?|W(\d{2})(?:-?(\d{1}))?|)$/,xs=/^(\d{2}(?:[.,]\d*)?)(?::?(\d{2}(?:[.,]\d*)?))?(?::?(\d{2}(?:[.,]\d*)?))?$/,_s=/^([+-])(\d{2})(?::?(\d{2}))?$/;function Ms(n){const a={},t=n.split(yt.dateTimeDelimiter);let e;if(t.length>2)return a;if(/:/.test(t[0])?e=t[0]:(a.date=t[0],e=t[1],yt.timeZoneDelimiter.test(a.date)&&(a.date=n.split(yt.timeZoneDelimiter)[0],e=n.substr(a.date.length,n.length))),e){const r=yt.timezone.exec(e);r?(a.time=e.replace(r[1],""),a.timezone=r[1]):a.time=e}return a}function Cs(n,a){const t=new RegExp("^(?:(\\d{4}|[+-]\\d{"+(4+a)+"})|(\\d{2}|[+-]\\d{"+(2+a)+"})$)"),e=n.match(t);if(!e)return{year:NaN,restDateString:""};const r=e[1]?parseInt(e[1]):null,o=e[2]?parseInt(e[2]):null;return{year:o===null?r:o*100,restDateString:n.slice((e[1]||e[2]).length)}}function Ss(n,a){if(a===null)return new Date(NaN);const t=n.match(ks);if(!t)return new Date(NaN);const e=!!t[4],r=ct(t[1]),o=ct(t[2])-1,s=ct(t[3]),i=ct(t[4]),c=ct(t[5])-1;if(e)return Ys(a,i,c)?Ts(a,i,c):new Date(NaN);{const l=new Date(0);return!Rs(a,o,s)||!Ns(a,r)?new Date(NaN):(l.setUTCFullYear(a,o,Math.max(r,s)),l)}}function ct(n){return n?parseInt(n):1}function Es(n){const a=n.match(xs);if(!a)return NaN;const t=rr(a[1]),e=rr(a[2]),r=rr(a[3]);return Ls(t,e,r)?t*Kt+e*qt+r*1e3:NaN}function rr(n){return n&&parseFloat(n.replace(",","."))||0}function Ps(n){if(n==="Z")return 0;const a=n.match(_s);if(!a)return 0;const t=a[1]==="+"?-1:1,e=parseInt(a[2]),r=a[3]&&parseInt(a[3])||0;return Fs(e,r)?t*(e*Kt+r*qt):NaN}function Ts(n,a,t){const e=new Date(0);e.setUTCFullYear(n,0,4);const r=e.getUTCDay()||7,o=(a-1)*7+t+1-r;return e.setUTCDate(e.getUTCDate()+o),e}const Os=[31,null,31,30,31,30,31,31,30,31,30,31];function Fn(n){return n%400===0||n%4===0&&n%100!==0}function Rs(n,a,t){return a>=0&&a<=11&&t>=1&&t<=(Os[a]||(Fn(n)?29:28))}function Ns(n,a){return a>=1&&a<=(Fn(n)?366:365)}function Ys(n,a,t){return a>=1&&a<=53&&t>=0&&t<=6}function Ls(n,a,t){return n===24?a===0&&t===0:t>=0&&t<60&&a>=0&&a<60&&n>=0&&n<25}function Fs(n,a){return a>=0&&a<=59}function ce(n,a,t){const e=P(n,t==null?void 0:t.in),r=e.getFullYear(),o=e.getDate(),s=U(n,0);s.setFullYear(r,a,15),s.setHours(0,0,0,0);const i=vo(s);return e.setMonth(a,Math.min(o,i)),e}function Ct(n,a,t){const e=P(n,t==null?void 0:t.in);return e.setHours(a),e}function St(n,a,t){const e=P(n,t==null?void 0:t.in);return e.setMinutes(a),e}function Ge(n,a,t){const e=P(n,t==null?void 0:t.in),r=Math.trunc(e.getMonth()/3)+1,o=a-r;return ce(e,e.getMonth()+o*3)}function Et(n,a,t){const e=P(n,t==null?void 0:t.in);return e.setSeconds(a),e}function me(n,a,t){const e=P(n,t==null?void 0:t.in);return isNaN(+e)?U(n,NaN):(e.setFullYear(a),e)}function je(n,a,t){return he(n,-a,t)}function In(n,a,t){return vr(n,-1,t)}function Qr(n,a,t){return Yt(n,-1,t)}function et(n,a,t){return xe(n,-a,t)}function Gt(){return typeof window<"u"}function ot(n){return An(n)?(n.nodeName||"").toLowerCase():"#document"}function se(n){var a;return(n==null||(a=n.ownerDocument)==null?void 0:a.defaultView)||window}function ke(n){var a;return(a=(An(n)?n.ownerDocument:n.document)||window.document)==null?void 0:a.documentElement}function An(n){return Gt()?n instanceof Node||n instanceof se(n).Node:!1}function oe(n){return Gt()?n instanceof Element||n instanceof se(n).Element:!1}function be(n){return Gt()?n instanceof HTMLElement||n instanceof se(n).HTMLElement:!1}function $r(n){return!Gt()||typeof ShadowRoot>"u"?!1:n instanceof ShadowRoot||n instanceof se(n).ShadowRoot}const Is=new Set(["inline","contents"]);function mt(n){const{overflow:a,overflowX:t,overflowY:e,display:r}=ue(n);return/auto|scroll|overlay|hidden|clip/.test(a+e+t)&&!Is.has(r)}const As=new Set(["table","td","th"]);function Ws(n){return As.has(ot(n))}const Hs=[":popover-open",":modal"];function zt(n){return Hs.some(a=>{try{return n.matches(a)}catch{return!1}})}const js=["transform","translate","scale","rotate","perspective"],Bs=["transform","translate","scale","rotate","perspective","filter"],Qs=["paint","layout","strict","content"];function kr(n){const a=xr(),t=oe(n)?ue(n):n;return js.some(e=>t[e]?t[e]!=="none":!1)||(t.containerType?t.containerType!=="normal":!1)||!a&&(t.backdropFilter?t.backdropFilter!=="none":!1)||!a&&(t.filter?t.filter!=="none":!1)||Bs.some(e=>(t.willChange||"").includes(e))||Qs.some(e=>(t.contain||"").includes(e))}function $s(n){let a=Fe(n);for(;be(a)&&!tt(a);){if(kr(a))return a;if(zt(a))return null;a=Fe(a)}return null}function xr(){return typeof CSS>"u"||!CSS.supports?!1:CSS.supports("-webkit-backdrop-filter","none")}const Vs=new Set(["html","body","#document"]);function tt(n){return Vs.has(ot(n))}function ue(n){return se(n).getComputedStyle(n)}function Xt(n){return oe(n)?{scrollLeft:n.scrollLeft,scrollTop:n.scrollTop}:{scrollLeft:n.scrollX,scrollTop:n.scrollY}}function Fe(n){if(ot(n)==="html")return n;const a=n.assignedSlot||n.parentNode||$r(n)&&n.host||ke(n);return $r(a)?a.host:a}function Wn(n){const a=Fe(n);return tt(a)?n.ownerDocument?n.ownerDocument.body:n.body:be(a)&&mt(a)?a:Wn(a)}function pt(n,a,t){var e;a===void 0&&(a=[]),t===void 0&&(t=!0);const r=Wn(n),o=r===((e=n.ownerDocument)==null?void 0:e.body),s=se(r);if(o){const i=hr(s);return a.concat(s,s.visualViewport||[],mt(r)?r:[],i&&t?pt(i):[])}return a.concat(r,pt(r,[],t))}function hr(n){return n.parent&&Object.getPrototypeOf(n.parent)?n.frameElement:null}const rt=Math.min,Be=Math.max,Ht=Math.round,Dt=Math.floor,De=n=>({x:n,y:n}),qs={left:"right",right:"left",bottom:"top",top:"bottom"},Ks={start:"end",end:"start"};function Us(n,a,t){return Be(n,rt(a,t))}function Zt(n,a){return typeof n=="function"?n(a):n}function nt(n){return n.split("-")[0]}function gt(n){return n.split("-")[1]}function Gs(n){return n==="x"?"y":"x"}function _r(n){return n==="y"?"height":"width"}const zs=new Set(["top","bottom"]);function Ae(n){return zs.has(nt(n))?"y":"x"}function Mr(n){return Gs(Ae(n))}function Xs(n,a,t){t===void 0&&(t=!1);const e=gt(n),r=Mr(n),o=_r(r);let s=r==="x"?e===(t?"end":"start")?"right":"left":e==="start"?"bottom":"top";return a.reference[o]>a.floating[o]&&(s=jt(s)),[s,jt(s)]}function Zs(n){const a=jt(n);return[mr(n),a,mr(a)]}function mr(n){return n.replace(/start|end/g,a=>Ks[a])}const Vr=["left","right"],qr=["right","left"],Js=["top","bottom"],ei=["bottom","top"];function ti(n,a,t){switch(n){case"top":case"bottom":return t?a?qr:Vr:a?Vr:qr;case"left":case"right":return a?Js:ei;default:return[]}}function ri(n,a,t,e){const r=gt(n);let o=ti(nt(n),t==="start",e);return r&&(o=o.map(s=>s+"-"+r),a&&(o=o.concat(o.map(mr)))),o}function jt(n){return n.replace(/left|right|bottom|top/g,a=>qs[a])}function ni(n){return{top:0,right:0,bottom:0,left:0,...n}}function Hn(n){return typeof n!="number"?ni(n):{top:n,right:n,bottom:n,left:n}}function Bt(n){const{x:a,y:t,width:e,height:r}=n;return{width:e,height:r,top:t,left:a,right:a+e,bottom:t+r,x:a,y:t}}var ai=typeof document<"u",oi=function(){},Qt=ai?v.useLayoutEffect:oi;const si={...bn},ii=si.useInsertionEffect,ci=ii||(n=>n());function li(n){const a=v.useRef(()=>{});return ci(()=>{a.current=n}),v.useCallback(function(){for(var t=arguments.length,e=new Array(t),r=0;r{const{placement:e="bottom",strategy:r="absolute",middleware:o=[],platform:s}=t,i=o.filter(Boolean),c=await(s.isRTL==null?void 0:s.isRTL(a));let l=await s.getElementRects({reference:n,floating:a,strategy:r}),{x:u,y:d}=Kr(l,e,c),f=e,p={},h=0;for(let w=0;w({name:"arrow",options:n,async fn(a){const{x:t,y:e,placement:r,rects:o,platform:s,elements:i,middlewareData:c}=a,{element:l,padding:u=0}=Zt(n,a)||{};if(l==null)return{};const d=Hn(u),f={x:t,y:e},p=Mr(r),h=_r(p),w=await s.getDimensions(l),m=p==="y",y=m?"top":"left",_=m?"bottom":"right",k=m?"clientHeight":"clientWidth",E=o.reference[h]+o.reference[p]-f[p]-o.floating[h],O=f[p]-o.reference[p],L=await(s.getOffsetParent==null?void 0:s.getOffsetParent(l));let T=L?L[k]:0;(!T||!await(s.isElement==null?void 0:s.isElement(L)))&&(T=i.floating[k]||o.floating[h]);const C=E/2-O/2,D=T/2-w[h]/2-1,b=rt(d[y],D),N=rt(d[_],D),R=b,A=T-w[h]-N,W=T/2-w[h]/2+C,q=Us(R,W,A),Y=!c.arrow&>(r)!=null&&W!==q&&o.reference[h]/2-(WW<=0)){var N,R;const W=(((N=o.flip)==null?void 0:N.index)||0)+1,q=T[W];if(q&&(!(d==="alignment"?_!==Ae(q):!1)||b.every(F=>F.overflows[0]>0&&Ae(F.placement)===_)))return{data:{index:W,overflows:b},reset:{placement:q}};let Y=(R=b.filter(H=>H.overflows[0]<=0).sort((H,F)=>H.overflows[1]-F.overflows[1])[0])==null?void 0:R.placement;if(!Y)switch(p){case"bestFit":{var A;const H=(A=b.filter(F=>{if(L){const ee=Ae(F.placement);return ee===_||ee==="y"}return!0}).map(F=>[F.placement,F.overflows.filter(ee=>ee>0).reduce((ee,de)=>ee+de,0)]).sort((F,ee)=>F[1]-ee[1])[0])==null?void 0:A[0];H&&(Y=H);break}case"initialPlacement":Y=i;break}if(r!==Y)return{reset:{placement:Y}}}return{}}}},hi=new Set(["left","top"]);async function mi(n,a){const{placement:t,platform:e,elements:r}=n,o=await(e.isRTL==null?void 0:e.isRTL(r.floating)),s=nt(t),i=gt(t),c=Ae(t)==="y",l=hi.has(s)?-1:1,u=o&&c?-1:1,d=Zt(a,n);let{mainAxis:f,crossAxis:p,alignmentAxis:h}=typeof d=="number"?{mainAxis:d,crossAxis:0,alignmentAxis:null}:{mainAxis:d.mainAxis||0,crossAxis:d.crossAxis||0,alignmentAxis:d.alignmentAxis};return i&&typeof h=="number"&&(p=i==="end"?h*-1:h),c?{x:p*u,y:f*l}:{x:f*l,y:p*u}}const gi=function(n){return n===void 0&&(n=0),{name:"offset",options:n,async fn(a){var t,e;const{x:r,y:o,placement:s,middlewareData:i}=a,c=await mi(a,n);return s===((t=i.offset)==null?void 0:t.placement)&&(e=i.arrow)!=null&&e.alignmentOffset?{}:{x:r+c.x,y:o+c.y,data:{...c,placement:s}}}}};function jn(n){const a=ue(n);let t=parseFloat(a.width)||0,e=parseFloat(a.height)||0;const r=be(n),o=r?n.offsetWidth:t,s=r?n.offsetHeight:e,i=Ht(t)!==o||Ht(e)!==s;return i&&(t=o,e=s),{width:t,height:e,$:i}}function Cr(n){return oe(n)?n:n.contextElement}function ze(n){const a=Cr(n);if(!be(a))return De(1);const t=a.getBoundingClientRect(),{width:e,height:r,$:o}=jn(a);let s=(o?Ht(t.width):t.width)/e,i=(o?Ht(t.height):t.height)/r;return(!s||!Number.isFinite(s))&&(s=1),(!i||!Number.isFinite(i))&&(i=1),{x:s,y:i}}const vi=De(0);function Bn(n){const a=se(n);return!xr()||!a.visualViewport?vi:{x:a.visualViewport.offsetLeft,y:a.visualViewport.offsetTop}}function wi(n,a,t){return a===void 0&&(a=!1),!t||a&&t!==se(n)?!1:a}function Ve(n,a,t,e){a===void 0&&(a=!1),t===void 0&&(t=!1);const r=n.getBoundingClientRect(),o=Cr(n);let s=De(1);a&&(e?oe(e)&&(s=ze(e)):s=ze(n));const i=wi(o,t,e)?Bn(o):De(0);let c=(r.left+i.x)/s.x,l=(r.top+i.y)/s.y,u=r.width/s.x,d=r.height/s.y;if(o){const f=se(o),p=e&&oe(e)?se(e):e;let h=f,w=hr(h);for(;w&&e&&p!==h;){const m=ze(w),y=w.getBoundingClientRect(),_=ue(w),k=y.left+(w.clientLeft+parseFloat(_.paddingLeft))*m.x,E=y.top+(w.clientTop+parseFloat(_.paddingTop))*m.y;c*=m.x,l*=m.y,u*=m.x,d*=m.y,c+=k,l+=E,h=se(w),w=hr(h)}}return Bt({width:u,height:d,x:c,y:l})}function Sr(n,a){const t=Xt(n).scrollLeft;return a?a.left+t:Ve(ke(n)).left+t}function Qn(n,a,t){t===void 0&&(t=!1);const e=n.getBoundingClientRect(),r=e.left+a.scrollLeft-(t?0:Sr(n,e)),o=e.top+a.scrollTop;return{x:r,y:o}}function yi(n){let{elements:a,rect:t,offsetParent:e,strategy:r}=n;const o=r==="fixed",s=ke(e),i=a?zt(a.floating):!1;if(e===s||i&&o)return t;let c={scrollLeft:0,scrollTop:0},l=De(1);const u=De(0),d=be(e);if((d||!d&&!o)&&((ot(e)!=="body"||mt(s))&&(c=Xt(e)),be(e))){const p=Ve(e);l=ze(e),u.x=p.x+e.clientLeft,u.y=p.y+e.clientTop}const f=s&&!d&&!o?Qn(s,c,!0):De(0);return{width:t.width*l.x,height:t.height*l.y,x:t.x*l.x-c.scrollLeft*l.x+u.x+f.x,y:t.y*l.y-c.scrollTop*l.y+u.y+f.y}}function Di(n){return Array.from(n.getClientRects())}function bi(n){const a=ke(n),t=Xt(n),e=n.ownerDocument.body,r=Be(a.scrollWidth,a.clientWidth,e.scrollWidth,e.clientWidth),o=Be(a.scrollHeight,a.clientHeight,e.scrollHeight,e.clientHeight);let s=-t.scrollLeft+Sr(n);const i=-t.scrollTop;return ue(e).direction==="rtl"&&(s+=Be(a.clientWidth,e.clientWidth)-r),{width:r,height:o,x:s,y:i}}function ki(n,a){const t=se(n),e=ke(n),r=t.visualViewport;let o=e.clientWidth,s=e.clientHeight,i=0,c=0;if(r){o=r.width,s=r.height;const l=xr();(!l||l&&a==="fixed")&&(i=r.offsetLeft,c=r.offsetTop)}return{width:o,height:s,x:i,y:c}}const xi=new Set(["absolute","fixed"]);function _i(n,a){const t=Ve(n,!0,a==="fixed"),e=t.top+n.clientTop,r=t.left+n.clientLeft,o=be(n)?ze(n):De(1),s=n.clientWidth*o.x,i=n.clientHeight*o.y,c=r*o.x,l=e*o.y;return{width:s,height:i,x:c,y:l}}function Ur(n,a,t){let e;if(a==="viewport")e=ki(n,t);else if(a==="document")e=bi(ke(n));else if(oe(a))e=_i(a,t);else{const r=Bn(n);e={x:a.x-r.x,y:a.y-r.y,width:a.width,height:a.height}}return Bt(e)}function $n(n,a){const t=Fe(n);return t===a||!oe(t)||tt(t)?!1:ue(t).position==="fixed"||$n(t,a)}function Mi(n,a){const t=a.get(n);if(t)return t;let e=pt(n,[],!1).filter(i=>oe(i)&&ot(i)!=="body"),r=null;const o=ue(n).position==="fixed";let s=o?Fe(n):n;for(;oe(s)&&!tt(s);){const i=ue(s),c=kr(s);!c&&i.position==="fixed"&&(r=null),(o?!c&&!r:!c&&i.position==="static"&&!!r&&xi.has(r.position)||mt(s)&&!c&&$n(n,s))?e=e.filter(u=>u!==s):r=i,s=Fe(s)}return a.set(n,e),e}function Ci(n){let{element:a,boundary:t,rootBoundary:e,strategy:r}=n;const s=[...t==="clippingAncestors"?zt(a)?[]:Mi(a,this._c):[].concat(t),e],i=s[0],c=s.reduce((l,u)=>{const d=Ur(a,u,r);return l.top=Be(d.top,l.top),l.right=rt(d.right,l.right),l.bottom=rt(d.bottom,l.bottom),l.left=Be(d.left,l.left),l},Ur(a,i,r));return{width:c.right-c.left,height:c.bottom-c.top,x:c.left,y:c.top}}function Si(n){const{width:a,height:t}=jn(n);return{width:a,height:t}}function Ei(n,a,t){const e=be(a),r=ke(a),o=t==="fixed",s=Ve(n,!0,o,a);let i={scrollLeft:0,scrollTop:0};const c=De(0);function l(){c.x=Sr(r)}if(e||!e&&!o)if((ot(a)!=="body"||mt(r))&&(i=Xt(a)),e){const p=Ve(a,!0,o,a);c.x=p.x+a.clientLeft,c.y=p.y+a.clientTop}else r&&l();o&&!e&&r&&l();const u=r&&!e&&!o?Qn(r,i):De(0),d=s.left+i.scrollLeft-c.x-u.x,f=s.top+i.scrollTop-c.y-u.y;return{x:d,y:f,width:s.width,height:s.height}}function nr(n){return ue(n).position==="static"}function Gr(n,a){if(!be(n)||ue(n).position==="fixed")return null;if(a)return a(n);let t=n.offsetParent;return ke(n)===t&&(t=t.ownerDocument.body),t}function Vn(n,a){const t=se(n);if(zt(n))return t;if(!be(n)){let r=Fe(n);for(;r&&!tt(r);){if(oe(r)&&!nr(r))return r;r=Fe(r)}return t}let e=Gr(n,a);for(;e&&Ws(e)&&nr(e);)e=Gr(e,a);return e&&tt(e)&&nr(e)&&!kr(e)?t:e||$s(n)||t}const Pi=async function(n){const a=this.getOffsetParent||Vn,t=this.getDimensions,e=await t(n.floating);return{reference:Ei(n.reference,await a(n.floating),n.strategy),floating:{x:0,y:0,width:e.width,height:e.height}}};function Ti(n){return ue(n).direction==="rtl"}const Oi={convertOffsetParentRelativeRectToViewportRelativeRect:yi,getDocumentElement:ke,getClippingRect:Ci,getOffsetParent:Vn,getElementRects:Pi,getClientRects:Di,getDimensions:Si,getScale:ze,isElement:oe,isRTL:Ti};function qn(n,a){return n.x===a.x&&n.y===a.y&&n.width===a.width&&n.height===a.height}function Ri(n,a){let t=null,e;const r=ke(n);function o(){var i;clearTimeout(e),(i=t)==null||i.disconnect(),t=null}function s(i,c){i===void 0&&(i=!1),c===void 0&&(c=1),o();const l=n.getBoundingClientRect(),{left:u,top:d,width:f,height:p}=l;if(i||a(),!f||!p)return;const h=Dt(d),w=Dt(r.clientWidth-(u+f)),m=Dt(r.clientHeight-(d+p)),y=Dt(u),k={rootMargin:-h+"px "+-w+"px "+-m+"px "+-y+"px",threshold:Be(0,rt(1,c))||1};let E=!0;function O(L){const T=L[0].intersectionRatio;if(T!==c){if(!E)return s();T?s(!1,T):e=setTimeout(()=>{s(!1,1e-7)},1e3)}T===1&&!qn(l,n.getBoundingClientRect())&&s(),E=!1}try{t=new IntersectionObserver(O,{...k,root:r.ownerDocument})}catch{t=new IntersectionObserver(O,k)}t.observe(n)}return s(!0),o}function Ni(n,a,t,e){e===void 0&&(e={});const{ancestorScroll:r=!0,ancestorResize:o=!0,elementResize:s=typeof ResizeObserver=="function",layoutShift:i=typeof IntersectionObserver=="function",animationFrame:c=!1}=e,l=Cr(n),u=r||o?[...l?pt(l):[],...pt(a)]:[];u.forEach(y=>{r&&y.addEventListener("scroll",t,{passive:!0}),o&&y.addEventListener("resize",t)});const d=l&&i?Ri(l,t):null;let f=-1,p=null;s&&(p=new ResizeObserver(y=>{let[_]=y;_&&_.target===l&&p&&(p.unobserve(a),cancelAnimationFrame(f),f=requestAnimationFrame(()=>{var k;(k=p)==null||k.observe(a)})),t()}),l&&!c&&p.observe(l),p.observe(a));let h,w=c?Ve(n):null;c&&m();function m(){const y=Ve(n);w&&!qn(w,y)&&t(),w=y,h=requestAnimationFrame(m)}return t(),()=>{var y;u.forEach(_=>{r&&_.removeEventListener("scroll",t),o&&_.removeEventListener("resize",t)}),d==null||d(),(y=p)==null||y.disconnect(),p=null,c&&cancelAnimationFrame(h)}}const Yi=gi,Li=pi,zr=fi,Fi=(n,a,t)=>{const e=new Map,r={platform:Oi,...t},o={...r.platform,_c:e};return ui(n,a,{...r,platform:o})};var Ii=typeof document<"u",Ai=function(){},Pt=Ii?v.useLayoutEffect:Ai;function $t(n,a){if(n===a)return!0;if(typeof n!=typeof a)return!1;if(typeof n=="function"&&n.toString()===a.toString())return!0;let t,e,r;if(n&&a&&typeof n=="object"){if(Array.isArray(n)){if(t=n.length,t!==a.length)return!1;for(e=t;e--!==0;)if(!$t(n[e],a[e]))return!1;return!0}if(r=Object.keys(n),t=r.length,t!==Object.keys(a).length)return!1;for(e=t;e--!==0;)if(!{}.hasOwnProperty.call(a,r[e]))return!1;for(e=t;e--!==0;){const o=r[e];if(!(o==="_owner"&&n.$$typeof)&&!$t(n[o],a[o]))return!1}return!0}return n!==n&&a!==a}function Kn(n){return typeof window>"u"?1:(n.ownerDocument.defaultView||window).devicePixelRatio||1}function Xr(n,a){const t=Kn(n);return Math.round(a*t)/t}function ar(n){const a=v.useRef(n);return Pt(()=>{a.current=n}),a}function Wi(n){n===void 0&&(n={});const{placement:a="bottom",strategy:t="absolute",middleware:e=[],platform:r,elements:{reference:o,floating:s}={},transform:i=!0,whileElementsMounted:c,open:l}=n,[u,d]=v.useState({x:0,y:0,strategy:t,placement:a,middlewareData:{},isPositioned:!1}),[f,p]=v.useState(e);$t(f,e)||p(e);const[h,w]=v.useState(null),[m,y]=v.useState(null),_=v.useCallback(F=>{F!==L.current&&(L.current=F,w(F))},[]),k=v.useCallback(F=>{F!==T.current&&(T.current=F,y(F))},[]),E=o||h,O=s||m,L=v.useRef(null),T=v.useRef(null),C=v.useRef(u),D=c!=null,b=ar(c),N=ar(r),R=ar(l),A=v.useCallback(()=>{if(!L.current||!T.current)return;const F={placement:a,strategy:t,middleware:f};N.current&&(F.platform=N.current),Fi(L.current,T.current,F).then(ee=>{const de={...ee,isPositioned:R.current!==!1};W.current&&!$t(C.current,de)&&(C.current=de,ia.flushSync(()=>{d(de)}))})},[f,a,t,N,R]);Pt(()=>{l===!1&&C.current.isPositioned&&(C.current.isPositioned=!1,d(F=>({...F,isPositioned:!1})))},[l]);const W=v.useRef(!1);Pt(()=>(W.current=!0,()=>{W.current=!1}),[]),Pt(()=>{if(E&&(L.current=E),O&&(T.current=O),E&&O){if(b.current)return b.current(E,O,A);A()}},[E,O,A,b,D]);const q=v.useMemo(()=>({reference:L,floating:T,setReference:_,setFloating:k}),[_,k]),Y=v.useMemo(()=>({reference:E,floating:O}),[E,O]),H=v.useMemo(()=>{const F={position:t,left:0,top:0};if(!Y.floating)return F;const ee=Xr(Y.floating,u.x),de=Xr(Y.floating,u.y);return i?{...F,transform:"translate("+ee+"px, "+de+"px)",...Kn(Y.floating)>=1.5&&{willChange:"transform"}}:{position:t,left:ee,top:de}},[t,i,Y.floating,u.x,u.y]);return v.useMemo(()=>({...u,update:A,refs:q,elements:Y,floatingStyles:H}),[u,A,q,Y,H])}const Hi=n=>{function a(t){return{}.hasOwnProperty.call(t,"current")}return{name:"arrow",options:n,fn(t){const{element:e,padding:r}=typeof n=="function"?n(t):n;return e&&a(e)?e.current!=null?zr({element:e.current,padding:r}).fn(t):{}:e?zr({element:e,padding:r}).fn(t):{}}}},ji=(n,a)=>({...Yi(n),options:[n,a]}),Bi=(n,a)=>({...Li(n),options:[n,a]}),Qi=(n,a)=>({...Hi(n),options:[n,a]}),$i={...bn};let Zr=!1,Vi=0;const Jr=()=>"floating-ui-"+Math.random().toString(36).slice(2,6)+Vi++;function qi(){const[n,a]=v.useState(()=>Zr?Jr():void 0);return Qt(()=>{n==null&&a(Jr())},[]),v.useEffect(()=>{Zr=!0},[]),n}const Ki=$i.useId,Un=Ki||qi,Ui=v.forwardRef(function(a,t){const{context:{placement:e,elements:{floating:r},middlewareData:{arrow:o,shift:s}},width:i=14,height:c=7,tipRadius:l=0,strokeWidth:u=0,staticOffset:d,stroke:f,d:p,style:{transform:h,...w}={},...m}=a,y=Un(),[_,k]=v.useState(!1);if(Qt(()=>{if(!r)return;ue(r).direction==="rtl"&&k(!0)},[r]),!r)return null;const[E,O]=e.split("-"),L=E==="top"||E==="bottom";let T=d;(L&&s!=null&&s.x||!L&&s!=null&&s.y)&&(T=null);const C=u*2,D=C/2,b=i/2*(l/-8+1),N=c/2*l/4,R=!!p,A=T&&O==="end"?"bottom":"top";let W=T&&O==="end"?"right":"left";T&&_&&(W=O==="end"?"left":"right");const q=(o==null?void 0:o.x)!=null?T||o.x:"",Y=(o==null?void 0:o.y)!=null?T||o.y:"",H=p||"M0,0"+(" H"+i)+(" L"+(i-b)+","+(c-N))+(" Q"+i/2+","+c+" "+b+","+(c-N))+" Z",F={top:R?"rotate(180deg)":"",left:R?"rotate(90deg)":"rotate(-90deg)",bottom:R?"":"rotate(180deg)",right:R?"rotate(-90deg)":"rotate(90deg)"}[E];return x.jsxs("svg",{...m,"aria-hidden":!0,ref:t,width:R?i:i+C,height:i,viewBox:"0 0 "+i+" "+(c>i?c:i),style:{position:"absolute",pointerEvents:"none",[W]:q,[A]:Y,[E]:L||R?"100%":"calc(100% - "+C/2+"px)",transform:[F,h].filter(ee=>!!ee).join(" "),...w},children:[C>0&&x.jsx("path",{clipPath:"url(#"+y+")",fill:"none",stroke:f,strokeWidth:C+(p?0:1),d:H}),x.jsx("path",{stroke:C&&!p?m.fill:"none",d:H}),x.jsx("clipPath",{id:y,children:x.jsx("rect",{x:-D,y:D*(R?-1:1),width:i+C,height:i})})]})});function Gi(){const n=new Map;return{emit(a,t){var e;(e=n.get(a))==null||e.forEach(r=>r(t))},on(a,t){n.has(a)||n.set(a,new Set),n.get(a).add(t)},off(a,t){var e;(e=n.get(a))==null||e.delete(t)}}}const zi=v.createContext(null),Xi=v.createContext(null),Zi=()=>{var n;return((n=v.useContext(zi))==null?void 0:n.id)||null},Ji=()=>v.useContext(Xi);function ec(n){const{open:a=!1,onOpenChange:t,elements:e}=n,r=Un(),o=v.useRef({}),[s]=v.useState(()=>Gi()),i=Zi()!=null,[c,l]=v.useState(e.reference),u=li((p,h,w)=>{o.current.openEvent=p?h:void 0,s.emit("openchange",{open:p,event:h,reason:w,nested:i}),t==null||t(p,h,w)}),d=v.useMemo(()=>({setPositionReference:l}),[]),f=v.useMemo(()=>({reference:c||e.reference||null,floating:e.floating||null,domReference:e.reference}),[c,e.reference,e.floating]);return v.useMemo(()=>({dataRef:o,open:a,onOpenChange:u,elements:f,events:s,floatingId:r,refs:d}),[a,u,f,s,r,d])}function tc(n){n===void 0&&(n={});const{nodeId:a}=n,t=ec({...n,elements:{reference:null,floating:null,...n.elements}}),e=n.rootContext||t,r=e.elements,[o,s]=v.useState(null),[i,c]=v.useState(null),u=(r==null?void 0:r.domReference)||o,d=v.useRef(null),f=Ji();Qt(()=>{u&&(d.current=u)},[u]);const p=Wi({...n,elements:{...r,...i&&{reference:i}}}),h=v.useCallback(k=>{const E=oe(k)?{getBoundingClientRect:()=>k.getBoundingClientRect(),getClientRects:()=>k.getClientRects(),contextElement:k}:k;c(E),p.refs.setReference(E)},[p.refs]),w=v.useCallback(k=>{(oe(k)||k===null)&&(d.current=k,s(k)),(oe(p.refs.reference.current)||p.refs.reference.current===null||k!==null&&!oe(k))&&p.refs.setReference(k)},[p.refs]),m=v.useMemo(()=>({...p.refs,setReference:w,setPositionReference:h,domReference:d}),[p.refs,w,h]),y=v.useMemo(()=>({...p.elements,domReference:u}),[p.elements,u]),_=v.useMemo(()=>({...p,...e,refs:m,elements:y,nodeId:a}),[p,m,y,a,e]);return Qt(()=>{e.dataRef.current.floatingContext=_;const k=f==null?void 0:f.nodesRef.current.find(E=>E.id===a);k&&(k.context=_)}),v.useMemo(()=>({...p,context:_,refs:m,elements:y}),[p,m,y,_])}/*! + react-datepicker v8.4.0 + https://github.com/Hacker0x01/react-datepicker + Released under the MIT License. +*/var gr=function(a,t){return gr=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(e,r){e.__proto__=r}||function(e,r){for(var o in r)Object.prototype.hasOwnProperty.call(r,o)&&(e[o]=r[o])},gr(a,t)};function te(n,a){if(typeof a!="function"&&a!==null)throw new TypeError("Class extends value "+String(a)+" is not a constructor or null");gr(n,a);function t(){this.constructor=n}n.prototype=a===null?Object.create(a):(t.prototype=a.prototype,new t)}var Q=function(){return Q=Object.assign||function(t){for(var e,r=1,o=arguments.length;r0?t[0]:t;return n&&G(n,r,e)||""}var zn=" - ";function ac(n,a,t){if(!n)return"";var e=ie(n,t),r=a?ie(a,t):"";return"".concat(e).concat(zn).concat(r)}function oc(n,a){if(!(n!=null&&n.length))return"";var t=n[0]?ie(n[0],a):"";if(n.length===1)return t;if(n.length===2&&n[1]){var e=ie(n[1],a);return"".concat(t,", ").concat(e)}var r=n.length-1;return"".concat(t," (+").concat(r,")")}function sr(n,a){var t=a.hour,e=t===void 0?0:t,r=a.minute,o=r===void 0?0:r,s=a.second,i=s===void 0?0:s;return Ct(St(Et(n,i),o),e)}function sc(n){return wr(n)}function ic(n,a){return G(n,"ddd",a)}function Tt(n){return Qe(n)}function Ye(n,a,t){var e=at(a||Er());return Ee(n,{locale:e,weekStartsOn:t})}function Se(n){return Sn(n)}function ut(n){return Ut(n)}function en(n){return dr(n)}function tn(){return Qe(V())}function rn(n){return Mn(n)}function cc(n){return Da(n)}function lc(n){return Cn(n)}function ge(n,a){return n&&a?ys(n,a):!n&&!a}function ae(n,a){return n&&a?vs(n,a):!n&&!a}function Vt(n,a){return n&&a?ws(n,a):!n&&!a}function B(n,a){return n&&a?wa(n,a):!n&&!a}function We(n,a){return n&&a?Do(n,a):!n&&!a}function dt(n,a,t){var e,r=Qe(a),o=Mn(t);try{e=ft(n,{start:r,end:o})}catch{e=!1}return e}function Er(){var n=Gn();return n.__localeId__}function at(n){if(typeof n=="string"){var a=Gn();return a.__localeData__?a.__localeData__[n]:void 0}else return n}function uc(n,a,t){return a(G(n,"EEEE",t))}function dc(n,a){return G(n,"EEEEEE",a)}function fc(n,a){return G(n,"EEE",a)}function Pr(n,a){return G(ce(V(),n),"LLLL",a)}function Xn(n,a){return G(ce(V(),n),"LLL",a)}function pc(n,a){return G(Ge(V(),n),"QQQ",a)}function le(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.maxDate,o=t.excludeDates,s=t.excludeDateIntervals,i=t.includeDates,c=t.includeDateIntervals,l=t.filterDate;return wt(n,{minDate:e,maxDate:r})||o&&o.some(function(u){return u instanceof Date?B(n,u):B(n,u.date)})||s&&s.some(function(u){var d=u.start,f=u.end;return ft(n,{start:d,end:f})})||i&&!i.some(function(u){return B(n,u)})||c&&!c.some(function(u){var d=u.start,f=u.end;return ft(n,{start:d,end:f})})||l&&!l(V(n))||!1}function Tr(n,a){var t=a===void 0?{}:a,e=t.excludeDates,r=t.excludeDateIntervals;return r&&r.length>0?r.some(function(o){var s=o.start,i=o.end;return ft(n,{start:s,end:i})}):e&&e.some(function(o){var s;return o instanceof Date?B(n,o):B(n,(s=o.date)!==null&&s!==void 0?s:new Date)})||!1}function Zn(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.maxDate,o=t.excludeDates,s=t.includeDates,i=t.filterDate;return wt(n,{minDate:e?Sn(e):void 0,maxDate:r?Cn(r):void 0})||(o==null?void 0:o.some(function(c){return ae(n,c instanceof Date?c:c.date)}))||s&&!s.some(function(c){return ae(n,c)})||i&&!i(V(n))||!1}function bt(n,a,t,e){var r=I(n),o=ne(n),s=I(a),i=ne(a),c=I(e);return r===s&&r===c?o<=t&&t<=i:r=t||cr:!1}function hc(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.maxDate,o=t.excludeDates,s=t.includeDates;return wt(n,{minDate:e,maxDate:r})||o&&o.some(function(i){return ae(i instanceof Date?i:i.date,n)})||s&&!s.some(function(i){return ae(i,n)})||!1}function kt(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.maxDate,o=t.excludeDates,s=t.includeDates,i=t.filterDate;return wt(n,{minDate:e,maxDate:r})||(o==null?void 0:o.some(function(c){return Vt(n,c instanceof Date?c:c.date)}))||s&&!s.some(function(c){return Vt(n,c)})||i&&!i(V(n))||!1}function xt(n,a,t){if(!a||!t||!Lt(a)||!Lt(t))return!1;var e=I(a),r=I(t);return e<=n&&r>=n}function Ot(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.maxDate,o=t.excludeDates,s=t.includeDates,i=t.filterDate,c=new Date(n,0,1);return wt(c,{minDate:e?Ut(e):void 0,maxDate:r?En(r):void 0})||(o==null?void 0:o.some(function(l){return ge(c,l instanceof Date?l:l.date)}))||s&&!s.some(function(l){return ge(c,l)})||i&&!i(V(c))||!1}function _t(n,a,t,e){var r=I(n),o=He(n),s=I(a),i=He(a),c=I(e);return r===s&&r===c?o<=t&&t<=i:r=t||cr:!1}function wt(n,a){var t,e=a===void 0?{}:a,r=e.minDate,o=e.maxDate;return(t=r&&Je(n,r)<0||o&&Je(n,o)>0)!==null&&t!==void 0?t:!1}function nn(n,a){return a.some(function(t){return Me(t)===Me(n)&&Ce(t)===Ce(n)&&Ne(t)===Ne(n)})}function an(n,a){var t=a===void 0?{}:a,e=t.excludeTimes,r=t.includeTimes,o=t.filterTime;return e&&nn(n,e)||r&&!nn(n,r)||o&&!o(n)||!1}function on(n,a){var t=a.minTime,e=a.maxTime;if(!t||!e)throw new Error("Both minTime and maxTime props required");var r=V();r=Ct(r,Me(n)),r=St(r,Ce(n)),r=Et(r,Ne(n));var o=V();o=Ct(o,Me(t)),o=St(o,Ce(t)),o=Et(o,Ne(t));var s=V();s=Ct(s,Me(e)),s=St(s,Ce(e)),s=Et(s,Ne(e));var i;try{i=!ft(r,{start:o,end:s})}catch{i=!1}return i}function sn(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.includeDates,o=je(n,1);return e&&Ft(e,o)>0||r&&r.every(function(s){return Ft(s,o)>0})||!1}function cn(n,a){var t=a===void 0?{}:a,e=t.maxDate,r=t.includeDates,o=he(n,1);return e&&Ft(o,e)>0||r&&r.every(function(s){return Ft(o,s)>0})||!1}function mc(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.includeDates,o=Ut(n),s=In(o);return e&&It(e,s)>0||r&&r.every(function(i){return It(i,s)>0})||!1}function gc(n,a){var t=a===void 0?{}:a,e=t.maxDate,r=t.includeDates,o=En(n),s=vr(o,1);return e&&It(s,e)>0||r&&r.every(function(i){return It(s,i)>0})||!1}function ln(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.includeDates,o=et(n,1);return e&&At(e,o)>0||r&&r.every(function(s){return At(s,o)>0})||!1}function vc(n,a){var t=a===void 0?{}:a,e=t.minDate,r=t.yearItemNumber,o=r===void 0?vt:r,s=ut(et(n,o)),i=Re(s,o).endPeriod,c=e&&I(e);return c&&c>i||!1}function un(n,a){var t=a===void 0?{}:a,e=t.maxDate,r=t.includeDates,o=xe(n,1);return e&&At(o,e)>0||r&&r.every(function(s){return At(o,s)>0})||!1}function wc(n,a){var t=a===void 0?{}:a,e=t.maxDate,r=t.yearItemNumber,o=r===void 0?vt:r,s=xe(n,o),i=Re(s,o).startPeriod,c=e&&I(e);return c&&c=0});return Lr(e)}else return t?Lr(t):a}function ea(n){var a=n.maxDate,t=n.includeDates;if(t&&a){var e=t.filter(function(r){return Je(r,a)<=0});return Yr(e)}else return t?Yr(t):a}function dn(n,a){var t;n===void 0&&(n=[]),a===void 0&&(a="react-datepicker__day--highlighted");for(var e=new Map,r=0,o=n.length;r=Ec,p=!r&&!t.isWeekInMonth(i);if(f||p)if(t.props.peekNextMonth)s=!0;else break}return e},t.onMonthClick=function(e,r){var o=t.isMonthDisabledForLabelDate(r),s=o.isDisabled,i=o.labelDate;s||t.handleDayClick(Se(i),e)},t.onMonthMouseEnter=function(e){var r=t.isMonthDisabledForLabelDate(e),o=r.isDisabled,s=r.labelDate;o||t.handleDayMouseEnter(Se(s))},t.handleMonthNavigation=function(e,r){var o,s,i,c;(s=(o=t.props).setPreSelection)===null||s===void 0||s.call(o,r),(c=(i=t.MONTH_REFS[e])===null||i===void 0?void 0:i.current)===null||c===void 0||c.focus()},t.handleKeyboardNavigation=function(e,r,o){var s,i=t.props,c=i.selected,l=i.preSelection,u=i.setPreSelection,d=i.minDate,f=i.maxDate,p=i.showFourColumnMonthYearPicker,h=i.showTwoColumnMonthYearPicker;if(l){var w=gn(p,h),m=t.getVerticalOffset(w),y=(s=ir[w])===null||s===void 0?void 0:s.grid,_=function(T,C,D){var b,N,R=C,A=D;switch(T){case M.ArrowRight:R=he(C,Mt),A=D===11?0:D+Mt;break;case M.ArrowLeft:R=je(C,Mt),A=D===0?11:D-Mt;break;case M.ArrowUp:R=je(C,m),A=!((b=y==null?void 0:y[0])===null||b===void 0)&&b.includes(D)?D+12-m:D-m;break;case M.ArrowDown:R=he(C,m),A=!((N=y==null?void 0:y[y.length-1])===null||N===void 0)&&N.includes(D)?D-12+m:D+m;break}return{newCalculatedDate:R,newCalculatedMonth:A}},k=function(T,C,D){for(var b=40,N=T,R=!1,A=0,W=_(N,C,D),q=W.newCalculatedDate,Y=W.newCalculatedMonth;!R;){if(A>=b){q=C,Y=D;break}if(d&&qf){N=M.ArrowLeft;var H=_(N,q,Y);q=H.newCalculatedDate,Y=H.newCalculatedMonth}if(hc(q,t.props)){var H=_(N,q,Y);q=H.newCalculatedDate,Y=H.newCalculatedMonth}else R=!0;A++}return{newCalculatedDate:q,newCalculatedMonth:Y}};if(r===M.Enter){t.isMonthDisabled(o)||(t.onMonthClick(e,o),u==null||u(c));return}var E=k(r,l,o),O=E.newCalculatedDate,L=E.newCalculatedMonth;switch(r){case M.ArrowRight:case M.ArrowLeft:case M.ArrowUp:case M.ArrowDown:t.handleMonthNavigation(L,O);break}}},t.getVerticalOffset=function(e){var r,o;return(o=(r=ir[e])===null||r===void 0?void 0:r.verticalNavigationOffset)!==null&&o!==void 0?o:0},t.onMonthKeyDown=function(e,r){var o=t.props,s=o.disabledKeyboardNavigation,i=o.handleOnMonthKeyDown,c=e.key;c!==M.Tab&&e.preventDefault(),s||t.handleKeyboardNavigation(e,c,r),i&&i(e)},t.onQuarterClick=function(e,r){var o=Ge(t.props.day,r);kt(o,t.props)||t.handleDayClick(en(o),e)},t.onQuarterMouseEnter=function(e){var r=Ge(t.props.day,e);kt(r,t.props)||t.handleDayMouseEnter(en(r))},t.handleQuarterNavigation=function(e,r){var o,s,i,c;t.isDisabled(r)||t.isExcluded(r)||((s=(o=t.props).setPreSelection)===null||s===void 0||s.call(o,r),(c=(i=t.QUARTER_REFS[e-1])===null||i===void 0?void 0:i.current)===null||c===void 0||c.focus())},t.onQuarterKeyDown=function(e,r){var o,s,i=e.key;if(!t.props.disabledKeyboardNavigation)switch(i){case M.Enter:t.onQuarterClick(e,r),(s=(o=t.props).setPreSelection)===null||s===void 0||s.call(o,t.props.selected);break;case M.ArrowRight:if(!t.props.preSelection)break;t.handleQuarterNavigation(r===4?1:r+1,vr(t.props.preSelection,1));break;case M.ArrowLeft:if(!t.props.preSelection)break;t.handleQuarterNavigation(r===1?4:r-1,In(t.props.preSelection));break}},t.isMonthDisabledForLabelDate=function(e){var r,o=t.props,s=o.day,i=o.minDate,c=o.maxDate,l=o.excludeDates,u=o.includeDates,d=ce(s,e);return{isDisabled:(r=(i||c||l||u)&&Zn(d,t.props))!==null&&r!==void 0?r:!1,labelDate:d}},t.isMonthDisabled=function(e){var r=t.isMonthDisabledForLabelDate(e).isDisabled;return r},t.getMonthClassNames=function(e){var r=t.props,o=r.day,s=r.startDate,i=r.endDate,c=r.preSelection,l=r.monthClassName,u=l?l(ce(o,e)):void 0,d=t.getSelection();return re("react-datepicker__month-text","react-datepicker__month-".concat(e),u,{"react-datepicker__month-text--disabled":t.isMonthDisabled(e),"react-datepicker__month-text--selected":d?t.isSelectMonthInList(o,e,d):void 0,"react-datepicker__month-text--keyboard-selected":!t.props.disabledKeyboardNavigation&&c&&t.isSelectedMonth(o,e,c)&&!t.isMonthSelected()&&!t.isMonthDisabled(e),"react-datepicker__month-text--in-selecting-range":t.isInSelectingRangeMonth(e),"react-datepicker__month-text--in-range":s&&i?bt(s,i,e,o):void 0,"react-datepicker__month-text--range-start":t.isRangeStartMonth(e),"react-datepicker__month-text--range-end":t.isRangeEndMonth(e),"react-datepicker__month-text--selecting-range-start":t.isSelectingMonthRangeStart(e),"react-datepicker__month-text--selecting-range-end":t.isSelectingMonthRangeEnd(e),"react-datepicker__month-text--today":t.isCurrentMonth(o,e)})},t.getTabIndex=function(e){if(t.props.preSelection==null)return"-1";var r=ne(t.props.preSelection),o=t.isMonthDisabledForLabelDate(r).isDisabled,s=e===r&&!(o||t.props.disabledKeyboardNavigation)?"0":"-1";return s},t.getQuarterTabIndex=function(e){if(t.props.preSelection==null)return"-1";var r=He(t.props.preSelection),o=kt(t.props.day,t.props),s=e===r&&!(o||t.props.disabledKeyboardNavigation)?"0":"-1";return s},t.getAriaLabel=function(e){var r=t.props,o=r.chooseDayAriaLabelPrefix,s=o===void 0?"Choose":o,i=r.disabledDayAriaLabelPrefix,c=i===void 0?"Not available":i,l=r.day,u=r.locale,d=ce(l,e),f=t.isDisabled(d)||t.isExcluded(d)?c:s;return"".concat(f," ").concat(G(d,"MMMM yyyy",u))},t.getQuarterClassNames=function(e){var r=t.props,o=r.day,s=r.startDate,i=r.endDate,c=r.selected,l=r.minDate,u=r.maxDate,d=r.excludeDates,f=r.includeDates,p=r.filterDate,h=r.preSelection,w=r.disabledKeyboardNavigation,m=(l||u||d||f||p)&&kt(Ge(o,e),t.props);return re("react-datepicker__quarter-text","react-datepicker__quarter-".concat(e),{"react-datepicker__quarter-text--disabled":m,"react-datepicker__quarter-text--selected":c?t.isSelectedQuarter(o,e,c):void 0,"react-datepicker__quarter-text--keyboard-selected":!w&&h&&t.isSelectedQuarter(o,e,h)&&!m,"react-datepicker__quarter-text--in-selecting-range":t.isInSelectingRangeQuarter(e),"react-datepicker__quarter-text--in-range":s&&i?_t(s,i,e,o):void 0,"react-datepicker__quarter-text--range-start":t.isRangeStartQuarter(e),"react-datepicker__quarter-text--range-end":t.isRangeEndQuarter(e),"react-datepicker__quarter-text--today":t.isCurrentQuarter(o,e)})},t.getMonthContent=function(e){var r=t.props,o=r.showFullMonthYearPicker,s=r.renderMonthContent,i=r.locale,c=r.day,l=Xn(e,i),u=Pr(e,i);return s?s(e,l,u,c):o?u:l},t.getQuarterContent=function(e){var r,o=t.props,s=o.renderQuarterContent,i=o.locale,c=pc(e,i);return(r=s==null?void 0:s(e,c))!==null&&r!==void 0?r:c},t.renderMonths=function(){var e,r=t.props,o=r.showTwoColumnMonthYearPicker,s=r.showFourColumnMonthYearPicker,i=r.day,c=r.selected,l=(e=ir[gn(s,o)])===null||e===void 0?void 0:e.grid;return l==null?void 0:l.map(function(u,d){return g.createElement("div",{className:"react-datepicker__month-wrapper",key:d},u.map(function(f,p){return g.createElement("div",{ref:t.MONTH_REFS[f],key:p,onClick:function(h){t.onMonthClick(h,f)},onKeyDown:function(h){ta(h)&&(h.preventDefault(),h.key=M.Enter),t.onMonthKeyDown(h,f)},onMouseEnter:t.props.usePointerEvent?void 0:function(){return t.onMonthMouseEnter(f)},onPointerEnter:t.props.usePointerEvent?function(){return t.onMonthMouseEnter(f)}:void 0,tabIndex:Number(t.getTabIndex(f)),className:t.getMonthClassNames(f),"aria-disabled":t.isMonthDisabled(f),role:"option","aria-label":t.getAriaLabel(f),"aria-current":t.isCurrentMonth(i,f)?"date":void 0,"aria-selected":c?t.isSelectedMonth(i,f,c):void 0},t.getMonthContent(f))}))})},t.renderQuarters=function(){var e=t.props,r=e.day,o=e.selected,s=[1,2,3,4];return g.createElement("div",{className:"react-datepicker__quarter-wrapper"},s.map(function(i,c){return g.createElement("div",{key:c,ref:t.QUARTER_REFS[c],role:"option",onClick:function(l){t.onQuarterClick(l,i)},onKeyDown:function(l){t.onQuarterKeyDown(l,i)},onMouseEnter:t.props.usePointerEvent?void 0:function(){return t.onQuarterMouseEnter(i)},onPointerEnter:t.props.usePointerEvent?function(){return t.onQuarterMouseEnter(i)}:void 0,className:t.getQuarterClassNames(i),"aria-selected":o?t.isSelectedQuarter(r,i,o):void 0,tabIndex:Number(t.getQuarterTabIndex(i)),"aria-current":t.isCurrentQuarter(r,i)?"date":void 0},t.getQuarterContent(i))}))},t.getClassNames=function(){var e=t.props,r=e.selectingDate,o=e.selectsStart,s=e.selectsEnd,i=e.showMonthYearPicker,c=e.showQuarterYearPicker,l=e.showWeekPicker;return re("react-datepicker__month",{"react-datepicker__month--selecting-range":r&&(o||s)},{"react-datepicker__monthPicker":i},{"react-datepicker__quarterPicker":c},{"react-datepicker__weekPicker":l})},t}return a.prototype.getSelection=function(){var t=this.props,e=t.selected,r=t.selectedDates,o=t.selectsMultiple;if(o)return r;if(e)return[e]},a.prototype.render=function(){var t=this.props,e=t.showMonthYearPicker,r=t.showQuarterYearPicker,o=t.day,s=t.ariaLabelPrefix,i=s===void 0?"Month ":s,c=i?i.trim()+" ":"";return g.createElement("div",{className:this.getClassNames(),onMouseLeave:this.props.usePointerEvent?void 0:this.handleMouseLeave,onPointerLeave:this.props.usePointerEvent?this.handleMouseLeave:void 0,"aria-label":"".concat(c).concat(G(o,"MMMM, yyyy",this.props.locale)),role:"listbox"},e?this.renderMonths():r?this.renderQuarters():this.renderWeeks())},a}(v.Component),Tc=function(n){te(a,n);function a(){var t=n!==null&&n.apply(this,arguments)||this;return t.isSelectedMonth=function(e){return t.props.month===e},t.renderOptions=function(){return t.props.monthNames.map(function(e,r){return g.createElement("div",{className:t.isSelectedMonth(r)?"react-datepicker__month-option react-datepicker__month-option--selected_month":"react-datepicker__month-option",key:e,onClick:t.onChange.bind(t,r),"aria-selected":t.isSelectedMonth(r)?"true":void 0},t.isSelectedMonth(r)?g.createElement("span",{className:"react-datepicker__month-option--selected"},"✓"):"",e)})},t.onChange=function(e){return t.props.onChange(e)},t.handleClickOutside=function(){return t.props.onCancel()},t}return a.prototype.render=function(){return g.createElement(Jt,{className:"react-datepicker__month-dropdown",onClickOutside:this.handleClickOutside},this.renderOptions())},a}(v.Component),Oc=function(n){te(a,n);function a(){var t=n!==null&&n.apply(this,arguments)||this;return t.state={dropdownVisible:!1},t.renderSelectOptions=function(e){return e.map(function(r,o){return g.createElement("option",{key:r,value:o},r)})},t.renderSelectMode=function(e){return g.createElement("select",{value:t.props.month,className:"react-datepicker__month-select",onChange:function(r){return t.onChange(parseInt(r.target.value))}},t.renderSelectOptions(e))},t.renderReadView=function(e,r){return g.createElement("div",{key:"read",style:{visibility:e?"visible":"hidden"},className:"react-datepicker__month-read-view",onClick:t.toggleDropdown},g.createElement("span",{className:"react-datepicker__month-read-view--down-arrow"}),g.createElement("span",{className:"react-datepicker__month-read-view--selected-month"},r[t.props.month]))},t.renderDropdown=function(e){return g.createElement(Tc,Q({key:"dropdown"},t.props,{monthNames:e,onChange:t.onChange,onCancel:t.toggleDropdown}))},t.renderScrollMode=function(e){var r=t.state.dropdownVisible,o=[t.renderReadView(!r,e)];return r&&o.unshift(t.renderDropdown(e)),o},t.onChange=function(e){t.toggleDropdown(),e!==t.props.month&&t.props.onChange(e)},t.toggleDropdown=function(){return t.setState({dropdownVisible:!t.state.dropdownVisible})},t}return a.prototype.render=function(){var t=this,e=[0,1,2,3,4,5,6,7,8,9,10,11].map(this.props.useShortMonthInDropdown?function(o){return Xn(o,t.props.locale)}:function(o){return Pr(o,t.props.locale)}),r;switch(this.props.dropdownMode){case"scroll":r=this.renderScrollMode(e);break;case"select":r=this.renderSelectMode(e);break}return g.createElement("div",{className:"react-datepicker__month-dropdown-container react-datepicker__month-dropdown-container--".concat(this.props.dropdownMode)},r)},a}(v.Component);function Rc(n,a){for(var t=[],e=Se(n),r=Se(a);!Le(e,r);)t.push(V(e)),e=he(e,1);return t}var Nc=function(n){te(a,n);function a(t){var e=n.call(this,t)||this;return e.renderOptions=function(){return e.state.monthYearsList.map(function(r){var o=pr(r),s=ge(e.props.date,r)&&ae(e.props.date,r);return g.createElement("div",{className:s?"react-datepicker__month-year-option--selected_month-year":"react-datepicker__month-year-option",key:o,onClick:e.onChange.bind(e,o),"aria-selected":s?"true":void 0},s?g.createElement("span",{className:"react-datepicker__month-year-option--selected"},"✓"):"",G(r,e.props.dateFormat,e.props.locale))})},e.onChange=function(r){return e.props.onChange(r)},e.handleClickOutside=function(){e.props.onCancel()},e.state={monthYearsList:Rc(e.props.minDate,e.props.maxDate)},e}return a.prototype.render=function(){var t=re({"react-datepicker__month-year-dropdown":!0,"react-datepicker__month-year-dropdown--scrollable":this.props.scrollableMonthYearDropdown});return g.createElement(Jt,{className:t,onClickOutside:this.handleClickOutside},this.renderOptions())},a}(v.Component),Yc=function(n){te(a,n);function a(){var t=n!==null&&n.apply(this,arguments)||this;return t.state={dropdownVisible:!1},t.renderSelectOptions=function(){for(var e=Se(t.props.minDate),r=Se(t.props.maxDate),o=[];!Le(e,r);){var s=pr(e);o.push(g.createElement("option",{key:s,value:s},G(e,t.props.dateFormat,t.props.locale))),e=he(e,1)}return o},t.onSelectChange=function(e){t.onChange(parseInt(e.target.value))},t.renderSelectMode=function(){return g.createElement("select",{value:pr(Se(t.props.date)),className:"react-datepicker__month-year-select",onChange:t.onSelectChange},t.renderSelectOptions())},t.renderReadView=function(e){var r=G(t.props.date,t.props.dateFormat,t.props.locale);return g.createElement("div",{key:"read",style:{visibility:e?"visible":"hidden"},className:"react-datepicker__month-year-read-view",onClick:t.toggleDropdown},g.createElement("span",{className:"react-datepicker__month-year-read-view--down-arrow"}),g.createElement("span",{className:"react-datepicker__month-year-read-view--selected-month-year"},r))},t.renderDropdown=function(){return g.createElement(Nc,Q({key:"dropdown"},t.props,{onChange:t.onChange,onCancel:t.toggleDropdown}))},t.renderScrollMode=function(){var e=t.state.dropdownVisible,r=[t.renderReadView(!e)];return e&&r.unshift(t.renderDropdown()),r},t.onChange=function(e){t.toggleDropdown();var r=V(e);ge(t.props.date,r)&&ae(t.props.date,r)||t.props.onChange(r)},t.toggleDropdown=function(){return t.setState({dropdownVisible:!t.state.dropdownVisible})},t}return a.prototype.render=function(){var t;switch(this.props.dropdownMode){case"scroll":t=this.renderScrollMode();break;case"select":t=this.renderSelectMode();break}return g.createElement("div",{className:"react-datepicker__month-year-dropdown-container react-datepicker__month-year-dropdown-container--".concat(this.props.dropdownMode)},t)},a}(v.Component),Lc=function(n){te(a,n);function a(){var t=n!==null&&n.apply(this,arguments)||this;return t.state={height:null},t.scrollToTheSelectedTime=function(){requestAnimationFrame(function(){var e,r,o;t.list&&(t.list.scrollTop=(o=t.centerLi&&a.calcCenterPosition(t.props.monthRef?t.props.monthRef.clientHeight-((r=(e=t.header)===null||e===void 0?void 0:e.clientHeight)!==null&&r!==void 0?r:0):t.list.clientHeight,t.centerLi))!==null&&o!==void 0?o:0)})},t.handleClick=function(e){var r,o;(t.props.minTime||t.props.maxTime)&&on(e,t.props)||(t.props.excludeTimes||t.props.includeTimes||t.props.filterTime)&&an(e,t.props)||(o=(r=t.props).onChange)===null||o===void 0||o.call(r,e)},t.isSelectedTime=function(e){return t.props.selected&&xc(t.props.selected,e)},t.isDisabledTime=function(e){return(t.props.minTime||t.props.maxTime)&&on(e,t.props)||(t.props.excludeTimes||t.props.includeTimes||t.props.filterTime)&&an(e,t.props)},t.liClasses=function(e){var r,o=["react-datepicker__time-list-item",t.props.timeClassName?t.props.timeClassName(e):void 0];return t.isSelectedTime(e)&&o.push("react-datepicker__time-list-item--selected"),t.isDisabledTime(e)&&o.push("react-datepicker__time-list-item--disabled"),t.props.injectTimes&&(Me(e)*3600+Ce(e)*60+Ne(e))%(((r=t.props.intervals)!==null&&r!==void 0?r:a.defaultProps.intervals)*60)!==0&&o.push("react-datepicker__time-list-item--injected"),o.join(" ")},t.handleOnKeyDown=function(e,r){var o,s;e.key===M.Space&&(e.preventDefault(),e.key=M.Enter),(e.key===M.ArrowUp||e.key===M.ArrowLeft)&&e.target instanceof HTMLElement&&e.target.previousSibling&&(e.preventDefault(),e.target.previousSibling instanceof HTMLElement&&e.target.previousSibling.focus()),(e.key===M.ArrowDown||e.key===M.ArrowRight)&&e.target instanceof HTMLElement&&e.target.nextSibling&&(e.preventDefault(),e.target.nextSibling instanceof HTMLElement&&e.target.nextSibling.focus()),e.key===M.Enter&&t.handleClick(r),(s=(o=t.props).handleOnKeyDown)===null||s===void 0||s.call(o,e)},t.renderTimes=function(){for(var e,r=[],o=typeof t.props.format=="string"?t.props.format:"p",s=(e=t.props.intervals)!==null&&e!==void 0?e:a.defaultProps.intervals,i=t.props.selected||t.props.openToDate||V(),c=Tt(i),l=t.props.injectTimes&&t.props.injectTimes.sort(function(m,y){return m.getTime()-y.getTime()}),u=60*kc(i),d=u/s,f=0;f=f?e.updateFocusOnPaginate(Math.abs(f-(r-p))):(l=(c=e.YEAR_REFS[r-p])===null||c===void 0?void 0:c.current)===null||l===void 0||l.focus())}},e.isSameDay=function(r,o){return B(r,o)},e.isCurrentYear=function(r){return r===I(V())},e.isRangeStart=function(r){return e.props.startDate&&e.props.endDate&&ge(me(V(),r),e.props.startDate)},e.isRangeEnd=function(r){return e.props.startDate&&e.props.endDate&&ge(me(V(),r),e.props.endDate)},e.isInRange=function(r){return xt(r,e.props.startDate,e.props.endDate)},e.isInSelectingRange=function(r){var o=e.props,s=o.selectsStart,i=o.selectsEnd,c=o.selectsRange,l=o.startDate,u=o.endDate;return!(s||i||c)||!e.selectingDate()?!1:s&&u?xt(r,e.selectingDate(),u):i&&l||c&&l&&!u?xt(r,l,e.selectingDate()):!1},e.isSelectingRangeStart=function(r){var o;if(!e.isInSelectingRange(r))return!1;var s=e.props,i=s.startDate,c=s.selectsStart,l=me(V(),r);return c?ge(l,(o=e.selectingDate())!==null&&o!==void 0?o:null):ge(l,i??null)},e.isSelectingRangeEnd=function(r){var o;if(!e.isInSelectingRange(r))return!1;var s=e.props,i=s.endDate,c=s.selectsEnd,l=s.selectsRange,u=me(V(),r);return c||l?ge(u,(o=e.selectingDate())!==null&&o!==void 0?o:null):ge(u,i??null)},e.isKeyboardSelected=function(r){if(!(e.props.date===void 0||e.props.selected==null||e.props.preSelection==null)){var o=e.props,s=o.minDate,i=o.maxDate,c=o.excludeDates,l=o.includeDates,u=o.filterDate,d=ut(me(e.props.date,r)),f=(s||i||c||l||u)&&Ot(r,e.props);return!e.props.disabledKeyboardNavigation&&!e.props.inline&&!B(d,ut(e.props.selected))&&B(d,ut(e.props.preSelection))&&!f}},e.isSelectedYear=function(r){var o=e.props,s=o.selectsMultiple,i=o.selected,c=o.selectedDates;return s?c==null?void 0:c.some(function(l){return r===I(l)}):!!i&&r===I(i)},e.onYearClick=function(r,o){var s=e.props.date;s!==void 0&&e.handleYearClick(ut(me(s,o)),r)},e.onYearKeyDown=function(r,o){var s,i,c=r.key,l=e.props,u=l.date,d=l.yearItemNumber,f=l.handleOnKeyDown;if(c!==M.Tab&&r.preventDefault(),!e.props.disabledKeyboardNavigation)switch(c){case M.Enter:if(e.props.selected==null)break;e.onYearClick(r,o),(i=(s=e.props).setPreSelection)===null||i===void 0||i.call(s,e.props.selected);break;case M.ArrowRight:if(e.props.preSelection==null)break;e.handleYearNavigation(o+1,xe(e.props.preSelection,1));break;case M.ArrowLeft:if(e.props.preSelection==null)break;e.handleYearNavigation(o-1,et(e.props.preSelection,1));break;case M.ArrowUp:{if(u===void 0||d===void 0||e.props.preSelection==null)break;var p=Re(u,d).startPeriod,h=vn,w=o-h;if(w=p&&oy){var m=d%h;o<=y&&o>y-m?h=m:h+=m,w=o+h}e.handleYearNavigation(w,xe(e.props.preSelection,h));break}}f&&f(r)},e.getYearClassNames=function(r){var o=e.props,s=o.date,i=o.minDate,c=o.maxDate,l=o.excludeDates,u=o.includeDates,d=o.filterDate,f=o.yearClassName;return re("react-datepicker__year-text","react-datepicker__year-".concat(r),s?f==null?void 0:f(me(s,r)):void 0,{"react-datepicker__year-text--selected":e.isSelectedYear(r),"react-datepicker__year-text--disabled":(i||c||l||u||d)&&Ot(r,e.props),"react-datepicker__year-text--keyboard-selected":e.isKeyboardSelected(r),"react-datepicker__year-text--range-start":e.isRangeStart(r),"react-datepicker__year-text--range-end":e.isRangeEnd(r),"react-datepicker__year-text--in-range":e.isInRange(r),"react-datepicker__year-text--in-selecting-range":e.isInSelectingRange(r),"react-datepicker__year-text--selecting-range-start":e.isSelectingRangeStart(r),"react-datepicker__year-text--selecting-range-end":e.isSelectingRangeEnd(r),"react-datepicker__year-text--today":e.isCurrentYear(r)})},e.getYearTabIndex=function(r){if(e.props.disabledKeyboardNavigation||e.props.preSelection==null)return"-1";var o=I(e.props.preSelection),s=Ot(r,e.props);return r===o&&!s?"0":"-1"},e.getYearContent=function(r){return e.props.renderYearContent?e.props.renderYearContent(r):r},e}return a.prototype.render=function(){var t=this,e=[],r=this.props,o=r.date,s=r.yearItemNumber,i=r.onYearMouseEnter,c=r.onYearMouseLeave;if(o===void 0)return null;for(var l=Re(o,s),u=l.startPeriod,d=l.endPeriod,f=function(w){e.push(g.createElement("div",{ref:p.YEAR_REFS[w-u],onClick:function(m){t.onYearClick(m,w)},onKeyDown:function(m){ta(m)&&(m.preventDefault(),m.key=M.Enter),t.onYearKeyDown(m,w)},tabIndex:Number(p.getYearTabIndex(w)),className:p.getYearClassNames(w),onMouseEnter:p.props.usePointerEvent?void 0:function(m){return i(m,w)},onPointerEnter:p.props.usePointerEvent?function(m){return i(m,w)}:void 0,onMouseLeave:p.props.usePointerEvent?void 0:function(m){return c(m,w)},onPointerLeave:p.props.usePointerEvent?function(m){return c(m,w)}:void 0,key:w,"aria-current":p.isCurrentYear(w)?"date":void 0},p.getYearContent(w)))},p=this,h=u;h<=d;h++)f(h);return g.createElement("div",{className:"react-datepicker__year"},g.createElement("div",{className:"react-datepicker__year-wrapper",onMouseLeave:this.props.usePointerEvent?void 0:this.props.clearSelectingDate,onPointerLeave:this.props.usePointerEvent?this.props.clearSelectingDate:void 0},e))},a}(v.Component);function Ic(n,a,t,e){for(var r=[],o=0;o<2*a+1;o++){var s=n+a-o,i=!0;t&&(i=I(t)<=s),e&&i&&(i=I(e)>=s),i&&r.push(s)}return r}var Ac=function(n){te(a,n);function a(t){var e=n.call(this,t)||this;e.renderOptions=function(){var i=e.props.year,c=e.state.yearsList.map(function(d){return g.createElement("div",{className:i===d?"react-datepicker__year-option react-datepicker__year-option--selected_year":"react-datepicker__year-option",key:d,onClick:e.onChange.bind(e,d),"aria-selected":i===d?"true":void 0},i===d?g.createElement("span",{className:"react-datepicker__year-option--selected"},"✓"):"",d)}),l=e.props.minDate?I(e.props.minDate):null,u=e.props.maxDate?I(e.props.maxDate):null;return(!u||!e.state.yearsList.find(function(d){return d===u}))&&c.unshift(g.createElement("div",{className:"react-datepicker__year-option",key:"upcoming",onClick:e.incrementYears},g.createElement("a",{className:"react-datepicker__navigation react-datepicker__navigation--years react-datepicker__navigation--years-upcoming"}))),(!l||!e.state.yearsList.find(function(d){return d===l}))&&c.push(g.createElement("div",{className:"react-datepicker__year-option",key:"previous",onClick:e.decrementYears},g.createElement("a",{className:"react-datepicker__navigation react-datepicker__navigation--years react-datepicker__navigation--years-previous"}))),c},e.onChange=function(i){e.props.onChange(i)},e.handleClickOutside=function(){e.props.onCancel()},e.shiftYears=function(i){var c=e.state.yearsList.map(function(l){return l+i});e.setState({yearsList:c})},e.incrementYears=function(){return e.shiftYears(1)},e.decrementYears=function(){return e.shiftYears(-1)};var r=t.yearDropdownItemNumber,o=t.scrollableYearDropdown,s=r||(o?10:5);return e.state={yearsList:Ic(e.props.year,s,e.props.minDate,e.props.maxDate)},e.dropdownRef=v.createRef(),e}return a.prototype.componentDidMount=function(){var t=this.dropdownRef.current;if(t){var e=t.children?Array.from(t.children):null,r=e?e.find(function(o){return o.ariaSelected}):null;t.scrollTop=r&&r instanceof HTMLElement?r.offsetTop+(r.clientHeight-t.clientHeight)/2:(t.scrollHeight-t.clientHeight)/2}},a.prototype.render=function(){var t=re({"react-datepicker__year-dropdown":!0,"react-datepicker__year-dropdown--scrollable":this.props.scrollableYearDropdown});return g.createElement(Jt,{className:t,containerRef:this.dropdownRef,onClickOutside:this.handleClickOutside},this.renderOptions())},a}(v.Component),Wc=function(n){te(a,n);function a(){var t=n!==null&&n.apply(this,arguments)||this;return t.state={dropdownVisible:!1},t.renderSelectOptions=function(){for(var e=t.props.minDate?I(t.props.minDate):1900,r=t.props.maxDate?I(t.props.maxDate):2100,o=[],s=e;s<=r;s++)o.push(g.createElement("option",{key:s,value:s},s));return o},t.onSelectChange=function(e){t.onChange(parseInt(e.target.value))},t.renderSelectMode=function(){return g.createElement("select",{value:t.props.year,className:"react-datepicker__year-select",onChange:t.onSelectChange},t.renderSelectOptions())},t.renderReadView=function(e){return g.createElement("div",{key:"read",style:{visibility:e?"visible":"hidden"},className:"react-datepicker__year-read-view",onClick:function(r){return t.toggleDropdown(r)}},g.createElement("span",{className:"react-datepicker__year-read-view--down-arrow"}),g.createElement("span",{className:"react-datepicker__year-read-view--selected-year"},t.props.year))},t.renderDropdown=function(){return g.createElement(Ac,Q({key:"dropdown"},t.props,{onChange:t.onChange,onCancel:t.toggleDropdown}))},t.renderScrollMode=function(){var e=t.state.dropdownVisible,r=[t.renderReadView(!e)];return e&&r.unshift(t.renderDropdown()),r},t.onChange=function(e){t.toggleDropdown(),e!==t.props.year&&t.props.onChange(e)},t.toggleDropdown=function(e){t.setState({dropdownVisible:!t.state.dropdownVisible},function(){t.props.adjustDateOnChange&&t.handleYearChange(t.props.date,e)})},t.handleYearChange=function(e,r){var o;(o=t.onSelect)===null||o===void 0||o.call(t,e,r),t.setOpen()},t.onSelect=function(e,r){var o,s;(s=(o=t.props).onSelect)===null||s===void 0||s.call(o,e,r)},t.setOpen=function(){var e,r;(r=(e=t.props).setOpen)===null||r===void 0||r.call(e,!0)},t}return a.prototype.render=function(){var t;switch(this.props.dropdownMode){case"scroll":t=this.renderScrollMode();break;case"select":t=this.renderSelectMode();break}return g.createElement("div",{className:"react-datepicker__year-dropdown-container react-datepicker__year-dropdown-container--".concat(this.props.dropdownMode)},t)},a}(v.Component),Hc=["react-datepicker__year-select","react-datepicker__month-select","react-datepicker__month-year-select"],jc=function(n){var a=(n.className||"").split(/\s+/);return Hc.some(function(t){return a.indexOf(t)>=0})},Bc=function(n){te(a,n);function a(t){var e=n.call(this,t)||this;return e.monthContainer=void 0,e.handleClickOutside=function(r){e.props.onClickOutside(r)},e.setClickOutsideRef=function(){return e.containerRef.current},e.handleDropdownFocus=function(r){var o,s;jc(r.target)&&((s=(o=e.props).onDropdownFocus)===null||s===void 0||s.call(o,r))},e.getDateInView=function(){var r=e.props,o=r.preSelection,s=r.selected,i=r.openToDate,c=Jn(e.props),l=ea(e.props),u=V(),d=i||s||o;return d||(c&&$e(u,c)?c:l&&Le(u,l)?l:u)},e.increaseMonth=function(){e.setState(function(r){var o=r.date;return{date:he(o,1)}},function(){return e.handleMonthChange(e.state.date)})},e.decreaseMonth=function(){e.setState(function(r){var o=r.date;return{date:je(o,1)}},function(){return e.handleMonthChange(e.state.date)})},e.handleDayClick=function(r,o,s){e.props.onSelect(r,o,s),e.props.setPreSelection&&e.props.setPreSelection(r)},e.handleDayMouseEnter=function(r){e.setState({selectingDate:r}),e.props.onDayMouseEnter&&e.props.onDayMouseEnter(r)},e.handleMonthMouseLeave=function(){e.setState({selectingDate:void 0}),e.props.onMonthMouseLeave&&e.props.onMonthMouseLeave()},e.handleYearMouseEnter=function(r,o){e.setState({selectingDate:me(V(),o)}),e.props.onYearMouseEnter&&e.props.onYearMouseEnter(r,o)},e.handleYearMouseLeave=function(r,o){e.props.onYearMouseLeave&&e.props.onYearMouseLeave(r,o)},e.handleYearChange=function(r){var o,s,i,c;(s=(o=e.props).onYearChange)===null||s===void 0||s.call(o,r),e.setState({isRenderAriaLiveMessage:!0}),e.props.adjustDateOnChange&&(e.props.onSelect(r),(c=(i=e.props).setOpen)===null||c===void 0||c.call(i,!0)),e.props.setPreSelection&&e.props.setPreSelection(r)},e.getEnabledPreSelectionDateForMonth=function(r){if(!le(r,e.props))return r;for(var o=Se(r),s=lc(r),i=ya(s,o),c=null,l=0;l<=i;l++){var u=pe(o,l);if(!le(u,e.props)){c=u;break}}return c},e.handleMonthChange=function(r){var o,s,i,c=(o=e.getEnabledPreSelectionDateForMonth(r))!==null&&o!==void 0?o:r;e.handleCustomMonthChange(c),e.props.adjustDateOnChange&&(e.props.onSelect(c),(i=(s=e.props).setOpen)===null||i===void 0||i.call(s,!0)),e.props.setPreSelection&&e.props.setPreSelection(c)},e.handleCustomMonthChange=function(r){var o,s;(s=(o=e.props).onMonthChange)===null||s===void 0||s.call(o,r),e.setState({isRenderAriaLiveMessage:!0})},e.handleMonthYearChange=function(r){e.handleYearChange(r),e.handleMonthChange(r)},e.changeYear=function(r){e.setState(function(o){var s=o.date;return{date:me(s,Number(r))}},function(){return e.handleYearChange(e.state.date)})},e.changeMonth=function(r){e.setState(function(o){var s=o.date;return{date:ce(s,Number(r))}},function(){return e.handleMonthChange(e.state.date)})},e.changeMonthYear=function(r){e.setState(function(o){var s=o.date;return{date:me(ce(s,ne(r)),I(r))}},function(){return e.handleMonthYearChange(e.state.date)})},e.header=function(r){r===void 0&&(r=e.state.date);var o=Ye(r,e.props.locale,e.props.calendarStartDay),s=[];return e.props.showWeekNumbers&&s.push(g.createElement("div",{key:"W",className:"react-datepicker__day-name"},e.props.weekLabel||"#")),s.concat([0,1,2,3,4,5,6].map(function(i){var c=pe(o,i),l=e.formatWeekday(c,e.props.locale),u=e.props.weekDayClassName?e.props.weekDayClassName(c):void 0;return g.createElement("div",{key:i,"aria-label":G(c,"EEEE",e.props.locale),className:re("react-datepicker__day-name",u)},l)}))},e.formatWeekday=function(r,o){return e.props.formatWeekDay?uc(r,e.props.formatWeekDay,o):e.props.useWeekdaysShort?fc(r,o):dc(r,o)},e.decreaseYear=function(){e.setState(function(r){var o,s=r.date;return{date:et(s,e.props.showYearPicker?(o=e.props.yearItemNumber)!==null&&o!==void 0?o:a.defaultProps.yearItemNumber:1)}},function(){return e.handleYearChange(e.state.date)})},e.clearSelectingDate=function(){e.setState({selectingDate:void 0})},e.renderPreviousButton=function(){var r,o,s;if(!e.props.renderCustomHeader){var i=(r=e.props.monthsShown)!==null&&r!==void 0?r:a.defaultProps.monthsShown,c=e.props.showPreviousMonths?i-1:0,l=(o=e.props.monthSelectedIn)!==null&&o!==void 0?o:c,u=je(e.state.date,l),d;switch(!0){case e.props.showMonthYearPicker:d=ln(e.state.date,e.props);break;case e.props.showYearPicker:d=vc(e.state.date,e.props);break;case e.props.showQuarterYearPicker:d=mc(e.state.date,e.props);break;default:d=sn(u,e.props);break}if(!(!((s=e.props.forceShowMonthNavigation)!==null&&s!==void 0?s:a.defaultProps.forceShowMonthNavigation)&&!e.props.showDisabledMonthNavigation&&d||e.props.showTimeSelectOnly)){var f=["react-datepicker__navigation-icon","react-datepicker__navigation-icon--previous"],p=["react-datepicker__navigation","react-datepicker__navigation--previous"],h=e.decreaseMonth;(e.props.showMonthYearPicker||e.props.showQuarterYearPicker||e.props.showYearPicker)&&(h=e.decreaseYear),d&&e.props.showDisabledMonthNavigation&&(p.push("react-datepicker__navigation--previous--disabled"),h=void 0);var w=e.props.showMonthYearPicker||e.props.showQuarterYearPicker||e.props.showYearPicker,m=e.props,y=m.previousMonthButtonLabel,_=y===void 0?a.defaultProps.previousMonthButtonLabel:y,k=m.previousYearButtonLabel,E=k===void 0?a.defaultProps.previousYearButtonLabel:k,O=e.props,L=O.previousMonthAriaLabel,T=L===void 0?typeof _=="string"?_:"Previous Month":L,C=O.previousYearAriaLabel,D=C===void 0?typeof E=="string"?E:"Previous Year":C;return g.createElement("button",{type:"button",className:p.join(" "),onClick:h,onKeyDown:e.props.handleOnKeyDown,"aria-label":w?D:T},g.createElement("span",{className:f.join(" ")},w?E:_))}}},e.increaseYear=function(){e.setState(function(r){var o,s=r.date;return{date:xe(s,e.props.showYearPicker?(o=e.props.yearItemNumber)!==null&&o!==void 0?o:a.defaultProps.yearItemNumber:1)}},function(){return e.handleYearChange(e.state.date)})},e.renderNextButton=function(){var r;if(!e.props.renderCustomHeader){var o;switch(!0){case e.props.showMonthYearPicker:o=un(e.state.date,e.props);break;case e.props.showYearPicker:o=wc(e.state.date,e.props);break;case e.props.showQuarterYearPicker:o=gc(e.state.date,e.props);break;default:o=cn(e.state.date,e.props);break}if(!(!((r=e.props.forceShowMonthNavigation)!==null&&r!==void 0?r:a.defaultProps.forceShowMonthNavigation)&&!e.props.showDisabledMonthNavigation&&o||e.props.showTimeSelectOnly)){var s=["react-datepicker__navigation","react-datepicker__navigation--next"],i=["react-datepicker__navigation-icon","react-datepicker__navigation-icon--next"];e.props.showTimeSelect&&s.push("react-datepicker__navigation--next--with-time"),e.props.todayButton&&s.push("react-datepicker__navigation--next--with-today-button");var c=e.increaseMonth;(e.props.showMonthYearPicker||e.props.showQuarterYearPicker||e.props.showYearPicker)&&(c=e.increaseYear),o&&e.props.showDisabledMonthNavigation&&(s.push("react-datepicker__navigation--next--disabled"),c=void 0);var l=e.props.showMonthYearPicker||e.props.showQuarterYearPicker||e.props.showYearPicker,u=e.props,d=u.nextMonthButtonLabel,f=d===void 0?a.defaultProps.nextMonthButtonLabel:d,p=u.nextYearButtonLabel,h=p===void 0?a.defaultProps.nextYearButtonLabel:p,w=e.props,m=w.nextMonthAriaLabel,y=m===void 0?typeof f=="string"?f:"Next Month":m,_=w.nextYearAriaLabel,k=_===void 0?typeof h=="string"?h:"Next Year":_;return g.createElement("button",{type:"button",className:s.join(" "),onClick:c,onKeyDown:e.props.handleOnKeyDown,"aria-label":l?k:y},g.createElement("span",{className:i.join(" ")},l?h:f))}}},e.renderCurrentMonth=function(r){r===void 0&&(r=e.state.date);var o=["react-datepicker__current-month"];return e.props.showYearDropdown&&o.push("react-datepicker__current-month--hasYearDropdown"),e.props.showMonthDropdown&&o.push("react-datepicker__current-month--hasMonthDropdown"),e.props.showMonthYearDropdown&&o.push("react-datepicker__current-month--hasMonthYearDropdown"),g.createElement("h2",{className:o.join(" ")},G(r,e.props.dateFormat,e.props.locale))},e.renderYearDropdown=function(r){if(r===void 0&&(r=!1),!(!e.props.showYearDropdown||r))return g.createElement(Wc,Q({},a.defaultProps,e.props,{date:e.state.date,onChange:e.changeYear,year:I(e.state.date)}))},e.renderMonthDropdown=function(r){if(r===void 0&&(r=!1),!(!e.props.showMonthDropdown||r))return g.createElement(Oc,Q({},a.defaultProps,e.props,{month:ne(e.state.date),onChange:e.changeMonth}))},e.renderMonthYearDropdown=function(r){if(r===void 0&&(r=!1),!(!e.props.showMonthYearDropdown||r))return g.createElement(Yc,Q({},a.defaultProps,e.props,{date:e.state.date,onChange:e.changeMonthYear}))},e.handleTodayButtonClick=function(r){e.props.onSelect(tn(),r),e.props.setPreSelection&&e.props.setPreSelection(tn())},e.renderTodayButton=function(){if(!(!e.props.todayButton||e.props.showTimeSelectOnly))return g.createElement("div",{className:"react-datepicker__today-button",onClick:e.handleTodayButtonClick},e.props.todayButton)},e.renderDefaultHeader=function(r){var o=r.monthDate,s=r.i;return g.createElement("div",{className:"react-datepicker__header ".concat(e.props.showTimeSelect?"react-datepicker__header--has-time-select":"")},e.renderCurrentMonth(o),g.createElement("div",{className:"react-datepicker__header__dropdown react-datepicker__header__dropdown--".concat(e.props.dropdownMode),onFocus:e.handleDropdownFocus},e.renderMonthDropdown(s!==0),e.renderMonthYearDropdown(s!==0),e.renderYearDropdown(s!==0)),g.createElement("div",{className:"react-datepicker__day-names"},e.header(o)))},e.renderCustomHeader=function(r){var o,s,i=r.monthDate,c=r.i;if(e.props.showTimeSelect&&!e.state.monthContainer||e.props.showTimeSelectOnly)return null;var l=sn(e.state.date,e.props),u=cn(e.state.date,e.props),d=ln(e.state.date,e.props),f=un(e.state.date,e.props),p=!e.props.showMonthYearPicker&&!e.props.showQuarterYearPicker&&!e.props.showYearPicker;return g.createElement("div",{className:"react-datepicker__header react-datepicker__header--custom",onFocus:e.props.onDropdownFocus},(s=(o=e.props).renderCustomHeader)===null||s===void 0?void 0:s.call(o,Q(Q({},e.state),{customHeaderCount:c,monthDate:i,changeMonth:e.changeMonth,changeYear:e.changeYear,decreaseMonth:e.decreaseMonth,increaseMonth:e.increaseMonth,decreaseYear:e.decreaseYear,increaseYear:e.increaseYear,prevMonthButtonDisabled:l,nextMonthButtonDisabled:u,prevYearButtonDisabled:d,nextYearButtonDisabled:f})),p&&g.createElement("div",{className:"react-datepicker__day-names"},e.header(i)))},e.renderYearHeader=function(r){var o=r.monthDate,s=e.props,i=s.showYearPicker,c=s.yearItemNumber,l=c===void 0?a.defaultProps.yearItemNumber:c,u=Re(o,l),d=u.startPeriod,f=u.endPeriod;return g.createElement("div",{className:"react-datepicker__header react-datepicker-year-header"},i?"".concat(d," - ").concat(f):I(o))},e.renderHeader=function(r){var o=r.monthDate,s=r.i,i=s===void 0?0:s,c={monthDate:o,i};switch(!0){case e.props.renderCustomHeader!==void 0:return e.renderCustomHeader(c);case(e.props.showMonthYearPicker||e.props.showQuarterYearPicker||e.props.showYearPicker):return e.renderYearHeader(c);default:return e.renderDefaultHeader(c)}},e.renderMonths=function(){var r,o;if(!(e.props.showTimeSelectOnly||e.props.showYearPicker)){for(var s=[],i=(r=e.props.monthsShown)!==null&&r!==void 0?r:a.defaultProps.monthsShown,c=e.props.showPreviousMonths?i-1:0,l=e.props.showMonthYearPicker||e.props.showQuarterYearPicker?xe(e.state.date,c):je(e.state.date,c),u=(o=e.props.monthSelectedIn)!==null&&o!==void 0?o:c,d=0;d0;s.push(g.createElement("div",{key:h,ref:function(y){e.monthContainer=y??void 0},className:"react-datepicker__month-container"},e.renderHeader({monthDate:p,i:d}),g.createElement(Pc,Q({},a.defaultProps,e.props,{containerRef:e.containerRef,ariaLabelPrefix:e.props.monthAriaLabelPrefix,day:p,onDayClick:e.handleDayClick,handleOnKeyDown:e.props.handleOnDayKeyDown,handleOnMonthKeyDown:e.props.handleOnKeyDown,onDayMouseEnter:e.handleDayMouseEnter,onMouseLeave:e.handleMonthMouseLeave,orderInDisplay:d,selectingDate:e.state.selectingDate,monthShowsDuplicateDaysEnd:w,monthShowsDuplicateDaysStart:m}))))}return s}},e.renderYears=function(){if(!e.props.showTimeSelectOnly&&e.props.showYearPicker)return g.createElement("div",{className:"react-datepicker__year--container"},e.renderHeader({monthDate:e.state.date}),g.createElement(Fc,Q({},a.defaultProps,e.props,{selectingDate:e.state.selectingDate,date:e.state.date,onDayClick:e.handleDayClick,clearSelectingDate:e.clearSelectingDate,onYearMouseEnter:e.handleYearMouseEnter,onYearMouseLeave:e.handleYearMouseLeave})))},e.renderTimeSection=function(){if(e.props.showTimeSelect&&(e.state.monthContainer||e.props.showTimeSelectOnly))return g.createElement(Lc,Q({},a.defaultProps,e.props,{onChange:e.props.onTimeChange,format:e.props.timeFormat,intervals:e.props.timeIntervals,monthRef:e.state.monthContainer}))},e.renderInputTimeSection=function(){var r=e.props.selected?new Date(e.props.selected):void 0,o=r&&er(r)&&!!e.props.selected,s=o?"".concat(fn(r.getHours()),":").concat(fn(r.getMinutes())):"";if(e.props.showTimeInput)return g.createElement(_c,Q({},a.defaultProps,e.props,{date:r,timeString:s,onChange:e.props.onTimeChange}))},e.renderAriaLiveRegion=function(){var r,o=Re(e.state.date,(r=e.props.yearItemNumber)!==null&&r!==void 0?r:a.defaultProps.yearItemNumber),s=o.startPeriod,i=o.endPeriod,c;return e.props.showYearPicker?c="".concat(s," - ").concat(i):e.props.showMonthYearPicker||e.props.showQuarterYearPicker?c=I(e.state.date):c="".concat(Pr(ne(e.state.date),e.props.locale)," ").concat(I(e.state.date)),g.createElement("span",{role:"alert","aria-live":"polite",className:"react-datepicker__aria-live"},e.state.isRenderAriaLiveMessage&&c)},e.renderChildren=function(){if(e.props.children)return g.createElement("div",{className:"react-datepicker__children-container"},e.props.children)},e.containerRef=v.createRef(),e.state={date:e.getDateInView(),selectingDate:void 0,monthContainer:void 0,isRenderAriaLiveMessage:!1},e}return Object.defineProperty(a,"defaultProps",{get:function(){return{monthsShown:1,forceShowMonthNavigation:!1,timeCaption:"Time",previousYearButtonLabel:"Previous Year",nextYearButtonLabel:"Next Year",previousMonthButtonLabel:"Previous Month",nextMonthButtonLabel:"Next Month",yearItemNumber:vt}},enumerable:!1,configurable:!0}),a.prototype.componentDidMount=function(){var t=this;this.props.showTimeSelect&&(this.assignMonthContainer=function(){t.setState({monthContainer:t.monthContainer})}())},a.prototype.componentDidUpdate=function(t){var e=this;if(this.props.preSelection&&(!B(this.props.preSelection,t.preSelection)||this.props.monthSelectedIn!==t.monthSelectedIn)){var r=!ae(this.state.date,this.props.preSelection);this.setState({date:this.props.preSelection},function(){return r&&e.handleCustomMonthChange(e.state.date)})}else this.props.openToDate&&!B(this.props.openToDate,t.openToDate)&&this.setState({date:this.props.openToDate})},a.prototype.render=function(){var t=this.props.container||rc;return g.createElement(Jt,{onClickOutside:this.handleClickOutside,style:{display:"contents"},ignoreClass:this.props.outsideClickIgnoreClass},g.createElement("div",{style:{display:"contents"},ref:this.containerRef},g.createElement(t,{className:re("react-datepicker",this.props.className,{"react-datepicker--time-only":this.props.showTimeSelectOnly}),showTime:this.props.showTimeSelect||this.props.showTimeInput,showTimeSelectOnly:this.props.showTimeSelectOnly},this.renderAriaLiveRegion(),this.renderPreviousButton(),this.renderNextButton(),this.renderMonths(),this.renderYears(),this.renderTodayButton(),this.renderTimeSection(),this.renderInputTimeSection(),this.renderChildren())))},a}(v.Component),Qc=function(n){var a=n.icon,t=n.className,e=t===void 0?"":t,r=n.onClick,o="react-datepicker__calendar-icon";if(typeof a=="string")return g.createElement("i",{className:"".concat(o," ").concat(a," ").concat(e),"aria-hidden":"true",onClick:r});if(g.isValidElement(a)){var s=a;return g.cloneElement(s,{className:"".concat(s.props.className||""," ").concat(o," ").concat(e),onClick:function(i){typeof s.props.onClick=="function"&&s.props.onClick(i),typeof r=="function"&&r(i)}})}return g.createElement("svg",{className:"".concat(o," ").concat(e),xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 448 512",onClick:r},g.createElement("path",{d:"M96 32V64H48C21.5 64 0 85.5 0 112v48H448V112c0-26.5-21.5-48-48-48H352V32c0-17.7-14.3-32-32-32s-32 14.3-32 32V64H160V32c0-17.7-14.3-32-32-32S96 14.3 96 32zM448 192H0V464c0 26.5 21.5 48 48 48H400c26.5 0 48-21.5 48-48V192z"}))},ra=function(n){te(a,n);function a(t){var e=n.call(this,t)||this;return e.portalRoot=null,e.el=document.createElement("div"),e}return a.prototype.componentDidMount=function(){this.portalRoot=(this.props.portalHost||document).getElementById(this.props.portalId),this.portalRoot||(this.portalRoot=document.createElement("div"),this.portalRoot.setAttribute("id",this.props.portalId),(this.props.portalHost||document.body).appendChild(this.portalRoot)),this.portalRoot.appendChild(this.el)},a.prototype.componentWillUnmount=function(){this.portalRoot&&this.portalRoot.removeChild(this.el)},a.prototype.render=function(){return ca.createPortal(this.props.children,this.el)},a}(v.Component),$c="[tabindex], a, button, input, select, textarea",Vc=function(n){return(n instanceof HTMLAnchorElement||!n.disabled)&&n.tabIndex!==-1},na=function(n){te(a,n);function a(t){var e=n.call(this,t)||this;return e.getTabChildren=function(){var r;return Array.prototype.slice.call((r=e.tabLoopRef.current)===null||r===void 0?void 0:r.querySelectorAll($c),1,-1).filter(Vc)},e.handleFocusStart=function(){var r=e.getTabChildren();r&&r.length>1&&r[r.length-1].focus()},e.handleFocusEnd=function(){var r=e.getTabChildren();r&&r.length>1&&r[0].focus()},e.tabLoopRef=v.createRef(),e}return a.prototype.render=function(){var t;return((t=this.props.enableTabLoop)!==null&&t!==void 0?t:a.defaultProps.enableTabLoop)?g.createElement("div",{className:"react-datepicker__tab-loop",ref:this.tabLoopRef},g.createElement("div",{className:"react-datepicker__tab-loop__start",tabIndex:0,onFocus:this.handleFocusStart}),this.props.children,g.createElement("div",{className:"react-datepicker__tab-loop__end",tabIndex:0,onFocus:this.handleFocusEnd})):this.props.children},a.defaultProps={enableTabLoop:!0},a}(v.Component);function qc(n){var a=function(t){var e,r=typeof t.hidePopper=="boolean"?t.hidePopper:!0,o=v.useRef(null),s=tc(Q({open:!r,whileElementsMounted:Ni,placement:t.popperPlacement,middleware:ye([Bi({padding:15}),ji(10),Qi({element:o})],(e=t.popperModifiers)!==null&&e!==void 0?e:[],!0)},t.popperProps)),i=Q(Q({},t),{hidePopper:r,popperProps:Q(Q({},s),{arrowRef:o})});return g.createElement(n,Q({},i))};return a}var Kc=function(n){te(a,n);function a(){return n!==null&&n.apply(this,arguments)||this}return Object.defineProperty(a,"defaultProps",{get:function(){return{hidePopper:!0}},enumerable:!1,configurable:!0}),a.prototype.render=function(){var t=this.props,e=t.className,r=t.wrapperClassName,o=t.hidePopper,s=o===void 0?a.defaultProps.hidePopper:o,i=t.popperComponent,c=t.targetComponent,l=t.enableTabLoop,u=t.popperOnKeyDown,d=t.portalId,f=t.portalHost,p=t.popperProps,h=t.showArrow,w=void 0;if(!s){var m=re("react-datepicker-popper",e);w=g.createElement(na,{enableTabLoop:l},g.createElement("div",{ref:p.refs.setFloating,style:p.floatingStyles,className:m,"data-placement":p.placement,onKeyDown:u},i,h&&g.createElement(Ui,{ref:p.arrowRef,context:p.context,fill:"currentColor",strokeWidth:1,height:8,width:16,style:{transform:"translateY(-1px)"},className:"react-datepicker__triangle"})))}this.props.popperContainer&&(w=v.createElement(this.props.popperContainer,{},w)),d&&!s&&(w=g.createElement(ra,{portalId:d,portalHost:f},w));var y=re("react-datepicker-wrapper",r);return g.createElement(g.Fragment,null,g.createElement("div",{ref:p.refs.setReference,className:y},c),w)},a}(v.Component),Uc=qc(Kc),wn="react-datepicker-ignore-onclickoutside";function Gc(n,a){return n&&a?ne(n)!==ne(a)||I(n)!==I(a):n!==a}var cr="Date input not valid.",zc=function(n){te(a,n);function a(t){var e=n.call(this,t)||this;return e.calendar=null,e.input=null,e.getPreSelection=function(){return e.props.openToDate?e.props.openToDate:e.props.selectsEnd&&e.props.startDate?e.props.startDate:e.props.selectsStart&&e.props.endDate?e.props.endDate:V()},e.modifyHolidays=function(){var r;return(r=e.props.holidays)===null||r===void 0?void 0:r.reduce(function(o,s){var i=new Date(s.date);return er(i)?ye(ye([],o,!0),[Q(Q({},s),{date:i})],!1):o},[])},e.calcInitialState=function(){var r,o=e.getPreSelection(),s=Jn(e.props),i=ea(e.props),c=s&&$e(o,Tt(s))?s:i&&Le(o,rn(i))?i:o;return{open:e.props.startOpen||!1,preventFocus:!1,inputValue:null,preSelection:(r=e.props.selectsRange?e.props.startDate:e.props.selected)!==null&&r!==void 0?r:c,highlightDates:dn(e.props.highlightDates),focused:!1,shouldFocusDayInline:!1,isRenderAriaLiveMessage:!1,wasHidden:!1}},e.resetHiddenStatus=function(){e.setState(Q(Q({},e.state),{wasHidden:!1}))},e.setHiddenStatus=function(){e.setState(Q(Q({},e.state),{wasHidden:!0}))},e.setHiddenStateOnVisibilityHidden=function(){document.visibilityState==="hidden"&&e.setHiddenStatus()},e.clearPreventFocusTimeout=function(){e.preventFocusTimeout&&clearTimeout(e.preventFocusTimeout)},e.setFocus=function(){var r,o;(o=(r=e.input)===null||r===void 0?void 0:r.focus)===null||o===void 0||o.call(r,{preventScroll:!0})},e.setBlur=function(){var r,o;(o=(r=e.input)===null||r===void 0?void 0:r.blur)===null||o===void 0||o.call(r),e.cancelFocusInput()},e.deferBlur=function(){requestAnimationFrame(function(){e.setBlur()})},e.setOpen=function(r,o){o===void 0&&(o=!1),e.setState({open:r,preSelection:r&&e.state.open?e.state.preSelection:e.calcInitialState().preSelection,lastPreSelectChange:lr},function(){r||e.setState(function(s){return{focused:o?s.focused:!1}},function(){!o&&e.deferBlur(),e.setState({inputValue:null})})})},e.inputOk=function(){return _e(e.state.preSelection)},e.isCalendarOpen=function(){return e.props.open===void 0?e.state.open&&!e.props.disabled&&!e.props.readOnly:e.props.open},e.handleFocus=function(r){var o,s,i=e.state.wasHidden,c=i?e.state.open:!0;i&&e.resetHiddenStatus(),e.state.preventFocus||((s=(o=e.props).onFocus)===null||s===void 0||s.call(o,r),c&&!e.props.preventOpenOnFocus&&!e.props.readOnly&&e.setOpen(!0)),e.setState({focused:!0})},e.sendFocusBackToInput=function(){e.preventFocusTimeout&&e.clearPreventFocusTimeout(),e.setState({preventFocus:!0},function(){e.preventFocusTimeout=setTimeout(function(){e.setFocus(),e.setState({preventFocus:!1})})})},e.cancelFocusInput=function(){clearTimeout(e.inputFocusTimeout),e.inputFocusTimeout=void 0},e.deferFocusInput=function(){e.cancelFocusInput(),e.inputFocusTimeout=setTimeout(function(){return e.setFocus()},1)},e.handleDropdownFocus=function(){e.cancelFocusInput()},e.handleBlur=function(r){var o,s;(!e.state.open||e.props.withPortal||e.props.showTimeInput)&&((s=(o=e.props).onBlur)===null||s===void 0||s.call(o,r)),e.state.open&&e.props.open===!1&&e.setOpen(!1),e.setState({focused:!1})},e.handleCalendarClickOutside=function(r){var o,s;e.props.inline||e.setOpen(!1),(s=(o=e.props).onClickOutside)===null||s===void 0||s.call(o,r),e.props.withPortal&&r.preventDefault()},e.handleChange=function(){for(var r,o,s,i,c,l=[],u=0;u=H){fe=Y;break}f&&fep&&(F=M.ArrowLeft,fe=le(p,e.props)?C(F,fe):p),le(fe,e.props)?((F===M.PageUp||F===M.Home)&&(F=M.ArrowRight),(F===M.PageDown||F===M.End)&&(F=M.ArrowLeft),fe=C(F,fe)):ee=!0,de++}return fe};if(O===M.Enter){r.preventDefault(),e.handleSelect(T,r),!m&&e.setPreSelection(T);return}else if(O===M.Escape){r.preventDefault(),e.setOpen(!1),e.inputOk()||(c=(i=e.props).onInputError)===null||c===void 0||c.call(i,{code:1,msg:cr});return}var b=null;switch(O){case M.ArrowLeft:case M.ArrowRight:case M.ArrowUp:case M.ArrowDown:case M.PageUp:case M.PageDown:case M.Home:case M.End:b=D(O,T);break}if(!b){(u=(l=e.props).onInputError)===null||u===void 0||u.call(l,{code:1,msg:cr});return}if(r.preventDefault(),e.setState({lastPreSelectChange:lr}),k&&e.setSelected(b),e.setPreSelection(b),E){var N=ne(T),R=ne(b),A=I(T),W=I(b);N!==R||A!==W?e.setState({shouldFocusDayInline:!0}):e.setState({shouldFocusDayInline:!1})}}},e.onPopperKeyDown=function(r){var o=r.key;o===M.Escape&&(r.preventDefault(),e.sendFocusBackToInput(),e.setOpen(!1))},e.onClearClick=function(r){r&&r.preventDefault&&r.preventDefault(),e.sendFocusBackToInput();var o=e.props,s=o.selectsRange,i=o.onChange;s?i==null||i([null,null],r):i==null||i(null,r),e.setState({inputValue:null})},e.clear=function(){e.onClearClick()},e.onScroll=function(r){typeof e.props.closeOnScroll=="boolean"&&e.props.closeOnScroll?(r.target===document||r.target===document.documentElement||r.target===document.body)&&e.setOpen(!1):typeof e.props.closeOnScroll=="function"&&e.props.closeOnScroll(r)&&e.setOpen(!1)},e.renderCalendar=function(){var r,o;return!e.props.inline&&!e.isCalendarOpen()?null:g.createElement(Bc,Q({showMonthYearDropdown:void 0,ref:function(s){e.calendar=s}},e.props,e.state,{setOpen:e.setOpen,dateFormat:(r=e.props.dateFormatCalendar)!==null&&r!==void 0?r:a.defaultProps.dateFormatCalendar,onSelect:e.handleSelect,onClickOutside:e.handleCalendarClickOutside,holidays:Dc(e.modifyHolidays()),outsideClickIgnoreClass:wn,onDropdownFocus:e.handleDropdownFocus,onTimeChange:e.handleTimeChange,className:e.props.calendarClassName,container:e.props.calendarContainer,handleOnKeyDown:e.props.onKeyDown,handleOnDayKeyDown:e.onDayKeyDown,setPreSelection:e.setPreSelection,dropdownMode:(o=e.props.dropdownMode)!==null&&o!==void 0?o:a.defaultProps.dropdownMode}),e.props.children)},e.renderAriaLiveRegion=function(){var r=e.props,o=r.dateFormat,s=o===void 0?a.defaultProps.dateFormat:o,i=r.locale,c=e.props.showTimeInput||e.props.showTimeSelect,l=c?"PPPPp":"PPPP",u;return e.props.selectsRange?u="Selected start date: ".concat(ie(e.props.startDate,{dateFormat:l,locale:i}),". ").concat(e.props.endDate?"End date: "+ie(e.props.endDate,{dateFormat:l,locale:i}):""):e.props.showTimeSelectOnly?u="Selected time: ".concat(ie(e.props.selected,{dateFormat:s,locale:i})):e.props.showYearPicker?u="Selected year: ".concat(ie(e.props.selected,{dateFormat:"yyyy",locale:i})):e.props.showMonthYearPicker?u="Selected month: ".concat(ie(e.props.selected,{dateFormat:"MMMM yyyy",locale:i})):e.props.showQuarterYearPicker?u="Selected quarter: ".concat(ie(e.props.selected,{dateFormat:"yyyy, QQQ",locale:i})):u="Selected date: ".concat(ie(e.props.selected,{dateFormat:l,locale:i})),g.createElement("span",{role:"alert","aria-live":"polite",className:"react-datepicker__aria-live"},u)},e.renderDateInput=function(){var r,o,s,i=re(e.props.className,(r={},r[wn]=e.state.open,r)),c=e.props.customInput||g.createElement("input",{type:"text"}),l=e.props.customInputRef||"ref",u=e.props,d=u.dateFormat,f=d===void 0?a.defaultProps.dateFormat:d,p=u.locale,h=typeof e.props.value=="string"?e.props.value:typeof e.state.inputValue=="string"?e.state.inputValue:e.props.selectsRange?ac(e.props.startDate,e.props.endDate,{dateFormat:f,locale:p}):e.props.selectsMultiple?oc((s=e.props.selectedDates)!==null&&s!==void 0?s:[],{dateFormat:f,locale:p}):ie(e.props.selected,{dateFormat:f,locale:p});return v.cloneElement(c,(o={},o[l]=function(w){e.input=w},o.value=h,o.onBlur=e.handleBlur,o.onChange=e.handleChange,o.onClick=e.onInputClick,o.onFocus=e.handleFocus,o.onKeyDown=e.onInputKeyDown,o.id=e.props.id,o.name=e.props.name,o.form=e.props.form,o.autoFocus=e.props.autoFocus,o.placeholder=e.props.placeholderText,o.disabled=e.props.disabled,o.autoComplete=e.props.autoComplete,o.className=re(c.props.className,i),o.title=e.props.title,o.readOnly=e.props.readOnly,o.required=e.props.required,o.tabIndex=e.props.tabIndex,o["aria-describedby"]=e.props.ariaDescribedBy,o["aria-invalid"]=e.props.ariaInvalid,o["aria-labelledby"]=e.props.ariaLabelledBy,o["aria-required"]=e.props.ariaRequired,o))},e.renderClearButton=function(){var r=e.props,o=r.isClearable,s=r.disabled,i=r.selected,c=r.startDate,l=r.endDate,u=r.clearButtonTitle,d=r.clearButtonClassName,f=d===void 0?"":d,p=r.ariaLabelClose,h=p===void 0?"Close":p,w=r.selectedDates,m=r.readOnly;return o&&!m&&(i!=null||c!=null||l!=null||w!=null&&w.length)?g.createElement("button",{type:"button",className:re("react-datepicker__close-icon",f,{"react-datepicker__close-icon--disabled":s}),disabled:s,"aria-label":h,onClick:e.onClearClick,title:u,tabIndex:-1}):null},e.state=e.calcInitialState(),e.preventFocusTimeout=void 0,e}return Object.defineProperty(a,"defaultProps",{get:function(){return{allowSameDay:!1,dateFormat:"MM/dd/yyyy",dateFormatCalendar:"LLLL yyyy",disabled:!1,disabledKeyboardNavigation:!1,dropdownMode:"scroll",preventOpenOnFocus:!1,monthsShown:1,readOnly:!1,withPortal:!1,selectsDisabledDaysInRange:!1,shouldCloseOnSelect:!0,showTimeSelect:!1,showTimeInput:!1,showPreviousMonths:!1,showMonthYearPicker:!1,showFullMonthYearPicker:!1,showTwoColumnMonthYearPicker:!1,showFourColumnMonthYearPicker:!1,showYearPicker:!1,showQuarterYearPicker:!1,showWeekPicker:!1,strictParsing:!1,swapRange:!1,timeIntervals:30,timeCaption:"Time",previousMonthAriaLabel:"Previous Month",previousMonthButtonLabel:"Previous Month",nextMonthAriaLabel:"Next Month",nextMonthButtonLabel:"Next Month",previousYearAriaLabel:"Previous Year",previousYearButtonLabel:"Previous Year",nextYearAriaLabel:"Next Year",nextYearButtonLabel:"Next Year",timeInputLabel:"Time",enableTabLoop:!0,yearItemNumber:vt,focusSelectedMonth:!1,showPopperArrow:!0,excludeScrollbar:!0,customTimeInput:null,calendarStartDay:void 0,toggleCalendarOnIconClick:!1,usePointerEvent:!1}},enumerable:!1,configurable:!0}),a.prototype.componentDidMount=function(){window.addEventListener("scroll",this.onScroll,!0),document.addEventListener("visibilitychange",this.setHiddenStateOnVisibilityHidden)},a.prototype.componentDidUpdate=function(t,e){var r,o,s,i;t.inline&&Gc(t.selected,this.props.selected)&&this.setPreSelection(this.props.selected),this.state.monthSelectedIn!==void 0&&t.monthsShown!==this.props.monthsShown&&this.setState({monthSelectedIn:0}),t.highlightDates!==this.props.highlightDates&&this.setState({highlightDates:dn(this.props.highlightDates)}),!e.focused&&!We(t.selected,this.props.selected)&&this.setState({inputValue:null}),e.open!==this.state.open&&(e.open===!1&&this.state.open===!0&&((o=(r=this.props).onCalendarOpen)===null||o===void 0||o.call(r)),e.open===!0&&this.state.open===!1&&((i=(s=this.props).onCalendarClose)===null||i===void 0||i.call(s)))},a.prototype.componentWillUnmount=function(){this.clearPreventFocusTimeout(),window.removeEventListener("scroll",this.onScroll,!0),document.removeEventListener("visibilitychange",this.setHiddenStateOnVisibilityHidden)},a.prototype.renderInputContainer=function(){var t=this.props,e=t.showIcon,r=t.icon,o=t.calendarIconClassname,s=t.calendarIconClassName,i=t.toggleCalendarOnIconClick,c=this.state.open;return o&&console.warn("calendarIconClassname props is deprecated. should use calendarIconClassName props."),g.createElement("div",{className:"react-datepicker__input-container".concat(e?" react-datepicker__view-calendar-icon":"")},e&&g.createElement(Qc,Q({icon:r,className:re(s,!s&&o,c&&"react-datepicker-ignore-onclickoutside")},i?{onClick:this.toggleCalendar}:null)),this.state.isRenderAriaLiveMessage&&this.renderAriaLiveRegion(),this.renderDateInput(),this.renderClearButton())},a.prototype.render=function(){var t=this.renderCalendar();if(this.props.inline)return t;if(this.props.withPortal){var e=this.state.open?g.createElement(na,{enableTabLoop:this.props.enableTabLoop},g.createElement("div",{className:"react-datepicker__portal",tabIndex:-1,onKeyDown:this.onPortalKeyDown},t)):null;return this.state.open&&this.props.portalId&&(e=g.createElement(ra,Q({portalId:this.props.portalId},this.props),e)),g.createElement("div",null,this.renderInputContainer(),e)}return g.createElement(Uc,Q({},this.props,{className:this.props.popperClassName,hidePopper:!this.isCalendarOpen(),targetComponent:this.renderInputContainer(),popperComponent:t,popperOnKeyDown:this.onPopperKeyDown,showArrow:this.props.showPopperArrow}))},a}(v.Component),Xc="input",lr="navigate";const Zc=j.div` + background: white; + border-radius: 0.75rem; + padding: 1.5rem; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; +`,Jc=j.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; +`,el=j.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`,tl=j.div` + display: flex; + gap: 1rem; + align-items: center; + flex-wrap: wrap; +`,rl=j.div` + display: flex; + flex-direction: column; + min-width: 150px; +`,nl=j.label` + font-size: 0.875rem; + font-weight: 500; + color: #374151; + margin-bottom: 0.5rem; +`,al=j(zc)` + width: 100%; + padding: 0.5rem 0.75rem; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + color: #111827; + background-color: white; + + &:focus { + outline: none; + border-color: #10b981; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } + + &:hover { + border-color: #9ca3af; + } +`,ol=j.button` + padding: 0.5rem 1rem; + background-color: #f3f4f6; + color: #374151; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + font-weight: 500; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background-color: #e5e7eb; + border-color: #9ca3af; + } + + &:focus { + outline: none; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } +`,sl=j.div` + display: flex; + align-items: center; + color: #6b7280; + font-size: 0.875rem; +`,yn=({label:n,selected:a,onChange:t,selectsStart:e,selectsEnd:r,startDate:o,endDate:s,minDate:i,maxDate:c,placeholderText:l})=>x.jsxs(rl,{children:[x.jsx(nl,{children:n}),x.jsx(al,{selected:a,onChange:t,selectsStart:e,selectsEnd:r,startDate:o,endDate:s,minDate:i,maxDate:c,dateFormat:"MMM dd, yyyy",placeholderText:l})]}),il=({onDateChange:n,initialStartDate:a,initialEndDate:t})=>{const{startDate:e,endDate:r,handleStartDateChange:o,handleEndDateChange:s,handleReset:i}=da(n,a,t);return x.jsxs(Zc,{children:[x.jsxs(Jc,{children:[x.jsx(el,{children:"Date Range"}),x.jsx(ol,{onClick:i,children:"Reset"})]}),x.jsxs(tl,{children:[x.jsx(yn,{label:"Start Date",selected:e,onChange:o,selectsStart:!0,startDate:e,endDate:r,maxDate:r,placeholderText:"Select start date"}),x.jsx(sl,{children:x.jsx("span",{children:"to"})}),x.jsx(yn,{label:"End Date",selected:r,onChange:s,selectsEnd:!0,startDate:e,endDate:r,minDate:e,maxDate:new Date,placeholderText:"Select end date"})]})]})},cl=j.div` + background: white; + border-radius: 0.75rem; + padding: 1.5rem; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; +`,ll=j.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; +`,ul=j.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`,dl=j.input` + width: 100%; + padding: 0.5rem 0.75rem; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + color: #111827; + background-color: white; + margin-bottom: 1rem; + + &:focus { + outline: none; + border-color: #10b981; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } + + &:hover { + border-color: #9ca3af; + } +`,fl=j.div` + max-height: 200px; + overflow-y: auto; + border: 1px solid #e5e7eb; + border-radius: 0.5rem; + background-color: white; +`,pl=j.label` + display: flex; + align-items: center; + padding: 0.75rem; + cursor: pointer; + border-bottom: 1px solid #f3f4f6; + transition: background-color 0.2s; + + &:last-child { + border-bottom: none; + } + + &:hover { + background-color: #f9fafb; + } + + &:focus-within { + background-color: #f0fdf4; + } +`,hl=j.input` + margin-right: 0.75rem; + width: 1rem; + height: 1rem; + accent-color: #10b981; +`,ml=j.div` + flex: 1; +`,gl=j.div` + font-weight: 500; + color: #111827; + font-size: 0.875rem; +`,vl=j.div` + font-size: 0.75rem; + color: #6b7280; + margin-top: 0.25rem; +`,wl=j.div` + font-size: 0.75rem; + color: #6b7280; + margin-top: 0.5rem; +`,yl=j.button` + padding: 0.5rem 1rem; + background-color: #f3f4f6; + color: #374151; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + font-weight: 500; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background-color: #e5e7eb; + border-color: #9ca3af; + } + + &:focus { + outline: none; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } +`,Dl=({selectedForests:n=[],onChange:a})=>{const[t,e]=v.useState(""),[r,o]=v.useState([]),[s,i]=v.useState([]),[c,l]=v.useState(!0),[u,d]=v.useState(null);v.useEffect(()=>{(async()=>{try{l(!0),d(null);const _=(await ua.getAll()).data.forests.map(k=>({id:k.id,name:k.name,region:k.region,treeCount:k.treeCount||0,area:`${Math.round(k.area)} ha`,areaNumeric:k.area,isActive:k.isActive})).filter(k=>k.isActive);o(_)}catch(y){console.error("Error fetching forests:",y),d("Failed to load forests"),o([])}finally{l(!1)}})()},[]),v.useEffect(()=>{const m=r.filter(y=>y.name.toLowerCase().includes(t.toLowerCase())||y.region.toLowerCase().includes(t.toLowerCase()));i(m)},[t,r]);const f=m=>{a&&(n.includes(m)?a(n.filter(y=>y!==m)):a([...n,m]))},p=()=>{a&&a(r.map(m=>m.id))},h=()=>{a&&a([])},w=()=>{a&&a([]),e("")};return x.jsxs(cl,{children:[x.jsxs(ll,{children:[x.jsx(ul,{children:"Forest Selection"}),x.jsx(yl,{onClick:w,children:"Reset"})]}),c?x.jsx("div",{className:"p-4 text-center",children:x.jsx("div",{className:"text-sm text-gray-500",children:"Loading forests..."})}):u?x.jsxs("div",{className:"p-4 text-center",children:[x.jsx("div",{className:"text-sm text-red-600",children:u}),x.jsx("button",{onClick:()=>window.location.reload(),className:"mt-2 text-xs text-blue-600 hover:text-blue-800 underline",children:"Retry"})]}):x.jsxs(x.Fragment,{children:[x.jsx(dl,{type:"text",placeholder:"Search forests by name or region...",value:t,onChange:m=>e(m.target.value)}),x.jsxs("div",{className:"flex gap-2 mb-3",children:[x.jsx("button",{onClick:p,className:"px-3 py-1 text-xs bg-green-100 text-green-700 rounded border border-green-200 hover:bg-green-200 transition-colors",children:"Select All"}),x.jsx("button",{onClick:h,className:"px-3 py-1 text-xs bg-gray-100 text-gray-700 rounded border border-gray-200 hover:bg-gray-200 transition-colors",children:"Select None"})]}),x.jsxs(fl,{children:[s.map(m=>x.jsxs(pl,{children:[x.jsx(hl,{type:"checkbox",checked:n.includes(m.id),onChange:()=>f(m.id)}),x.jsxs(ml,{children:[x.jsx(gl,{children:m.name}),x.jsxs(vl,{children:[m.region," • ",m.treeCount," trees • ",m.area]})]})]},m.id)),s.length===0&&!c&&x.jsx("div",{className:"p-4 text-center text-gray-500 text-sm",children:r.length===0?"No forests available.":"No forests found matching your search."})]})]}),x.jsxs(wl,{children:[n.length," of ",r.length," forests selected"]})]})},bl=({errors:n})=>!n||Object.keys(n).length===0?null:x.jsxs("div",{className:"mb-4 p-3 bg-red-50 border border-red-200 rounded-lg",children:[x.jsxs("div",{className:"flex items-center mb-2",children:[x.jsx("svg",{className:"w-5 h-5 text-red-500 mr-2",fill:"currentColor",viewBox:"0 0 20 20",children:x.jsx("path",{fillRule:"evenodd",d:"M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z",clipRule:"evenodd"})}),x.jsx("span",{className:"font-medium text-red-800",children:"Filter Validation Errors"})]}),x.jsx("ul",{className:"text-sm text-red-700 space-y-1",children:Object.entries(n).map(([a,t])=>x.jsxs("li",{children:["• ",t]},a))})]}),ht={MAX_DATE_RANGE_YEARS:5,MAX_SELECTED_FORESTS:10,MILLISECONDS_PER_YEAR:1e3*60*60*24*365},Rt={DATE_ORDER:"Start date must be before or equal to end date",DATE_FUTURE:"End date cannot be in the future",DATE_RANGE_TOO_LARGE:`Date range cannot exceed ${ht.MAX_DATE_RANGE_YEARS} years`,TOO_MANY_FORESTS:`Cannot select more than ${ht.MAX_SELECTED_FORESTS} forests at once`},kl=n=>{if(!n||!n.startDate||!n.endDate)return null;const{startDate:a,endDate:t}=n;return a>t?Rt.DATE_ORDER:t>new Date?Rt.DATE_FUTURE:(t-a)/ht.MILLISECONDS_PER_YEAR>ht.MAX_DATE_RANGE_YEARS?Rt.DATE_RANGE_TOO_LARGE:null},xl=n=>!n||!Array.isArray(n)?null:n.length>ht.MAX_SELECTED_FORESTS?Rt.TOO_MANY_FORESTS:null,_l=n=>{const a={},t=kl(n.dateRange);t&&(a.dateRange=t);const e=xl(n.selectedForests);return e&&(a.selectedForests=e),a},Dn=()=>({dateRange:{startDate:null,endDate:null},selectedForests:[]}),Ml=j.div` + background: #f9fafb; + border-radius: 0.75rem; + padding: ${n=>n.$isCollapsed?"1rem":"1.5rem"}; + margin-bottom: 2rem; + transition: all 0.3s ease; + border: 1px solid #e5e7eb; +`,Cl=j.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: ${n=>n.$isCollapsed?"0":"1.5rem"}; +`,Sl=j.h2` + font-size: 1.5rem; + font-weight: 700; + color: #111827; + margin: 0; +`,El=j.p` + color: #4b5563; + margin: 0.5rem 0 0 0; + font-size: 0.875rem; +`,Pl=j.div` + display: grid; + grid-template-columns: 1fr; + gap: 1.5rem; + + @media (min-width: 768px) { + grid-template-columns: 1fr 1fr; + } + + @media (min-width: 1024px) { + grid-template-columns: 1fr 1fr; + } +`,Tl=j.div` + display: flex; + flex-wrap: wrap; + gap: 0.5rem; + margin-top: 1rem; + padding-top: 1rem; + border-top: 1px solid #e5e7eb; +`,Ol=j.span` + display: inline-flex; + align-items: center; + padding: 0.25rem 0.75rem; + background-color: #10b981; + color: white; + border-radius: 9999px; + font-size: 0.75rem; + font-weight: 500; +`,Rl=j.button` + padding: 0.5rem 1rem; + background-color: #ef4444; + color: white; + border: none; + border-radius: 0.5rem; + font-size: 0.875rem; + font-weight: 500; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background-color: #dc2626; + } + + &:focus-visible { + outline: none; + box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.45); + } +`,Nl=j.button` + display: flex; + align-items: center; + gap: 0.5rem; + padding: 0.5rem 1rem; + background-color: #10b981; + color: white; + border: none; + border-radius: 0.5rem; + font-size: 0.875rem; + font-weight: 500; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background-color: #059669; + } + + &:focus-visible { + outline: none; + box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.45); + } + + svg { + width: 1rem; + height: 1rem; + } +`,Yl=j.div` + display: flex; + align-items: center; + gap: 1rem; +`,Ll=j.div` + display: flex; + align-items: center; + gap: 0.5rem; +`,Fl=j.span` + color: #4b5563; + font-size: 0.875rem; + margin-left: 0.5rem; +`,Il=j.div` + display: ${n=>n.$isCollapsed?"none":"block"}; +`,Al=j.div` + display: flex; + flex-wrap: wrap; + gap: 0.5rem; + margin-bottom: 1rem; +`,Oe=j.button` + padding: 0.375rem 0.75rem; + background-color: ${n=>n.$isActive?"#10b981":"#ffffff"}; + color: ${n=>n.$isActive?"white":"#374151"}; + border: 1px solid ${n=>n.$isActive?"#10b981":"#d1d5db"}; + border-radius: 0.375rem; + font-size: 0.875rem; + font-weight: 500; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background-color: ${n=>n.$isActive?"#059669":"#f3f4f6"}; + border-color: ${n=>n.$isActive?"#059669":"#9ca3af"}; + } + + &:focus { + outline: none; + box-shadow: 0 0 0 2px rgba(16, 185, 129, 0.2); + } +`,Bl=({onFiltersChange:n,initialFilters:a={}})=>{const[t,e]=la(),r=v.useRef(!1),o=v.useRef(n),s=v.useRef(!1),[i,c]=v.useState(!0),[l,u]=v.useState(null);v.useEffect(()=>{o.current=n},[n]);const d=v.useCallback(()=>{const D=t.get("startDate"),b=t.get("endDate"),N=t.get("forests"),R={};if(D&&b)try{const A=new Date(D),W=new Date(b);!isNaN(A.getTime())&&!isNaN(W.getTime())&&A<=W&&(R.dateRange={startDate:A,endDate:W})}catch(A){console.warn("Invalid date parameters in URL:",A)}if(N)try{const A=N.split(",").map(W=>parseInt(W,10)).filter(W=>!isNaN(W));A.length>0&&(R.selectedForests=A)}catch(A){console.warn("Invalid forest parameters in URL:",A)}return R},[t]),[f,p]=v.useState(()=>{const D=d();return{...Dn(),...a,...D}}),[h,w]=v.useState({}),m=v.useCallback(D=>{const b=_l(D);return w(b),Object.keys(b).length===0},[]),y=v.useCallback(D=>{var N,R;const b=new URLSearchParams;(N=D.dateRange)!=null&&N.startDate&&((R=D.dateRange)!=null&&R.endDate)&&(b.set("startDate",Or(D.dateRange.startDate)),b.set("endDate",Or(D.dateRange.endDate))),D.selectedForests&&D.selectedForests.length>0&&b.set("forests",D.selectedForests.join(",")),e(b,{replace:!0})},[e]);v.useEffect(()=>{!r.current&&Object.keys(a).length>0&&(p(D=>({...D,...a})),r.current=!0)},[]);const[_,k]=v.useState([]);v.useEffect(()=>{const D=[];if(f.dateRange.startDate&&f.dateRange.endDate){const b=f.dateRange.startDate.toLocaleDateString(),N=f.dateRange.endDate.toLocaleDateString();D.push(`Date: ${b} - ${N}`)}f.selectedForests.length>0&&D.push(`${f.selectedForests.length} forests selected`),k(D)},[f]),v.useEffect(()=>{if(!s.current){s.current=!0;return}const D=setTimeout(()=>{m(f)&&(y(f),o.current&&o.current(f))},1e3);return()=>clearTimeout(D)},[f,m,y]);const E=v.useCallback(D=>{p(b=>({...b,dateRange:D}))},[]),O=v.useCallback(D=>{p(b=>({...b,selectedForests:D}))},[]),L=v.useCallback(()=>{const D=Dn();p(D),w({}),u(null),e({},{replace:!0})},[e]),T=v.useCallback(D=>{const b=new Date;let N,R;switch(D){case"mtd":N=new Date(b.getFullYear(),b.getMonth(),1),R=b;break;case"ytd":N=new Date(b.getFullYear(),0,1),R=b;break;case"30days":N=new Date(b.getTime()-30*24*60*60*1e3),R=b;break;case"90days":N=new Date(b.getTime()-90*24*60*60*1e3),R=b;break;case"1year":N=new Date(b.getFullYear()-1,b.getMonth(),b.getDate()),R=b;break;case"3years":N=new Date(b.getFullYear()-3,b.getMonth(),b.getDate()),R=b;break;case"5years":N=new Date(b.getFullYear()-5,b.getMonth(),b.getDate()),R=b;break;case"all":N=null,R=null;break;default:return}u(D),E({startDate:N,endDate:R})},[E]),C=v.useCallback(()=>{c(D=>!D)},[]);return v.useEffect(()=>{if(l&&f.dateRange.startDate&&f.dateRange.endDate){const{startDate:D,endDate:b}=f.dateRange;u(null)}},[f.dateRange,l]),x.jsxs(Ml,{$isCollapsed:i,children:[x.jsxs(Cl,{$isCollapsed:i,children:[x.jsx(Yl,{children:x.jsxs("div",{children:[x.jsxs(Sl,{children:["Global Filters",i&&_.length>0&&x.jsxs(Fl,{children:["(",_.length," active)"]})]}),!i&&x.jsx(El,{children:"Filter your data by date range and forest selection"})]})}),x.jsxs(Ll,{children:[x.jsx(Rl,{onClick:L,style:{visibility:!i&&_.length>0?"visible":"hidden",pointerEvents:!i&&_.length>0?"auto":"none"},children:"Clear All"}),x.jsx(Nl,{onClick:C,"aria-expanded":!i,"aria-controls":"global-filters-content",children:i?x.jsxs(x.Fragment,{children:[x.jsx("svg",{xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24",strokeWidth:1.5,stroke:"currentColor",children:x.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",d:"M19.5 8.25l-7.5 7.5-7.5-7.5"})}),"Show Filters"]}):x.jsxs(x.Fragment,{children:[x.jsx("svg",{xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24",strokeWidth:1.5,stroke:"currentColor",children:x.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",d:"M4.5 15.75l7.5-7.5 7.5 7.5"})}),"Hide Filters"]})})]})]}),x.jsxs(Il,{id:"global-filters-content",$isCollapsed:i,children:[x.jsx(bl,{errors:h}),x.jsxs(Al,{children:[x.jsx(Oe,{$isActive:l==="mtd",onClick:()=>T("mtd"),children:"Month to Date"}),x.jsx(Oe,{$isActive:l==="ytd",onClick:()=>T("ytd"),children:"Year to Date"}),x.jsx(Oe,{$isActive:l==="30days",onClick:()=>T("30days"),children:"Last 30 Days"}),x.jsx(Oe,{$isActive:l==="90days",onClick:()=>T("90days"),children:"Last 90 Days"}),x.jsx(Oe,{$isActive:l==="1year",onClick:()=>T("1year"),children:"Last 1 Year"}),x.jsx(Oe,{$isActive:l==="3years",onClick:()=>T("3years"),children:"Last 3 Years"}),x.jsx(Oe,{$isActive:l==="5years",onClick:()=>T("5years"),children:"Last 5 Years"}),x.jsx(Oe,{$isActive:l==="all",onClick:()=>T("all"),children:"All Time"})]}),x.jsxs(Pl,{children:[x.jsx(il,{onDateChange:E,initialStartDate:f.dateRange.startDate,initialEndDate:f.dateRange.endDate}),x.jsx(Dl,{selectedForests:f.selectedForests,onChange:O})]}),_.length>0&&x.jsx(Tl,{children:_.map((D,b)=>x.jsx(Ol,{children:D},b))})]})]})};export{Bl as G,jl as t}; diff --git a/frontend/dist/assets/GlobalFilters-DdjxYr1F.css b/frontend/dist/assets/GlobalFilters-DdjxYr1F.css new file mode 100644 index 0000000000..87be4be039 --- /dev/null +++ b/frontend/dist/assets/GlobalFilters-DdjxYr1F.css @@ -0,0 +1 @@ +@charset "UTF-8";.react-datepicker__navigation-icon:before,.react-datepicker__year-read-view--down-arrow,.react-datepicker__month-read-view--down-arrow,.react-datepicker__month-year-read-view--down-arrow{border-color:#ccc;border-style:solid;border-width:3px 3px 0 0;content:"";display:block;height:9px;position:absolute;top:6px;width:9px}.react-datepicker-wrapper{display:inline-block;padding:0;border:0}.react-datepicker{font-family:Helvetica Neue,helvetica,arial,sans-serif;font-size:.8rem;background-color:#fff;color:#000;border:1px solid #aeaeae;border-radius:.3rem;display:inline-block;position:relative;line-height:initial}.react-datepicker--time-only .react-datepicker__time-container{border-left:0}.react-datepicker--time-only .react-datepicker__time,.react-datepicker--time-only .react-datepicker__time-box{border-bottom-left-radius:.3rem;border-bottom-right-radius:.3rem}.react-datepicker-popper{z-index:1;line-height:0}.react-datepicker-popper .react-datepicker__triangle{stroke:#aeaeae}.react-datepicker-popper[data-placement^=bottom] .react-datepicker__triangle{fill:#f0f0f0;color:#f0f0f0}.react-datepicker-popper[data-placement^=top] .react-datepicker__triangle{fill:#fff;color:#fff}.react-datepicker__header{text-align:center;background-color:#f0f0f0;border-bottom:1px solid #aeaeae;border-top-left-radius:.3rem;padding:8px 0;position:relative}.react-datepicker__header--time{padding-bottom:8px;padding-left:5px;padding-right:5px}.react-datepicker__header--time:not(.react-datepicker__header--time--only){border-top-left-radius:0}.react-datepicker__header:not(.react-datepicker__header--has-time-select){border-top-right-radius:.3rem}.react-datepicker__year-dropdown-container--select,.react-datepicker__month-dropdown-container--select,.react-datepicker__month-year-dropdown-container--select,.react-datepicker__year-dropdown-container--scroll,.react-datepicker__month-dropdown-container--scroll,.react-datepicker__month-year-dropdown-container--scroll{display:inline-block;margin:0 15px}.react-datepicker__current-month,.react-datepicker-time__header,.react-datepicker-year-header{margin-top:0;color:#000;font-weight:700;font-size:.944rem}h2.react-datepicker__current-month{padding:0;margin:0}.react-datepicker-time__header{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.react-datepicker__navigation{align-items:center;background:none;display:flex;justify-content:center;text-align:center;cursor:pointer;position:absolute;top:2px;padding:0;border:none;z-index:1;height:32px;width:32px;text-indent:-999em;overflow:hidden}.react-datepicker__navigation--previous{left:2px}.react-datepicker__navigation--next{right:2px}.react-datepicker__navigation--next--with-time:not(.react-datepicker__navigation--next--with-today-button){right:85px}.react-datepicker__navigation--years{position:relative;top:0;display:block;margin-left:auto;margin-right:auto}.react-datepicker__navigation--years-previous{top:4px}.react-datepicker__navigation--years-upcoming{top:-4px}.react-datepicker__navigation:hover *:before{border-color:#a6a6a6}.react-datepicker__navigation-icon{position:relative;top:-1px;font-size:20px;width:0}.react-datepicker__navigation-icon--next{left:-2px}.react-datepicker__navigation-icon--next:before{transform:rotate(45deg);left:-7px}.react-datepicker__navigation-icon--previous{right:-2px}.react-datepicker__navigation-icon--previous:before{transform:rotate(225deg);right:-7px}.react-datepicker__month-container{float:left}.react-datepicker__year{margin:.4rem;text-align:center}.react-datepicker__year-wrapper{display:flex;flex-wrap:wrap;max-width:180px}.react-datepicker__year .react-datepicker__year-text{display:inline-block;width:4rem;margin:2px}.react-datepicker__month{margin:.4rem;text-align:center}.react-datepicker__month .react-datepicker__month-text,.react-datepicker__month .react-datepicker__quarter-text{display:inline-block;width:4rem;margin:2px}.react-datepicker__input-time-container{clear:both;width:100%;float:left;margin:5px 0 10px 15px;text-align:left}.react-datepicker__input-time-container .react-datepicker-time__caption,.react-datepicker__input-time-container .react-datepicker-time__input-container{display:inline-block}.react-datepicker__input-time-container .react-datepicker-time__input-container .react-datepicker-time__input{display:inline-block;margin-left:10px}.react-datepicker__input-time-container .react-datepicker-time__input-container .react-datepicker-time__input input{width:auto}.react-datepicker__input-time-container .react-datepicker-time__input-container .react-datepicker-time__input input[type=time]::-webkit-inner-spin-button,.react-datepicker__input-time-container .react-datepicker-time__input-container .react-datepicker-time__input input[type=time]::-webkit-outer-spin-button{-webkit-appearance:none;margin:0}.react-datepicker__input-time-container .react-datepicker-time__input-container .react-datepicker-time__input input[type=time]{-moz-appearance:textfield}.react-datepicker__input-time-container .react-datepicker-time__input-container .react-datepicker-time__delimiter{margin-left:5px;display:inline-block}.react-datepicker__time-container{float:right;border-left:1px solid #aeaeae;width:85px}.react-datepicker__time-container--with-today-button{display:inline;border:1px solid #aeaeae;border-radius:.3rem;position:absolute;right:-87px;top:0}.react-datepicker__time-container .react-datepicker__time{position:relative;background:#fff;border-bottom-right-radius:.3rem}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box{width:85px;overflow-x:hidden;margin:0 auto;text-align:center;border-bottom-right-radius:.3rem}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box ul.react-datepicker__time-list{list-style:none;margin:0;height:calc(195px + .85rem);overflow-y:scroll;padding-right:0;padding-left:0;width:100%;box-sizing:content-box}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box ul.react-datepicker__time-list li.react-datepicker__time-list-item{height:30px;padding:5px 10px;white-space:nowrap}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box ul.react-datepicker__time-list li.react-datepicker__time-list-item:hover{cursor:pointer;background-color:#f0f0f0}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box ul.react-datepicker__time-list li.react-datepicker__time-list-item--selected{background-color:#216ba5;color:#fff;font-weight:700}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box ul.react-datepicker__time-list li.react-datepicker__time-list-item--selected:hover{background-color:#216ba5}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box ul.react-datepicker__time-list li.react-datepicker__time-list-item--disabled{color:#ccc}.react-datepicker__time-container .react-datepicker__time .react-datepicker__time-box ul.react-datepicker__time-list li.react-datepicker__time-list-item--disabled:hover{cursor:default;background-color:transparent}.react-datepicker__week-number{color:#ccc;display:inline-block;width:1.7rem;line-height:1.7rem;text-align:center;margin:.166rem}.react-datepicker__week-number.react-datepicker__week-number--clickable{cursor:pointer}.react-datepicker__week-number.react-datepicker__week-number--clickable:not(.react-datepicker__week-number--selected):hover{border-radius:.3rem;background-color:#f0f0f0}.react-datepicker__week-number--selected{border-radius:.3rem;background-color:#216ba5;color:#fff}.react-datepicker__week-number--selected:hover{background-color:#1d5d90}.react-datepicker__day-names{white-space:nowrap;margin-bottom:-8px}.react-datepicker__week{white-space:nowrap}.react-datepicker__day-name,.react-datepicker__day,.react-datepicker__time-name{color:#000;display:inline-block;width:1.7rem;line-height:1.7rem;text-align:center;margin:.166rem}.react-datepicker__day,.react-datepicker__month-text,.react-datepicker__quarter-text,.react-datepicker__year-text{cursor:pointer}.react-datepicker__day:not([aria-disabled=true]):hover,.react-datepicker__month-text:not([aria-disabled=true]):hover,.react-datepicker__quarter-text:not([aria-disabled=true]):hover,.react-datepicker__year-text:not([aria-disabled=true]):hover{border-radius:.3rem;background-color:#f0f0f0}.react-datepicker__day--today,.react-datepicker__month-text--today,.react-datepicker__quarter-text--today,.react-datepicker__year-text--today{font-weight:700}.react-datepicker__day--highlighted,.react-datepicker__month-text--highlighted,.react-datepicker__quarter-text--highlighted,.react-datepicker__year-text--highlighted{border-radius:.3rem;background-color:#3dcc4a;color:#fff}.react-datepicker__day--highlighted:not([aria-disabled=true]):hover,.react-datepicker__month-text--highlighted:not([aria-disabled=true]):hover,.react-datepicker__quarter-text--highlighted:not([aria-disabled=true]):hover,.react-datepicker__year-text--highlighted:not([aria-disabled=true]):hover{background-color:#32be3f}.react-datepicker__day--highlighted-custom-1,.react-datepicker__month-text--highlighted-custom-1,.react-datepicker__quarter-text--highlighted-custom-1,.react-datepicker__year-text--highlighted-custom-1{color:#f0f}.react-datepicker__day--highlighted-custom-2,.react-datepicker__month-text--highlighted-custom-2,.react-datepicker__quarter-text--highlighted-custom-2,.react-datepicker__year-text--highlighted-custom-2{color:green}.react-datepicker__day--holidays,.react-datepicker__month-text--holidays,.react-datepicker__quarter-text--holidays,.react-datepicker__year-text--holidays{position:relative;border-radius:.3rem;background-color:#ff6803;color:#fff}.react-datepicker__day--holidays .overlay,.react-datepicker__month-text--holidays .overlay,.react-datepicker__quarter-text--holidays .overlay,.react-datepicker__year-text--holidays .overlay{position:absolute;bottom:100%;left:50%;transform:translate(-50%);background-color:#333;color:#fff;padding:4px;border-radius:4px;white-space:nowrap;visibility:hidden;opacity:0;transition:visibility 0s,opacity .3s ease-in-out}.react-datepicker__day--holidays:not([aria-disabled=true]):hover,.react-datepicker__month-text--holidays:not([aria-disabled=true]):hover,.react-datepicker__quarter-text--holidays:not([aria-disabled=true]):hover,.react-datepicker__year-text--holidays:not([aria-disabled=true]):hover{background-color:#cf5300}.react-datepicker__day--holidays:hover .overlay,.react-datepicker__month-text--holidays:hover .overlay,.react-datepicker__quarter-text--holidays:hover .overlay,.react-datepicker__year-text--holidays:hover .overlay{visibility:visible;opacity:1}.react-datepicker__day--selected,.react-datepicker__day--in-selecting-range,.react-datepicker__day--in-range,.react-datepicker__month-text--selected,.react-datepicker__month-text--in-selecting-range,.react-datepicker__month-text--in-range,.react-datepicker__quarter-text--selected,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__quarter-text--in-range,.react-datepicker__year-text--selected,.react-datepicker__year-text--in-selecting-range,.react-datepicker__year-text--in-range{border-radius:.3rem;background-color:#216ba5;color:#fff}.react-datepicker__day--selected:not([aria-disabled=true]):hover,.react-datepicker__day--in-selecting-range:not([aria-disabled=true]):hover,.react-datepicker__day--in-range:not([aria-disabled=true]):hover,.react-datepicker__month-text--selected:not([aria-disabled=true]):hover,.react-datepicker__month-text--in-selecting-range:not([aria-disabled=true]):hover,.react-datepicker__month-text--in-range:not([aria-disabled=true]):hover,.react-datepicker__quarter-text--selected:not([aria-disabled=true]):hover,.react-datepicker__quarter-text--in-selecting-range:not([aria-disabled=true]):hover,.react-datepicker__quarter-text--in-range:not([aria-disabled=true]):hover,.react-datepicker__year-text--selected:not([aria-disabled=true]):hover,.react-datepicker__year-text--in-selecting-range:not([aria-disabled=true]):hover,.react-datepicker__year-text--in-range:not([aria-disabled=true]):hover{background-color:#1d5d90}.react-datepicker__day--keyboard-selected,.react-datepicker__month-text--keyboard-selected,.react-datepicker__quarter-text--keyboard-selected,.react-datepicker__year-text--keyboard-selected{border-radius:.3rem;background-color:#bad9f1;color:#000}.react-datepicker__day--keyboard-selected:not([aria-disabled=true]):hover,.react-datepicker__month-text--keyboard-selected:not([aria-disabled=true]):hover,.react-datepicker__quarter-text--keyboard-selected:not([aria-disabled=true]):hover,.react-datepicker__year-text--keyboard-selected:not([aria-disabled=true]):hover{background-color:#1d5d90}.react-datepicker__day--in-selecting-range:not(.react-datepicker__day--in-range,.react-datepicker__month-text--in-range,.react-datepicker__quarter-text--in-range,.react-datepicker__year-text--in-range),.react-datepicker__month-text--in-selecting-range:not(.react-datepicker__day--in-range,.react-datepicker__month-text--in-range,.react-datepicker__quarter-text--in-range,.react-datepicker__year-text--in-range),.react-datepicker__quarter-text--in-selecting-range:not(.react-datepicker__day--in-range,.react-datepicker__month-text--in-range,.react-datepicker__quarter-text--in-range,.react-datepicker__year-text--in-range),.react-datepicker__year-text--in-selecting-range:not(.react-datepicker__day--in-range,.react-datepicker__month-text--in-range,.react-datepicker__quarter-text--in-range,.react-datepicker__year-text--in-range){background-color:#216ba580}.react-datepicker__month--selecting-range .react-datepicker__day--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range),.react-datepicker__year--selecting-range .react-datepicker__day--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range),.react-datepicker__month--selecting-range .react-datepicker__month-text--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range),.react-datepicker__year--selecting-range .react-datepicker__month-text--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range),.react-datepicker__month--selecting-range .react-datepicker__quarter-text--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range),.react-datepicker__year--selecting-range .react-datepicker__quarter-text--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range),.react-datepicker__month--selecting-range .react-datepicker__year-text--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range),.react-datepicker__year--selecting-range .react-datepicker__year-text--in-range:not(.react-datepicker__day--in-selecting-range,.react-datepicker__month-text--in-selecting-range,.react-datepicker__quarter-text--in-selecting-range,.react-datepicker__year-text--in-selecting-range){background-color:#f0f0f0;color:#000}.react-datepicker__day--disabled,.react-datepicker__month-text--disabled,.react-datepicker__quarter-text--disabled,.react-datepicker__year-text--disabled{cursor:default;color:#ccc}.react-datepicker__day--disabled .overlay,.react-datepicker__month-text--disabled .overlay,.react-datepicker__quarter-text--disabled .overlay,.react-datepicker__year-text--disabled .overlay{position:absolute;bottom:70%;left:50%;transform:translate(-50%);background-color:#333;color:#fff;padding:4px;border-radius:4px;white-space:nowrap;visibility:hidden;opacity:0;transition:visibility 0s,opacity .3s ease-in-out}.react-datepicker__input-container{position:relative;display:inline-block;width:100%}.react-datepicker__input-container .react-datepicker__calendar-icon{position:absolute;padding:.5rem;box-sizing:content-box}.react-datepicker__view-calendar-icon input{padding:6px 10px 5px 25px}.react-datepicker__year-read-view,.react-datepicker__month-read-view,.react-datepicker__month-year-read-view{border:1px solid transparent;border-radius:.3rem;position:relative}.react-datepicker__year-read-view:hover,.react-datepicker__month-read-view:hover,.react-datepicker__month-year-read-view:hover{cursor:pointer}.react-datepicker__year-read-view:hover .react-datepicker__year-read-view--down-arrow,.react-datepicker__year-read-view:hover .react-datepicker__month-read-view--down-arrow,.react-datepicker__month-read-view:hover .react-datepicker__year-read-view--down-arrow,.react-datepicker__month-read-view:hover .react-datepicker__month-read-view--down-arrow,.react-datepicker__month-year-read-view:hover .react-datepicker__year-read-view--down-arrow,.react-datepicker__month-year-read-view:hover .react-datepicker__month-read-view--down-arrow{border-top-color:#b3b3b3}.react-datepicker__year-read-view--down-arrow,.react-datepicker__month-read-view--down-arrow,.react-datepicker__month-year-read-view--down-arrow{transform:rotate(135deg);right:-16px;top:0}.react-datepicker__year-dropdown,.react-datepicker__month-dropdown,.react-datepicker__month-year-dropdown{background-color:#f0f0f0;position:absolute;width:50%;left:25%;top:30px;z-index:1;text-align:center;border-radius:.3rem;border:1px solid #aeaeae}.react-datepicker__year-dropdown:hover,.react-datepicker__month-dropdown:hover,.react-datepicker__month-year-dropdown:hover{cursor:pointer}.react-datepicker__year-dropdown--scrollable,.react-datepicker__month-dropdown--scrollable,.react-datepicker__month-year-dropdown--scrollable{height:150px;overflow-y:scroll}.react-datepicker__year-option,.react-datepicker__month-option,.react-datepicker__month-year-option{line-height:20px;width:100%;display:block;margin-left:auto;margin-right:auto}.react-datepicker__year-option:first-of-type,.react-datepicker__month-option:first-of-type,.react-datepicker__month-year-option:first-of-type{border-top-left-radius:.3rem;border-top-right-radius:.3rem}.react-datepicker__year-option:last-of-type,.react-datepicker__month-option:last-of-type,.react-datepicker__month-year-option:last-of-type{-webkit-user-select:none;-moz-user-select:none;user-select:none;border-bottom-left-radius:.3rem;border-bottom-right-radius:.3rem}.react-datepicker__year-option:hover,.react-datepicker__month-option:hover,.react-datepicker__month-year-option:hover{background-color:#ccc}.react-datepicker__year-option:hover .react-datepicker__navigation--years-upcoming,.react-datepicker__month-option:hover .react-datepicker__navigation--years-upcoming,.react-datepicker__month-year-option:hover .react-datepicker__navigation--years-upcoming{border-bottom-color:#b3b3b3}.react-datepicker__year-option:hover .react-datepicker__navigation--years-previous,.react-datepicker__month-option:hover .react-datepicker__navigation--years-previous,.react-datepicker__month-year-option:hover .react-datepicker__navigation--years-previous{border-top-color:#b3b3b3}.react-datepicker__year-option--selected,.react-datepicker__month-option--selected,.react-datepicker__month-year-option--selected{position:absolute;left:15px}.react-datepicker__close-icon{cursor:pointer;background-color:transparent;border:0;outline:0;padding:0 6px 0 0;position:absolute;top:0;right:0;height:100%;display:table-cell;vertical-align:middle}.react-datepicker__close-icon:after{cursor:pointer;background-color:#216ba5;color:#fff;border-radius:50%;height:16px;width:16px;padding:2px;font-size:12px;line-height:1;text-align:center;display:table-cell;vertical-align:middle;content:"×"}.react-datepicker__close-icon--disabled{cursor:default}.react-datepicker__close-icon--disabled:after{cursor:default;background-color:#ccc}.react-datepicker__today-button{background:#f0f0f0;border-top:1px solid #aeaeae;cursor:pointer;text-align:center;font-weight:700;padding:5px 0;clear:left}.react-datepicker__portal{position:fixed;width:100vw;height:100vh;background-color:#000c;left:0;top:0;justify-content:center;align-items:center;display:flex;z-index:2147483647}.react-datepicker__portal .react-datepicker__day-name,.react-datepicker__portal .react-datepicker__day,.react-datepicker__portal .react-datepicker__time-name{width:3rem;line-height:3rem}@media (max-width: 400px),(max-height: 550px){.react-datepicker__portal .react-datepicker__day-name,.react-datepicker__portal .react-datepicker__day,.react-datepicker__portal .react-datepicker__time-name{width:2rem;line-height:2rem}}.react-datepicker__portal .react-datepicker__current-month,.react-datepicker__portal .react-datepicker-time__header{font-size:1.44rem}.react-datepicker__children-container{width:13.8rem;margin:.4rem;padding-right:.2rem;padding-left:.2rem;height:auto}.react-datepicker__aria-live{position:absolute;clip-path:circle(0);border:0;height:1px;margin:-1px;overflow:hidden;padding:0;width:1px;white-space:nowrap}.react-datepicker__calendar-icon{width:1em;height:1em;vertical-align:-.125em} diff --git a/frontend/dist/assets/IconButton-DmJ5sToe.js b/frontend/dist/assets/IconButton-DmJ5sToe.js new file mode 100644 index 0000000000..d6c4c1310a --- /dev/null +++ b/frontend/dist/assets/IconButton-DmJ5sToe.js @@ -0,0 +1 @@ +import{R as m,j as l,B as i,n as u}from"./index-tEqMizXb.js";const b=m.forwardRef(({className:t,children:o,"aria-label":n,size:a="md",variant:s="ghost",type:r="button",...c},e)=>l.jsx(i,{className:u("btn-icon",a==="sm"&&"h-8 w-8",a==="md"&&"h-10 w-10",a==="lg"&&"h-12 w-12",t),ref:e,size:"icon",variant:s,type:r,"aria-label":n,...c,children:o}));b.displayName="IconButton";export{b as I}; diff --git a/frontend/dist/assets/LandingPage-EXm-0yOi.js b/frontend/dist/assets/LandingPage-EXm-0yOi.js new file mode 100644 index 0000000000..fdd95d1157 --- /dev/null +++ b/frontend/dist/assets/LandingPage-EXm-0yOi.js @@ -0,0 +1 @@ +import{j as e,L as a}from"./index-tEqMizXb.js";import"./IconButton-DmJ5sToe.js";import"./Pagination-EPc_5dQ6.js";import"./TreeDetailModal-PvfFs-NG.js";import"./ChartComponents-BKyCVLLP.js";import{N as i}from"./Navbar-BjYzKaxh.js";import"./vendor-BtP0CW_r.js";import"./useKeyboardNavigation-DT9lybaQ.js";import"./DarkModeToggle-ZfYhBz0e.js";const n=()=>e.jsxs("section",{className:"pt-20 bg-gradient-to-br from-green-500 to-green-700 dark:from-green-600 dark:to-green-800 min-h-screen flex items-center justify-center relative overflow-hidden",children:[e.jsx("div",{className:"absolute inset-0 opacity-30",style:{backgroundImage:`url("data:image/svg+xml,%3Csvg width='60' height='60' viewBox='0 0 60 60' xmlns='http://www.w3.org/2000/svg'%3E%3Cg fill='none' fill-rule='evenodd'%3E%3Cg fill='%23ffffff' fill-opacity='0.1'%3E%3Ccircle cx='30' cy='30' r='2'/%3E%3C/g%3E%3C/g%3E%3C/svg%3E")`}}),e.jsx("div",{className:"relative z-10 max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-24",children:e.jsxs("div",{className:"text-center",children:[e.jsxs("h1",{className:"text-4xl md:text-6xl font-bold text-white mb-6",children:["Discover, Monitor & Export",e.jsx("span",{className:"block text-green-200",children:"Real-Time Tree Insights"})]}),e.jsx("p",{className:"text-xl md:text-2xl text-green-100 mb-8 max-w-3xl mx-auto",children:"Nanwa provides investors, growers, and admins with comprehensive tools to track every tree in our registry with real-time data and actionable insights."}),e.jsxs("div",{className:"flex flex-col sm:flex-row gap-4 justify-center",children:[e.jsx(a,{to:"/register",className:"bg-white dark:bg-gray-100 text-green-600 dark:text-green-700 px-8 py-4 rounded-lg text-lg font-semibold hover:bg-gray-100 dark:hover:bg-gray-200 transition-colors",children:"Start Your Free Trial"}),e.jsx("a",{href:"#features",className:"border-2 border-white text-white px-8 py-4 rounded-lg text-lg font-semibold hover:bg-white hover:text-green-600 dark:hover:bg-gray-100 dark:hover:text-green-700 transition-colors",children:"Learn More"})]})]})})]}),l=({icon:r,title:t,description:s})=>e.jsxs("div",{className:"bg-white dark:bg-gray-900 p-8 rounded-xl shadow-lg dark:shadow-gray-900/50",children:[e.jsx("div",{className:"w-12 h-12 bg-green-100 dark:bg-green-900/30 rounded-lg flex items-center justify-center mb-6",children:r}),e.jsx("h3",{className:"text-xl font-semibold text-gray-900 dark:text-white mb-4",children:t}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300",children:s})]}),o=()=>{const r=[{icon:e.jsx("svg",{className:"w-6 h-6 text-green-600 dark:text-green-400",fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"})}),title:"Real-Time Analytics",description:"Monitor survival rates, average height, and CO₂ absorption with live-updating charts and metrics."},{icon:e.jsxs("svg",{className:"w-6 h-6 text-green-600 dark:text-green-400",fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:[e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z"}),e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M15 11a3 3 0 11-6 0 3 3 0 016 0z"})]}),title:"Interactive Mapping",description:"Visualize forests and individual trees with marker clustering and detailed tree information."},{icon:e.jsx("svg",{className:"w-6 h-6 text-green-600 dark:text-green-400",fill:"none",stroke:"currentColor",viewBox:"0 0 24 24",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"})}),title:"Data Export",description:"Export filtered datasets to CSV or XLSX format for reporting and analysis."}];return e.jsx("section",{id:"features",className:"py-20 bg-gray-50 dark:bg-gray-800",children:e.jsxs("div",{className:"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8",children:[e.jsxs("div",{className:"text-center mb-16",children:[e.jsx("h2",{className:"text-3xl md:text-4xl font-bold text-gray-900 dark:text-white mb-4",children:"Powerful Features for Tree Management"}),e.jsx("p",{className:"text-xl text-gray-600 dark:text-gray-300 max-w-2xl mx-auto",children:"Everything you need to monitor, analyze, and export tree data with precision and ease."})]}),e.jsx("div",{className:"grid md:grid-cols-3 gap-8",children:r.map((t,s)=>e.jsx(l,{icon:t.icon,title:t.title,description:t.description},s))})]})})},d=()=>{const r=[{value:"10K+",label:"Trees Monitored"},{value:"95%",label:"Survival Rate"},{value:"50+",label:"Forest Projects"},{value:"24/7",label:"Real-Time Data"}];return e.jsx("div",{className:"bg-green-50 dark:bg-green-900/20 p-8 rounded-xl",children:e.jsx("div",{className:"grid grid-cols-2 gap-4",children:r.map((t,s)=>e.jsxs("div",{className:"text-center",children:[e.jsx("div",{className:"text-3xl font-bold text-green-600 dark:text-green-400 mb-2",children:t.value}),e.jsx("div",{className:"text-gray-600 dark:text-gray-300",children:t.label})]},s))})})},c=()=>e.jsx("section",{id:"about",className:"py-20 bg-white dark:bg-gray-900",children:e.jsx("div",{className:"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8",children:e.jsxs("div",{className:"grid md:grid-cols-2 gap-12 items-center",children:[e.jsxs("div",{children:[e.jsx("h2",{className:"text-3xl md:text-4xl font-bold text-gray-900 dark:text-white mb-6",children:"Empowering Sustainable Forestry"}),e.jsx("p",{className:"text-lg text-gray-600 dark:text-gray-300 mb-6",children:"Nanwa is dedicated to providing comprehensive tree monitoring solutions that help investors, growers, and environmental analysts make data-driven decisions for sustainable forestry projects."}),e.jsx("p",{className:"text-lg text-gray-600 dark:text-gray-300 mb-8",children:"Our platform offers real-time insights, advanced analytics, and seamless data export capabilities to support your forestry management needs."}),e.jsx(a,{to:"/register",className:"bg-green-600 dark:bg-green-500 text-white px-6 py-3 rounded-lg text-lg font-semibold hover:bg-green-700 dark:hover:bg-green-600 transition-colors inline-block",children:"Join Nanwa Today"})]}),e.jsx(d,{})]})})}),x=()=>e.jsx("footer",{className:"bg-gray-900 text-white py-12",children:e.jsxs("div",{className:"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8",children:[e.jsxs("div",{className:"grid md:grid-cols-4 gap-8",children:[e.jsxs("div",{children:[e.jsx("h3",{className:"text-2xl font-bold text-green-400 mb-4",children:"Nanwa"}),e.jsx("p",{className:"text-gray-400",children:"Empowering sustainable forestry through data-driven insights and real-time monitoring."})]}),e.jsxs("div",{children:[e.jsx("h4",{className:"text-lg font-semibold mb-4",children:"Product"}),e.jsxs("ul",{className:"space-y-2 text-gray-400",children:[e.jsx("li",{children:e.jsx("a",{href:"#features",className:"hover:text-white transition-colors",children:"Features"})}),e.jsx("li",{children:e.jsx(a,{to:"/login",className:"hover:text-white transition-colors",children:"Login"})}),e.jsx("li",{children:e.jsx(a,{to:"/register",className:"hover:text-white transition-colors",children:"Register"})})]})]}),e.jsxs("div",{children:[e.jsx("h4",{className:"text-lg font-semibold mb-4",children:"Company"}),e.jsxs("ul",{className:"space-y-2 text-gray-400",children:[e.jsx("li",{children:e.jsx("a",{href:"#about",className:"hover:text-white transition-colors",children:"About"})}),e.jsx("li",{children:e.jsx("a",{href:"#",className:"hover:text-white transition-colors",children:"Contact"})}),e.jsx("li",{children:e.jsx("a",{href:"#",className:"hover:text-white transition-colors",children:"Privacy"})})]})]}),e.jsxs("div",{children:[e.jsx("h4",{className:"text-lg font-semibold mb-4",children:"Support"}),e.jsxs("ul",{className:"space-y-2 text-gray-400",children:[e.jsx("li",{children:e.jsx("a",{href:"#",className:"hover:text-white transition-colors",children:"Help Center"})}),e.jsx("li",{children:e.jsx("a",{href:"#",className:"hover:text-white transition-colors",children:"Documentation"})}),e.jsx("li",{children:e.jsx("a",{href:"#",className:"hover:text-white transition-colors",children:"API"})})]})]})]}),e.jsx("div",{className:"border-t border-gray-800 mt-8 pt-8 text-center text-gray-400",children:e.jsx("p",{children:"© 2024 Nanwa. All rights reserved."})})]})}),N=()=>e.jsxs("div",{className:"min-h-screen bg-white dark:bg-gray-900",children:[e.jsx(i,{}),e.jsx(n,{}),e.jsx(o,{}),e.jsx(c,{}),e.jsx(x,{})]});export{N as LandingPage}; diff --git a/frontend/dist/assets/LoginPage-BeBDoSBE.js b/frontend/dist/assets/LoginPage-BeBDoSBE.js new file mode 100644 index 0000000000..7fb1ad5115 --- /dev/null +++ b/frontend/dist/assets/LoginPage-BeBDoSBE.js @@ -0,0 +1 @@ +import{r as o,j as e,L as f,u as j,a as k}from"./index-tEqMizXb.js";import{N as w}from"./Navbar-BjYzKaxh.js";import{F as y,P as v}from"./PasswordInput-8zSydSJy.js";import"./vendor-BtP0CW_r.js";import"./useKeyboardNavigation-DT9lybaQ.js";import"./DarkModeToggle-ZfYhBz0e.js";import"./IconButton-DmJ5sToe.js";const C=({connectionState:r,onCancel:t,showCancel:l=!0})=>{const[a,i]=o.useState(0),[c,u]=o.useState(0),{isConnecting:s,isColdStart:n,retryAttempt:d,totalAttempts:h,estimatedWaitTime:p,message:g}=r;return o.useEffect(()=>{if(!s){i(0),u(0);return}const x=setInterval(()=>{if(u(m=>m+1),n){const b=Math.min(c/60*100,90);i(b)}else{const b=Math.min(c/10*100,90);i(b)}},1e3);return()=>clearInterval(x)},[s,n,c]),o.useEffect(()=>{!s&&a>0&&i(100)},[s,a]),s?e.jsx("div",{className:"fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg p-8 max-w-md w-full mx-4 shadow-2xl",children:[e.jsxs("div",{className:"text-center mb-6",children:[e.jsx("div",{className:"w-16 h-16 bg-green-100 dark:bg-green-900/30 rounded-full flex items-center justify-center mx-auto mb-4",children:e.jsxs("svg",{className:"w-8 h-8 text-green-600 dark:text-green-400 animate-spin",fill:"none",viewBox:"0 0 24 24",children:[e.jsx("circle",{className:"opacity-25",cx:"12",cy:"12",r:"10",stroke:"currentColor",strokeWidth:"4"}),e.jsx("path",{className:"opacity-75",fill:"currentColor",d:"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"})]})}),e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white",children:n?"Server Starting Up":"Connecting"})]}),e.jsxs("div",{className:"text-center mb-6",children:[e.jsx("p",{className:"text-gray-600 dark:text-gray-300 mb-2",children:g}),n&&e.jsxs("div",{className:"text-sm text-gray-500 dark:text-gray-400 space-y-1",children:[e.jsx("p",{children:"The server needs to start up from sleep mode."}),e.jsx("p",{children:"This usually takes 30-60 seconds."})]})]}),e.jsxs("div",{className:"mb-6",children:[e.jsxs("div",{className:"flex justify-between text-sm text-gray-500 dark:text-gray-400 mb-2",children:[e.jsx("span",{children:"Progress"}),e.jsxs("span",{children:[Math.round(a),"%"]})]}),e.jsx("div",{className:"w-full bg-gray-200 dark:bg-gray-700 rounded-full h-2",children:e.jsx("div",{className:"bg-green-600 h-2 rounded-full transition-all duration-1000 ease-out",style:{width:`${a}%`}})})]}),d>0&&e.jsx("div",{className:"bg-blue-50 dark:bg-blue-900/20 p-3 rounded-lg mb-4",children:e.jsxs("div",{className:"flex items-center justify-between text-sm",children:[e.jsxs("span",{className:"text-blue-600 dark:text-blue-400",children:["Attempt ",d," of ",h]}),e.jsxs("span",{className:"text-blue-500 dark:text-blue-300",children:[c,"s elapsed"]})]})}),p>0&&e.jsxs("div",{className:"text-center text-sm text-gray-500 dark:text-gray-400 mb-4",children:["Estimated wait time: ",p," seconds"]}),l&&t&&e.jsx("div",{className:"text-center",children:e.jsx("button",{onClick:t,className:"text-gray-500 dark:text-gray-400 hover:text-gray-700 dark:hover:text-gray-200 text-sm underline",children:"Cancel and try again later"})})]})}):null},N=(r,t)=>{const[l,a]=o.useState(r),[i,c]=o.useState({}),[u,s]=o.useState({}),n=o.useCallback(g=>{const{name:x,value:m}=g.target;a(b=>({...b,[x]:m})),i[x]&&c(b=>({...b,[x]:""}))},[i]),d=o.useCallback(g=>{const{name:x}=g.target;s(m=>({...m,[x]:!0}))},[]),h=o.useCallback(()=>{const g=t(l);return c(g),Object.keys(g).length===0},[l,t]),p=o.useCallback(()=>{a(r),c({}),s({})},[r]);return{values:l,errors:i,touched:u,handleChange:n,handleBlur:d,validateForm:h,resetForm:p,setErrors:c}},S=r=>{const t={};return r.email?/\S+@\S+\.\S+/.test(r.email)||(t.email="Email is invalid"):t.email="Email is required",r.password?r.password.length<6&&(t.password="Password must be at least 6 characters"):t.password="Password is required",t},A=({onSubmit:r,isSubmitting:t,errors:l})=>{const{values:a,errors:i,handleChange:c,handleBlur:u,validateForm:s}=N({email:"",password:""},S),n={...i,...l},d=h=>{h.preventDefault(),s()&&r(a)};return e.jsxs("form",{onSubmit:d,className:"space-y-6",children:[n.general&&e.jsx("div",{className:"bg-red-50 dark:bg-red-900/30 border border-red-200 dark:border-red-800 text-red-700 dark:text-red-400 px-4 py-3 rounded-lg",children:n.general}),e.jsx(y,{label:"Email address",id:"email",error:n.email,required:!0,children:e.jsx("input",{type:"email",id:"email",name:"email",value:a.email,onChange:c,onBlur:u,className:`w-full px-3 py-2 border rounded-lg focus:outline-none focus:ring-2 focus:ring-green-500 focus:border-transparent bg-white dark:bg-gray-700 text-gray-900 dark:text-white placeholder-gray-500 dark:placeholder-gray-400 ${n.email?"border-red-300 dark:border-red-600":"border-gray-300 dark:border-gray-600"}`,placeholder:"Enter your email",disabled:t})}),e.jsx(y,{children:e.jsx(v,{id:"password",name:"password",value:a.password,onChange:c,onBlur:u,error:n.password,placeholder:"Enter your password",required:!0,disabled:t})}),e.jsxs("div",{className:"flex items-center justify-between",children:[e.jsx(E,{disabled:t}),e.jsx(F,{})]}),e.jsx(T,{isSubmitting:t}),e.jsx(P,{})]})},E=({disabled:r})=>e.jsxs("div",{className:"flex items-center",children:[e.jsx("input",{id:"remember-me",name:"remember-me",type:"checkbox",className:"h-4 w-4 text-green-600 focus:ring-green-500 border-gray-300 dark:border-gray-600 rounded bg-white dark:bg-gray-700",disabled:r}),e.jsx("label",{htmlFor:"remember-me",className:"ml-2 block text-sm text-gray-900 dark:text-white",children:"Remember me"})]}),F=()=>e.jsx("div",{className:"text-sm",children:e.jsx("a",{href:"#",className:"font-medium text-green-600 dark:text-green-400 hover:text-green-500 dark:hover:text-green-300",children:"Forgot your password?"})}),T=({isSubmitting:r})=>e.jsx("button",{type:"submit",disabled:r,className:"w-full flex justify-center py-2 px-4 border border-transparent rounded-lg shadow-sm text-sm font-medium text-white bg-green-600 dark:bg-green-500 hover:bg-green-700 dark:hover:bg-green-600 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-green-500 disabled:opacity-50 disabled:cursor-not-allowed",children:r?e.jsxs("div",{className:"flex items-center",children:[e.jsxs("svg",{className:"animate-spin -ml-1 mr-3 h-5 w-5 text-white",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24",children:[e.jsx("circle",{className:"opacity-25",cx:"12",cy:"12",r:"10",stroke:"currentColor",strokeWidth:"4"}),e.jsx("path",{className:"opacity-75",fill:"currentColor",d:"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"})]}),"Signing in..."]}):"Sign in"}),P=()=>e.jsx("div",{className:"text-center",children:e.jsxs("p",{className:"text-sm text-gray-600 dark:text-gray-300",children:["Don't have an account?"," ",e.jsx(f,{to:"/register",className:"font-medium text-green-600 dark:text-green-400 hover:text-green-500 dark:hover:text-green-300",children:"Sign up"})]})}),L=()=>e.jsxs("div",{className:"mt-8 p-4 bg-gray-50 dark:bg-gray-700 rounded-lg",children:[e.jsx("p",{className:"text-sm text-gray-600 dark:text-gray-300 mb-2",children:"Demo credentials:"}),e.jsxs("div",{className:"text-xs text-gray-500 dark:text-gray-400 space-y-1",children:[e.jsxs("p",{children:[e.jsx("strong",{children:"Admin:"})," admin@nanwa.com / admin123"]}),e.jsxs("p",{children:[e.jsx("strong",{children:"User:"})," user@nanwa.com / user123"]})]})]}),M=()=>{const[r,t]=o.useState({isConnecting:!1,isColdStart:!1,retryAttempt:0,totalAttempts:0,estimatedWaitTime:0,message:"",error:null}),l=o.useCallback(s=>{const{attempt:n,totalAttempts:d,delay:h,error:p,isColdStart:g}=s;t(x=>({...x,isConnecting:!0,isColdStart:g,retryAttempt:n,totalAttempts:d,estimatedWaitTime:Math.ceil(h/1e3),message:W(n,d,g),error:null}))},[]),a=o.useCallback(()=>{t({isConnecting:!0,isColdStart:!1,retryAttempt:0,totalAttempts:0,estimatedWaitTime:10,message:"Connecting to server...",error:null})},[]),i=o.useCallback(()=>{t({isConnecting:!1,isColdStart:!1,retryAttempt:0,totalAttempts:0,estimatedWaitTime:0,message:"Connected successfully!",error:null})},[]),c=o.useCallback(s=>{const n=s&&typeof s=="object"?s.message||String(s):String(s||"Connection failed");t(d=>({...d,isConnecting:!1,error:n,message:"Connection failed. Please try again."}))},[]),u=o.useCallback(()=>{t({isConnecting:!1,isColdStart:!1,retryAttempt:0,totalAttempts:0,estimatedWaitTime:0,message:"",error:null})},[]);return{connectionState:r,handleRetry:l,startConnection:a,connectionSuccess:i,connectionFailed:c,resetConnection:u}},W=(r,t,l)=>r===1&&l?"Server is starting up, this may take up to 60 seconds...":r===2&&l?"Still starting up, please wait...":r===3&&l?"Almost ready, just a few more seconds...":r>=4&&l?"Final attempt, server should be ready soon...":`Retrying connection (${r}/${t})...`,B=()=>{const[r,t]=o.useState({}),[l,a]=o.useState(!1),{login:i}=j();k();const{connectionState:c,handleRetry:u,startConnection:s,connectionSuccess:n,connectionFailed:d,resetConnection:h}=M();return{errors:r,isSubmitting:l,connectionState:c,handleLogin:async x=>{a(!0),t({}),s();try{const m=await i(x.email,x.password,u);m.success?n():(d(new Error(m.error)),t({general:m.error}))}catch(m){d(m),t({general:"Connection failed. Please check your internet connection and try again."})}finally{a(!1)}},handleCancelConnection:()=>{h(),a(!1)}}},V=()=>{const{errors:r,isSubmitting:t,connectionState:l,handleLogin:a,handleCancelConnection:i}=B();return e.jsxs(e.Fragment,{children:[e.jsx(w,{}),e.jsx(C,{connectionState:l,onCancel:i,showCancel:!0}),e.jsx("div",{className:"min-h-screen bg-gradient-to-br from-green-50 to-green-100 dark:from-gray-900 dark:to-gray-800 flex items-center justify-center p-4",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-2xl shadow-2xl p-8 w-full max-w-md",children:[e.jsx(R,{}),e.jsx(A,{onSubmit:a,isSubmitting:t,errors:r}),e.jsx(L,{})]})})]})},R=()=>e.jsxs("div",{className:"text-center mb-8",children:[e.jsx(f,{to:"/",className:"inline-block mb-6",children:e.jsx("h1",{className:"text-3xl font-bold text-green-600 dark:text-green-400",children:"Nanwa"})}),e.jsx("h2",{className:"text-2xl font-bold text-gray-900 dark:text-white mb-2",children:"Welcome back"}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300",children:"Sign in to your account to continue"})]});export{V as LoginPage}; diff --git a/frontend/dist/assets/MapPage-25t4MxmA.css b/frontend/dist/assets/MapPage-25t4MxmA.css new file mode 100644 index 0000000000..5b4cb38cc7 --- /dev/null +++ b/frontend/dist/assets/MapPage-25t4MxmA.css @@ -0,0 +1 @@ +.leaflet-cluster-anim .leaflet-marker-icon,.leaflet-cluster-anim .leaflet-marker-shadow{transition:transform .3s ease-out,opacity .3s ease-in}.leaflet-cluster-spider-leg{transition:stroke-dashoffset .3s ease-out,stroke-opacity .3s ease-in}.marker-cluster-small{background-color:#b5e28c99}.marker-cluster-small div{background-color:#6ecc3999}.marker-cluster-medium{background-color:#f1d35799}.marker-cluster-medium div{background-color:#f0c20c99}.marker-cluster-large{background-color:#fd9c7399}.marker-cluster-large div{background-color:#f1801799}.leaflet-oldie .marker-cluster-small{background-color:#b5e28c}.leaflet-oldie .marker-cluster-small div{background-color:#6ecc39}.leaflet-oldie .marker-cluster-medium{background-color:#f1d357}.leaflet-oldie .marker-cluster-medium div{background-color:#f0c20c}.leaflet-oldie .marker-cluster-large{background-color:#fd9c73}.leaflet-oldie .marker-cluster-large div{background-color:#f18017}.marker-cluster{background-clip:padding-box;border-radius:20px}.marker-cluster div{width:30px;height:30px;margin-left:5px;margin-top:5px;text-align:center;border-radius:15px;font:12px Helvetica Neue,Arial,Helvetica,sans-serif}.marker-cluster span{line-height:30px}.leaflet-pane,.leaflet-tile,.leaflet-marker-icon,.leaflet-marker-shadow,.leaflet-tile-container,.leaflet-pane>svg,.leaflet-pane>canvas,.leaflet-zoom-box,.leaflet-image-layer,.leaflet-layer{position:absolute;left:0;top:0}.leaflet-container{overflow:hidden}.leaflet-tile,.leaflet-marker-icon,.leaflet-marker-shadow{-webkit-user-select:none;-moz-user-select:none;user-select:none;-webkit-user-drag:none}.leaflet-tile::-moz-selection{background:transparent}.leaflet-tile::selection{background:transparent}.leaflet-safari .leaflet-tile{image-rendering:-webkit-optimize-contrast}.leaflet-safari .leaflet-tile-container{width:1600px;height:1600px;-webkit-transform-origin:0 0}.leaflet-marker-icon,.leaflet-marker-shadow{display:block}.leaflet-container .leaflet-overlay-pane svg{max-width:none!important;max-height:none!important}.leaflet-container .leaflet-marker-pane img,.leaflet-container .leaflet-shadow-pane img,.leaflet-container .leaflet-tile-pane img,.leaflet-container img.leaflet-image-layer,.leaflet-container .leaflet-tile{max-width:none!important;max-height:none!important;width:auto;padding:0}.leaflet-container img.leaflet-tile{mix-blend-mode:plus-lighter}.leaflet-container.leaflet-touch-zoom{touch-action:pan-x pan-y}.leaflet-container.leaflet-touch-drag{touch-action:none;touch-action:pinch-zoom}.leaflet-container.leaflet-touch-drag.leaflet-touch-zoom{touch-action:none}.leaflet-container{-webkit-tap-highlight-color:transparent}.leaflet-container a{-webkit-tap-highlight-color:rgba(51,181,229,.4)}.leaflet-tile{filter:inherit;visibility:hidden}.leaflet-tile-loaded{visibility:inherit}.leaflet-zoom-box{width:0;height:0;box-sizing:border-box;z-index:800}.leaflet-overlay-pane svg{-moz-user-select:none}.leaflet-pane{z-index:400}.leaflet-tile-pane{z-index:200}.leaflet-overlay-pane{z-index:400}.leaflet-shadow-pane{z-index:500}.leaflet-marker-pane{z-index:600}.leaflet-tooltip-pane{z-index:650}.leaflet-popup-pane{z-index:700}.leaflet-map-pane canvas{z-index:100}.leaflet-map-pane svg{z-index:200}.leaflet-vml-shape{width:1px;height:1px}.lvml{behavior:url(#default#VML);display:inline-block;position:absolute}.leaflet-control{position:relative;z-index:800;pointer-events:visiblePainted;pointer-events:auto}.leaflet-top,.leaflet-bottom{position:absolute;z-index:1000;pointer-events:none}.leaflet-top{top:0}.leaflet-right{right:0}.leaflet-bottom{bottom:0}.leaflet-left{left:0}.leaflet-control{float:left;clear:both}.leaflet-right .leaflet-control{float:right}.leaflet-top .leaflet-control{margin-top:10px}.leaflet-bottom .leaflet-control{margin-bottom:10px}.leaflet-left .leaflet-control{margin-left:10px}.leaflet-right .leaflet-control{margin-right:10px}.leaflet-fade-anim .leaflet-popup{opacity:0;transition:opacity .2s linear}.leaflet-fade-anim .leaflet-map-pane .leaflet-popup{opacity:1}.leaflet-zoom-animated{transform-origin:0 0}svg.leaflet-zoom-animated{will-change:transform}.leaflet-zoom-anim .leaflet-zoom-animated{transition:transform .25s cubic-bezier(0,0,.25,1)}.leaflet-zoom-anim .leaflet-tile,.leaflet-pan-anim .leaflet-tile{transition:none}.leaflet-zoom-anim .leaflet-zoom-hide{visibility:hidden}.leaflet-interactive{cursor:pointer}.leaflet-grab{cursor:grab}.leaflet-crosshair,.leaflet-crosshair .leaflet-interactive{cursor:crosshair}.leaflet-popup-pane,.leaflet-control{cursor:auto}.leaflet-dragging .leaflet-grab,.leaflet-dragging .leaflet-grab .leaflet-interactive,.leaflet-dragging .leaflet-marker-draggable{cursor:move;cursor:grabbing}.leaflet-marker-icon,.leaflet-marker-shadow,.leaflet-image-layer,.leaflet-pane>svg path,.leaflet-tile-container{pointer-events:none}.leaflet-marker-icon.leaflet-interactive,.leaflet-image-layer.leaflet-interactive,.leaflet-pane>svg path.leaflet-interactive,svg.leaflet-image-layer.leaflet-interactive path{pointer-events:visiblePainted;pointer-events:auto}.leaflet-container{background:#ddd;outline-offset:1px}.leaflet-container a{color:#0078a8}.leaflet-zoom-box{border:2px dotted #38f;background:#ffffff80}.leaflet-container{font-family:Helvetica Neue,Arial,Helvetica,sans-serif;font-size:12px;font-size:.75rem;line-height:1.5}.leaflet-bar{box-shadow:0 1px 5px #000000a6;border-radius:4px}.leaflet-bar a{background-color:#fff;border-bottom:1px solid #ccc;width:26px;height:26px;line-height:26px;display:block;text-align:center;text-decoration:none;color:#000}.leaflet-bar a,.leaflet-control-layers-toggle{background-position:50% 50%;background-repeat:no-repeat;display:block}.leaflet-bar a:hover,.leaflet-bar a:focus{background-color:#f4f4f4}.leaflet-bar a:first-child{border-top-left-radius:4px;border-top-right-radius:4px}.leaflet-bar a:last-child{border-bottom-left-radius:4px;border-bottom-right-radius:4px;border-bottom:none}.leaflet-bar a.leaflet-disabled{cursor:default;background-color:#f4f4f4;color:#bbb}.leaflet-touch .leaflet-bar a{width:30px;height:30px;line-height:30px}.leaflet-touch .leaflet-bar a:first-child{border-top-left-radius:2px;border-top-right-radius:2px}.leaflet-touch .leaflet-bar a:last-child{border-bottom-left-radius:2px;border-bottom-right-radius:2px}.leaflet-control-zoom-in,.leaflet-control-zoom-out{font:700 18px Lucida Console,Monaco,monospace;text-indent:1px}.leaflet-touch .leaflet-control-zoom-in,.leaflet-touch .leaflet-control-zoom-out{font-size:22px}.leaflet-control-layers{box-shadow:0 1px 5px #0006;background:#fff;border-radius:5px}.leaflet-control-layers-toggle{background-image:url();width:36px;height:36px}.leaflet-retina .leaflet-control-layers-toggle{background-image:url();background-size:26px 26px}.leaflet-touch .leaflet-control-layers-toggle{width:44px;height:44px}.leaflet-control-layers .leaflet-control-layers-list,.leaflet-control-layers-expanded .leaflet-control-layers-toggle{display:none}.leaflet-control-layers-expanded .leaflet-control-layers-list{display:block;position:relative}.leaflet-control-layers-expanded{padding:6px 10px 6px 6px;color:#333;background:#fff}.leaflet-control-layers-scrollbar{overflow-y:scroll;overflow-x:hidden;padding-right:5px}.leaflet-control-layers-selector{margin-top:2px;position:relative;top:1px}.leaflet-control-layers label{display:block;font-size:13px;font-size:1.08333em}.leaflet-control-layers-separator{height:0;border-top:1px solid #ddd;margin:5px -10px 5px -6px}.leaflet-default-icon-path{background-image:url()}.leaflet-container .leaflet-control-attribution{background:#fff;background:#fffc;margin:0}.leaflet-control-attribution,.leaflet-control-scale-line{padding:0 5px;color:#333;line-height:1.4}.leaflet-control-attribution a{text-decoration:none}.leaflet-control-attribution a:hover,.leaflet-control-attribution a:focus{text-decoration:underline}.leaflet-attribution-flag{display:inline!important;vertical-align:baseline!important;width:1em;height:.6669em}.leaflet-left .leaflet-control-scale{margin-left:5px}.leaflet-bottom .leaflet-control-scale{margin-bottom:5px}.leaflet-control-scale-line{border:2px solid #777;border-top:none;line-height:1.1;padding:2px 5px 1px;white-space:nowrap;box-sizing:border-box;background:#fffc;text-shadow:1px 1px #fff}.leaflet-control-scale-line:not(:first-child){border-top:2px solid #777;border-bottom:none;margin-top:-2px}.leaflet-control-scale-line:not(:first-child):not(:last-child){border-bottom:2px solid #777}.leaflet-touch .leaflet-control-attribution,.leaflet-touch .leaflet-control-layers,.leaflet-touch .leaflet-bar{box-shadow:none}.leaflet-touch .leaflet-control-layers,.leaflet-touch .leaflet-bar{border:2px solid rgba(0,0,0,.2);background-clip:padding-box}.leaflet-popup{position:absolute;text-align:center;margin-bottom:20px}.leaflet-popup-content-wrapper{padding:1px;text-align:left;border-radius:12px}.leaflet-popup-content{margin:13px 24px 13px 20px;line-height:1.3;font-size:13px;font-size:1.08333em;min-height:1px}.leaflet-popup-content p{margin:1.3em 0}.leaflet-popup-tip-container{width:40px;height:20px;position:absolute;left:50%;margin-top:-1px;margin-left:-20px;overflow:hidden;pointer-events:none}.leaflet-popup-tip{width:17px;height:17px;padding:1px;margin:-10px auto 0;pointer-events:auto;transform:rotate(45deg)}.leaflet-popup-content-wrapper,.leaflet-popup-tip{background:#fff;color:#333;box-shadow:0 3px 14px #0006}.leaflet-container a.leaflet-popup-close-button{position:absolute;top:0;right:0;border:none;text-align:center;width:24px;height:24px;font:16px/24px Tahoma,Verdana,sans-serif;color:#757575;text-decoration:none;background:transparent}.leaflet-container a.leaflet-popup-close-button:hover,.leaflet-container a.leaflet-popup-close-button:focus{color:#585858}.leaflet-popup-scrolled{overflow:auto}.leaflet-oldie .leaflet-popup-content-wrapper{-ms-zoom:1}.leaflet-oldie .leaflet-popup-tip{width:24px;margin:0 auto;-ms-filter:"progid:DXImageTransform.Microsoft.Matrix(M11=0.70710678, M12=0.70710678, M21=-0.70710678, M22=0.70710678)";filter:progid:DXImageTransform.Microsoft.Matrix(M11=.70710678,M12=.70710678,M21=-.70710678,M22=.70710678)}.leaflet-oldie .leaflet-control-zoom,.leaflet-oldie .leaflet-control-layers,.leaflet-oldie .leaflet-popup-content-wrapper,.leaflet-oldie .leaflet-popup-tip{border:1px solid #999}.leaflet-div-icon{background:#fff;border:1px solid #666}.leaflet-tooltip{position:absolute;padding:6px;background-color:#fff;border:1px solid #fff;border-radius:3px;color:#222;white-space:nowrap;-webkit-user-select:none;-moz-user-select:none;user-select:none;pointer-events:none;box-shadow:0 1px 3px #0006}.leaflet-tooltip.leaflet-interactive{cursor:pointer;pointer-events:auto}.leaflet-tooltip-top:before,.leaflet-tooltip-bottom:before,.leaflet-tooltip-left:before,.leaflet-tooltip-right:before{position:absolute;pointer-events:none;border:6px solid transparent;background:transparent;content:""}.leaflet-tooltip-bottom{margin-top:6px}.leaflet-tooltip-top{margin-top:-6px}.leaflet-tooltip-bottom:before,.leaflet-tooltip-top:before{left:50%;margin-left:-6px}.leaflet-tooltip-top:before{bottom:0;margin-bottom:-12px;border-top-color:#fff}.leaflet-tooltip-bottom:before{top:0;margin-top:-12px;margin-left:-6px;border-bottom-color:#fff}.leaflet-tooltip-left{margin-left:-6px}.leaflet-tooltip-right{margin-left:6px}.leaflet-tooltip-left:before,.leaflet-tooltip-right:before{top:50%;margin-top:-6px}.leaflet-tooltip-left:before{right:0;margin-right:-12px;border-left-color:#fff}.leaflet-tooltip-right:before{left:0;margin-left:-12px;border-right-color:#fff}@media print{.leaflet-control{-webkit-print-color-adjust:exact;print-color-adjust:exact}} diff --git a/frontend/dist/assets/MapPage-B-W_JiyY.js b/frontend/dist/assets/MapPage-B-W_JiyY.js new file mode 100644 index 0000000000..3197affe82 --- /dev/null +++ b/frontend/dist/assets/MapPage-B-W_JiyY.js @@ -0,0 +1,113 @@ +import{A as cs,r as N,R as Mn,d as Ot,j as z,b as In}from"./index-tEqMizXb.js";import{g as ds}from"./vendor-BtP0CW_r.js";import{u as fs,D as _s,a as ps}from"./useSidebarState-CJJaSDFr.js";import{T as ms}from"./TreeDetailModal-PvfFs-NG.js";import"./DarkModeToggle-ZfYhBz0e.js";import"./IconButton-DmJ5sToe.js";import"./useKeyboardNavigation-DT9lybaQ.js";const Ct=new cs,gs={getAll:(x={})=>Ct.get("/trees",x),getById:x=>Ct.get(`/trees/${x}`),getMeasurements:x=>Ct.get(`/trees/${x}/measurements`),getByForest:x=>Ct.get(`/trees/forest/${x}`),create:x=>Ct.post("/trees",x),update:(x,w)=>Ct.put(`/trees/${x}`,w),delete:x=>Ct.delete(`/trees/${x}`),addMeasurement:(x,w)=>Ct.post(`/trees/${x}/measurements`,w),markDead:x=>Ct.patch(`/trees/${x}/mark-dead`)};function vs(x,w){const f=N.useRef(w);N.useEffect(function(){w!==f.current&&x.attributionControl!=null&&(f.current!=null&&x.attributionControl.removeAttribution(f.current),w!=null&&x.attributionControl.addAttribution(w)),f.current=w},[x,w])}const ys=1;function Ls(x){return Object.freeze({__version:ys,map:x})}const Bn=N.createContext(null),xs=Bn.Provider;function An(){const x=N.useContext(Bn);if(x==null)throw new Error("No context provided: useLeafletContext() can only be used in a descendant of ");return x}function ws(x){function w(f,T){const{instance:b}=x(f).current;return N.useImperativeHandle(T,()=>b),null}return N.forwardRef(w)}function Ps(x,w){const f=N.useRef();N.useEffect(function(){return w!=null&&x.instance.on(w),f.current=w,function(){f.current!=null&&x.instance.off(f.current),f.current=null}},[x,w])}function Nn(x,w){const f=x.pane??w.pane;return f?{...x,pane:f}:x}var re={exports:{}};/* @preserve + * Leaflet 1.9.4, a JS library for interactive maps. https://leafletjs.com + * (c) 2010-2023 Vladimir Agafonkin, (c) 2010-2011 CloudMade + */var bs=re.exports,Sn;function Cs(){return Sn||(Sn=1,function(x,w){(function(f,T){T(w)})(bs,function(f){var T="1.9.4";function b(t){var e,i,n,o;for(i=1,n=arguments.length;i"u"||!L||!L.Mixin)){t=U(t)?t:[t];for(var e=0;e0?Math.floor(t):Math.ceil(t)};Z.prototype={clone:function(){return new Z(this.x,this.y)},add:function(t){return this.clone()._add(E(t))},_add:function(t){return this.x+=t.x,this.y+=t.y,this},subtract:function(t){return this.clone()._subtract(E(t))},_subtract:function(t){return this.x-=t.x,this.y-=t.y,this},divideBy:function(t){return this.clone()._divideBy(t)},_divideBy:function(t){return this.x/=t,this.y/=t,this},multiplyBy:function(t){return this.clone()._multiplyBy(t)},_multiplyBy:function(t){return this.x*=t,this.y*=t,this},scaleBy:function(t){return new Z(this.x*t.x,this.y*t.y)},unscaleBy:function(t){return new Z(this.x/t.x,this.y/t.y)},round:function(){return this.clone()._round()},_round:function(){return this.x=Math.round(this.x),this.y=Math.round(this.y),this},floor:function(){return this.clone()._floor()},_floor:function(){return this.x=Math.floor(this.x),this.y=Math.floor(this.y),this},ceil:function(){return this.clone()._ceil()},_ceil:function(){return this.x=Math.ceil(this.x),this.y=Math.ceil(this.y),this},trunc:function(){return this.clone()._trunc()},_trunc:function(){return this.x=pi(this.x),this.y=pi(this.y),this},distanceTo:function(t){t=E(t);var e=t.x-this.x,i=t.y-this.y;return Math.sqrt(e*e+i*i)},equals:function(t){return t=E(t),t.x===this.x&&t.y===this.y},contains:function(t){return t=E(t),Math.abs(t.x)<=Math.abs(this.x)&&Math.abs(t.y)<=Math.abs(this.y)},toString:function(){return"Point("+v(this.x)+", "+v(this.y)+")"}};function E(t,e,i){return t instanceof Z?t:U(t)?new Z(t[0],t[1]):t==null?t:typeof t=="object"&&"x"in t&&"y"in t?new Z(t.x,t.y):new Z(t,e,i)}function V(t,e){if(t)for(var i=e?[t,e]:t,n=0,o=i.length;n=this.min.x&&i.x<=this.max.x&&e.y>=this.min.y&&i.y<=this.max.y},intersects:function(t){t=nt(t);var e=this.min,i=this.max,n=t.min,o=t.max,r=o.x>=e.x&&n.x<=i.x,u=o.y>=e.y&&n.y<=i.y;return r&&u},overlaps:function(t){t=nt(t);var e=this.min,i=this.max,n=t.min,o=t.max,r=o.x>e.x&&n.xe.y&&n.y=e.lat&&o.lat<=i.lat&&n.lng>=e.lng&&o.lng<=i.lng},intersects:function(t){t=$(t);var e=this._southWest,i=this._northEast,n=t.getSouthWest(),o=t.getNorthEast(),r=o.lat>=e.lat&&n.lat<=i.lat,u=o.lng>=e.lng&&n.lng<=i.lng;return r&&u},overlaps:function(t){t=$(t);var e=this._southWest,i=this._northEast,n=t.getSouthWest(),o=t.getNorthEast(),r=o.lat>e.lat&&n.late.lng&&n.lng1,Qn=function(){var t=!1;try{var e=Object.defineProperty({},"passive",{get:function(){t=!0}});window.addEventListener("testPassiveEventSupport",d,e),window.removeEventListener("testPassiveEventSupport",d,e)}catch{}return t}(),to=function(){return!!document.createElement("canvas").getContext}(),Be=!!(document.createElementNS&&gi("svg").createSVGRect),eo=!!Be&&function(){var t=document.createElement("div");return t.innerHTML="",(t.firstChild&&t.firstChild.namespaceURI)==="http://www.w3.org/2000/svg"}(),io=!Be&&function(){try{var t=document.createElement("div");t.innerHTML='';var e=t.firstChild;return e.style.behavior="url(#default#VML)",e&&typeof e.adj=="object"}catch{return!1}}(),no=navigator.platform.indexOf("Mac")===0,oo=navigator.platform.indexOf("Linux")===0;function pt(t){return navigator.userAgent.toLowerCase().indexOf(t)>=0}var k={ie:he,ielt9:Fn,edge:yi,webkit:Ee,android:Li,android23:xi,androidStock:jn,opera:Ze,chrome:wi,gecko:Pi,safari:Un,phantom:bi,opera12:Ci,win:Wn,ie3d:Ti,webkit3d:Ie,gecko3d:Mi,any3d:Vn,mobile:qt,mobileWebkit:qn,mobileWebkit3d:Kn,msPointer:Si,pointer:ki,touch:$n,touchNative:Oi,mobileOpera:Yn,mobileGecko:Xn,retina:Jn,passiveEvents:Qn,canvas:to,svg:Be,vml:io,inlineSvg:eo,mac:no,linux:oo},zi=k.msPointer?"MSPointerDown":"pointerdown",Ei=k.msPointer?"MSPointerMove":"pointermove",Zi=k.msPointer?"MSPointerUp":"pointerup",Ii=k.msPointer?"MSPointerCancel":"pointercancel",Ae={touchstart:zi,touchmove:Ei,touchend:Zi,touchcancel:Ii},Bi={touchstart:lo,touchmove:ue,touchend:ue,touchcancel:ue},At={},Ai=!1;function so(t,e,i){return e==="touchstart"&&uo(),Bi[e]?(i=Bi[e].bind(this,i),t.addEventListener(Ae[e],i,!1),i):(console.warn("wrong event specified:",e),d)}function ro(t,e,i){if(!Ae[e]){console.warn("wrong event specified:",e);return}t.removeEventListener(Ae[e],i,!1)}function ao(t){At[t.pointerId]=t}function ho(t){At[t.pointerId]&&(At[t.pointerId]=t)}function Ni(t){delete At[t.pointerId]}function uo(){Ai||(document.addEventListener(zi,ao,!0),document.addEventListener(Ei,ho,!0),document.addEventListener(Zi,Ni,!0),document.addEventListener(Ii,Ni,!0),Ai=!0)}function ue(t,e){if(e.pointerType!==(e.MSPOINTER_TYPE_MOUSE||"mouse")){e.touches=[];for(var i in At)e.touches.push(At[i]);e.changedTouches=[e],t(e)}}function lo(t,e){e.MSPOINTER_TYPE_TOUCH&&e.pointerType===e.MSPOINTER_TYPE_TOUCH&&Q(e),ue(t,e)}function co(t){var e={},i,n;for(n in t)i=t[n],e[n]=i&&i.bind?i.bind(t):i;return t=e,e.type="dblclick",e.detail=2,e.isTrusted=!1,e._simulated=!0,e}var fo=200;function _o(t,e){t.addEventListener("dblclick",e);var i=0,n;function o(r){if(r.detail!==1){n=r.detail;return}if(!(r.pointerType==="mouse"||r.sourceCapabilities&&!r.sourceCapabilities.firesTouchEvents)){var u=Gi(r);if(!(u.some(function(p){return p instanceof HTMLLabelElement&&p.attributes.for})&&!u.some(function(p){return p instanceof HTMLInputElement||p instanceof HTMLSelectElement}))){var c=Date.now();c-i<=fo?(n++,n===2&&e(co(r))):n=1,i=c}}}return t.addEventListener("click",o),{dblclick:e,simDblclick:o}}function po(t,e){t.removeEventListener("dblclick",e.dblclick),t.removeEventListener("click",e.simDblclick)}var Ne=de(["transform","webkitTransform","OTransform","MozTransform","msTransform"]),Kt=de(["webkitTransition","transition","OTransition","MozTransition","msTransition"]),Ri=Kt==="webkitTransition"||Kt==="OTransition"?Kt+"End":"transitionend";function Di(t){return typeof t=="string"?document.getElementById(t):t}function $t(t,e){var i=t.style[e]||t.currentStyle&&t.currentStyle[e];if((!i||i==="auto")&&document.defaultView){var n=document.defaultView.getComputedStyle(t,null);i=n?n[e]:null}return i==="auto"?null:i}function G(t,e,i){var n=document.createElement(t);return n.className=e||"",i&&i.appendChild(n),n}function q(t){var e=t.parentNode;e&&e.removeChild(t)}function le(t){for(;t.firstChild;)t.removeChild(t.firstChild)}function Nt(t){var e=t.parentNode;e&&e.lastChild!==t&&e.appendChild(t)}function Rt(t){var e=t.parentNode;e&&e.firstChild!==t&&e.insertBefore(t,e.firstChild)}function Re(t,e){if(t.classList!==void 0)return t.classList.contains(e);var i=ce(t);return i.length>0&&new RegExp("(^|\\s)"+e+"(\\s|$)").test(i)}function A(t,e){if(t.classList!==void 0)for(var i=C(e),n=0,o=i.length;n0?2*window.devicePixelRatio:1;function Ui(t){return k.edge?t.wheelDeltaY/2:t.deltaY&&t.deltaMode===0?-t.deltaY/vo:t.deltaY&&t.deltaMode===1?-t.deltaY*20:t.deltaY&&t.deltaMode===2?-t.deltaY*60:t.deltaX||t.deltaZ?0:t.wheelDelta?(t.wheelDeltaY||t.wheelDelta)/2:t.detail&&Math.abs(t.detail)<32765?-t.detail*20:t.detail?t.detail/-32765*60:0}function Ye(t,e){var i=e.relatedTarget;if(!i)return!0;try{for(;i&&i!==t;)i=i.parentNode}catch{return!1}return i!==t}var yo={__proto__:null,on:I,off:W,stopPropagation:Zt,disableScrollPropagation:$e,disableClickPropagation:Qt,preventDefault:Q,stop:It,getPropagationPath:Gi,getMousePosition:ji,getWheelDelta:Ui,isExternalTarget:Ye,addListener:I,removeListener:W},Wi=Wt.extend({run:function(t,e,i,n){this.stop(),this._el=t,this._inProgress=!0,this._duration=i||.25,this._easeOutPower=1/Math.max(n||.5,.2),this._startPos=Et(t),this._offset=e.subtract(this._startPos),this._startTime=+new Date,this.fire("start"),this._animate()},stop:function(){this._inProgress&&(this._step(!0),this._complete())},_animate:function(){this._animId=it(this._animate,this),this._step()},_step:function(t){var e=+new Date-this._startTime,i=this._duration*1e3;ethis.options.maxZoom)?this.setZoom(t):this},panInsideBounds:function(t,e){this._enforcingBounds=!0;var i=this.getCenter(),n=this._limitCenter(i,this._zoom,$(t));return i.equals(n)||this.panTo(n,e),this._enforcingBounds=!1,this},panInside:function(t,e){e=e||{};var i=E(e.paddingTopLeft||e.padding||[0,0]),n=E(e.paddingBottomRight||e.padding||[0,0]),o=this.project(this.getCenter()),r=this.project(t),u=this.getPixelBounds(),c=nt([u.min.add(i),u.max.subtract(n)]),p=c.getSize();if(!c.contains(r)){this._enforcingBounds=!0;var y=r.subtract(c.getCenter()),M=c.extend(r).getSize().subtract(p);o.x+=y.x<0?-M.x:M.x,o.y+=y.y<0?-M.y:M.y,this.panTo(this.unproject(o),e),this._enforcingBounds=!1}return this},invalidateSize:function(t){if(!this._loaded)return this;t=b({animate:!1,pan:!0},t===!0?{animate:!0}:t);var e=this.getSize();this._sizeChanged=!0,this._lastCenter=null;var i=this.getSize(),n=e.divideBy(2).round(),o=i.divideBy(2).round(),r=n.subtract(o);return!r.x&&!r.y?this:(t.animate&&t.pan?this.panBy(r):(t.pan&&this._rawPanBy(r),this.fire("move"),t.debounceMoveend?(clearTimeout(this._sizeTimer),this._sizeTimer=setTimeout(a(this.fire,this,"moveend"),200)):this.fire("moveend")),this.fire("resize",{oldSize:e,newSize:i}))},stop:function(){return this.setZoom(this._limitZoom(this._zoom)),this.options.zoomSnap||this.fire("viewreset"),this._stop()},locate:function(t){if(t=this._locateOptions=b({timeout:1e4,watch:!1},t),!("geolocation"in navigator))return this._handleGeolocationError({code:0,message:"Geolocation not supported."}),this;var e=a(this._handleGeolocationResponse,this),i=a(this._handleGeolocationError,this);return t.watch?this._locationWatchId=navigator.geolocation.watchPosition(e,i,t):navigator.geolocation.getCurrentPosition(e,i,t),this},stopLocate:function(){return navigator.geolocation&&navigator.geolocation.clearWatch&&navigator.geolocation.clearWatch(this._locationWatchId),this._locateOptions&&(this._locateOptions.setView=!1),this},_handleGeolocationError:function(t){if(this._container._leaflet_id){var e=t.code,i=t.message||(e===1?"permission denied":e===2?"position unavailable":"timeout");this._locateOptions.setView&&!this._loaded&&this.fitWorld(),this.fire("locationerror",{code:e,message:"Geolocation error: "+i+"."})}},_handleGeolocationResponse:function(t){if(this._container._leaflet_id){var e=t.coords.latitude,i=t.coords.longitude,n=new j(e,i),o=n.toBounds(t.coords.accuracy*2),r=this._locateOptions;if(r.setView){var u=this.getBoundsZoom(o);this.setView(n,r.maxZoom?Math.min(u,r.maxZoom):u)}var c={latlng:n,bounds:o,timestamp:t.timestamp};for(var p in t.coords)typeof t.coords[p]=="number"&&(c[p]=t.coords[p]);this.fire("locationfound",c)}},addHandler:function(t,e){if(!e)return this;var i=this[t]=new e(this);return this._handlers.push(i),this.options[t]&&i.enable(),this},remove:function(){if(this._initEvents(!0),this.options.maxBounds&&this.off("moveend",this._panInsideMaxBounds),this._containerId!==this._container._leaflet_id)throw new Error("Map container is being reused by another instance");try{delete this._container._leaflet_id,delete this._containerId}catch{this._container._leaflet_id=void 0,this._containerId=void 0}this._locationWatchId!==void 0&&this.stopLocate(),this._stop(),q(this._mapPane),this._clearControlPos&&this._clearControlPos(),this._resizeRequest&&(ht(this._resizeRequest),this._resizeRequest=null),this._clearHandlers(),this._loaded&&this.fire("unload");var t;for(t in this._layers)this._layers[t].remove();for(t in this._panes)q(this._panes[t]);return this._layers=[],this._panes=[],delete this._mapPane,delete this._renderer,this},createPane:function(t,e){var i="leaflet-pane"+(t?" leaflet-"+t.replace("Pane","")+"-pane":""),n=G("div",i,e||this._mapPane);return t&&(this._panes[t]=n),n},getCenter:function(){return this._checkIfLoaded(),this._lastCenter&&!this._moved()?this._lastCenter.clone():this.layerPointToLatLng(this._getCenterLayerPoint())},getZoom:function(){return this._zoom},getBounds:function(){var t=this.getPixelBounds(),e=this.unproject(t.getBottomLeft()),i=this.unproject(t.getTopRight());return new ot(e,i)},getMinZoom:function(){return this.options.minZoom===void 0?this._layersMinZoom||0:this.options.minZoom},getMaxZoom:function(){return this.options.maxZoom===void 0?this._layersMaxZoom===void 0?1/0:this._layersMaxZoom:this.options.maxZoom},getBoundsZoom:function(t,e,i){t=$(t),i=E(i||[0,0]);var n=this.getZoom()||0,o=this.getMinZoom(),r=this.getMaxZoom(),u=t.getNorthWest(),c=t.getSouthEast(),p=this.getSize().subtract(i),y=nt(this.project(c,n),this.project(u,n)).getSize(),M=k.any3d?this.options.zoomSnap:1,O=p.x/y.x,R=p.y/y.y,et=e?Math.max(O,R):Math.min(O,R);return n=this.getScaleZoom(et,n),M&&(n=Math.round(n/(M/100))*(M/100),n=e?Math.ceil(n/M)*M:Math.floor(n/M)*M),Math.max(o,Math.min(r,n))},getSize:function(){return(!this._size||this._sizeChanged)&&(this._size=new Z(this._container.clientWidth||0,this._container.clientHeight||0),this._sizeChanged=!1),this._size.clone()},getPixelBounds:function(t,e){var i=this._getTopLeftPoint(t,e);return new V(i,i.add(this.getSize()))},getPixelOrigin:function(){return this._checkIfLoaded(),this._pixelOrigin},getPixelWorldBounds:function(t){return this.options.crs.getProjectedBounds(t===void 0?this.getZoom():t)},getPane:function(t){return typeof t=="string"?this._panes[t]:t},getPanes:function(){return this._panes},getContainer:function(){return this._container},getZoomScale:function(t,e){var i=this.options.crs;return e=e===void 0?this._zoom:e,i.scale(t)/i.scale(e)},getScaleZoom:function(t,e){var i=this.options.crs;e=e===void 0?this._zoom:e;var n=i.zoom(t*i.scale(e));return isNaN(n)?1/0:n},project:function(t,e){return e=e===void 0?this._zoom:e,this.options.crs.latLngToPoint(D(t),e)},unproject:function(t,e){return e=e===void 0?this._zoom:e,this.options.crs.pointToLatLng(E(t),e)},layerPointToLatLng:function(t){var e=E(t).add(this.getPixelOrigin());return this.unproject(e)},latLngToLayerPoint:function(t){var e=this.project(D(t))._round();return e._subtract(this.getPixelOrigin())},wrapLatLng:function(t){return this.options.crs.wrapLatLng(D(t))},wrapLatLngBounds:function(t){return this.options.crs.wrapLatLngBounds($(t))},distance:function(t,e){return this.options.crs.distance(D(t),D(e))},containerPointToLayerPoint:function(t){return E(t).subtract(this._getMapPanePos())},layerPointToContainerPoint:function(t){return E(t).add(this._getMapPanePos())},containerPointToLatLng:function(t){var e=this.containerPointToLayerPoint(E(t));return this.layerPointToLatLng(e)},latLngToContainerPoint:function(t){return this.layerPointToContainerPoint(this.latLngToLayerPoint(D(t)))},mouseEventToContainerPoint:function(t){return ji(t,this._container)},mouseEventToLayerPoint:function(t){return this.containerPointToLayerPoint(this.mouseEventToContainerPoint(t))},mouseEventToLatLng:function(t){return this.layerPointToLatLng(this.mouseEventToLayerPoint(t))},_initContainer:function(t){var e=this._container=Di(t);if(e){if(e._leaflet_id)throw new Error("Map container is already initialized.")}else throw new Error("Map container not found.");I(e,"scroll",this._onScroll,this),this._containerId=l(e)},_initLayout:function(){var t=this._container;this._fadeAnimated=this.options.fadeAnimation&&k.any3d,A(t,"leaflet-container"+(k.touch?" leaflet-touch":"")+(k.retina?" leaflet-retina":"")+(k.ielt9?" leaflet-oldie":"")+(k.safari?" leaflet-safari":"")+(this._fadeAnimated?" leaflet-fade-anim":""));var e=$t(t,"position");e!=="absolute"&&e!=="relative"&&e!=="fixed"&&e!=="sticky"&&(t.style.position="relative"),this._initPanes(),this._initControlPos&&this._initControlPos()},_initPanes:function(){var t=this._panes={};this._paneRenderers={},this._mapPane=this.createPane("mapPane",this._container),Y(this._mapPane,new Z(0,0)),this.createPane("tilePane"),this.createPane("overlayPane"),this.createPane("shadowPane"),this.createPane("markerPane"),this.createPane("tooltipPane"),this.createPane("popupPane"),this.options.markerZoomAnimation||(A(t.markerPane,"leaflet-zoom-hide"),A(t.shadowPane,"leaflet-zoom-hide"))},_resetView:function(t,e,i){Y(this._mapPane,new Z(0,0));var n=!this._loaded;this._loaded=!0,e=this._limitZoom(e),this.fire("viewprereset");var o=this._zoom!==e;this._moveStart(o,i)._move(t,e)._moveEnd(o),this.fire("viewreset"),n&&this.fire("load")},_moveStart:function(t,e){return t&&this.fire("zoomstart"),e||this.fire("movestart"),this},_move:function(t,e,i,n){e===void 0&&(e=this._zoom);var o=this._zoom!==e;return this._zoom=e,this._lastCenter=t,this._pixelOrigin=this._getNewPixelOrigin(t),n?i&&i.pinch&&this.fire("zoom",i):((o||i&&i.pinch)&&this.fire("zoom",i),this.fire("move",i)),this},_moveEnd:function(t){return t&&this.fire("zoomend"),this.fire("moveend")},_stop:function(){return ht(this._flyToFrame),this._panAnim&&this._panAnim.stop(),this},_rawPanBy:function(t){Y(this._mapPane,this._getMapPanePos().subtract(t))},_getZoomSpan:function(){return this.getMaxZoom()-this.getMinZoom()},_panInsideMaxBounds:function(){this._enforcingBounds||this.panInsideBounds(this.options.maxBounds)},_checkIfLoaded:function(){if(!this._loaded)throw new Error("Set map center and zoom first.")},_initEvents:function(t){this._targets={},this._targets[l(this._container)]=this;var e=t?W:I;e(this._container,"click dblclick mousedown mouseup mouseover mouseout mousemove contextmenu keypress keydown keyup",this._handleDOMEvent,this),this.options.trackResize&&e(window,"resize",this._onResize,this),k.any3d&&this.options.transform3DLimit&&(t?this.off:this.on).call(this,"moveend",this._onMoveEnd)},_onResize:function(){ht(this._resizeRequest),this._resizeRequest=it(function(){this.invalidateSize({debounceMoveend:!0})},this)},_onScroll:function(){this._container.scrollTop=0,this._container.scrollLeft=0},_onMoveEnd:function(){var t=this._getMapPanePos();Math.max(Math.abs(t.x),Math.abs(t.y))>=this.options.transform3DLimit&&this._resetView(this.getCenter(),this.getZoom())},_findEventTargets:function(t,e){for(var i=[],n,o=e==="mouseout"||e==="mouseover",r=t.target||t.srcElement,u=!1;r;){if(n=this._targets[l(r)],n&&(e==="click"||e==="preclick")&&this._draggableMoved(n)){u=!0;break}if(n&&n.listens(e,!0)&&(o&&!Ye(r,t)||(i.push(n),o))||r===this._container)break;r=r.parentNode}return!i.length&&!u&&!o&&this.listens(e,!0)&&(i=[this]),i},_isClickDisabled:function(t){for(;t&&t!==this._container;){if(t._leaflet_disable_click)return!0;t=t.parentNode}},_handleDOMEvent:function(t){var e=t.target||t.srcElement;if(!(!this._loaded||e._leaflet_disable_events||t.type==="click"&&this._isClickDisabled(e))){var i=t.type;i==="mousedown"&&Ue(e),this._fireDOMEvent(t,i)}},_mouseEvents:["click","dblclick","mouseover","mouseout","contextmenu"],_fireDOMEvent:function(t,e,i){if(t.type==="click"){var n=b({},t);n.type="preclick",this._fireDOMEvent(n,n.type,i)}var o=this._findEventTargets(t,e);if(i){for(var r=[],u=0;u0?Math.round(t-e)/2:Math.max(0,Math.ceil(t))-Math.max(0,Math.floor(e))},_limitZoom:function(t){var e=this.getMinZoom(),i=this.getMaxZoom(),n=k.any3d?this.options.zoomSnap:1;return n&&(t=Math.round(t/n)*n),Math.max(e,Math.min(i,t))},_onPanTransitionStep:function(){this.fire("move")},_onPanTransitionEnd:function(){K(this._mapPane,"leaflet-pan-anim"),this.fire("moveend")},_tryAnimatedPan:function(t,e){var i=this._getCenterOffset(t)._trunc();return(e&&e.animate)!==!0&&!this.getSize().contains(i)?!1:(this.panBy(i,e),!0)},_createAnimProxy:function(){var t=this._proxy=G("div","leaflet-proxy leaflet-zoom-animated");this._panes.mapPane.appendChild(t),this.on("zoomanim",function(e){var i=Ne,n=this._proxy.style[i];zt(this._proxy,this.project(e.center,e.zoom),this.getZoomScale(e.zoom,1)),n===this._proxy.style[i]&&this._animatingZoom&&this._onZoomTransitionEnd()},this),this.on("load moveend",this._animMoveEnd,this),this._on("unload",this._destroyAnimProxy,this)},_destroyAnimProxy:function(){q(this._proxy),this.off("load moveend",this._animMoveEnd,this),delete this._proxy},_animMoveEnd:function(){var t=this.getCenter(),e=this.getZoom();zt(this._proxy,this.project(t,e),this.getZoomScale(e,1))},_catchTransitionEnd:function(t){this._animatingZoom&&t.propertyName.indexOf("transform")>=0&&this._onZoomTransitionEnd()},_nothingToAnimate:function(){return!this._container.getElementsByClassName("leaflet-zoom-animated").length},_tryAnimatedZoom:function(t,e,i){if(this._animatingZoom)return!0;if(i=i||{},!this._zoomAnimated||i.animate===!1||this._nothingToAnimate()||Math.abs(e-this._zoom)>this.options.zoomAnimationThreshold)return!1;var n=this.getZoomScale(e),o=this._getCenterOffset(t)._divideBy(1-1/n);return i.animate!==!0&&!this.getSize().contains(o)?!1:(it(function(){this._moveStart(!0,i.noMoveStart||!1)._animateZoom(t,e,!0)},this),!0)},_animateZoom:function(t,e,i,n){this._mapPane&&(i&&(this._animatingZoom=!0,this._animateToCenter=t,this._animateToZoom=e,A(this._mapPane,"leaflet-zoom-anim")),this.fire("zoomanim",{center:t,zoom:e,noUpdate:n}),this._tempFireZoomEvent||(this._tempFireZoomEvent=this._zoom!==this._animateToZoom),this._move(this._animateToCenter,this._animateToZoom,void 0,!0),setTimeout(a(this._onZoomTransitionEnd,this),250))},_onZoomTransitionEnd:function(){this._animatingZoom&&(this._mapPane&&K(this._mapPane,"leaflet-zoom-anim"),this._animatingZoom=!1,this._move(this._animateToCenter,this._animateToZoom,void 0,!0),this._tempFireZoomEvent&&this.fire("zoom"),delete this._tempFireZoomEvent,this.fire("move"),this._moveEnd(!0))}});function Lo(t,e){return new H(t,e)}var ft=yt.extend({options:{position:"topright"},initialize:function(t){P(this,t)},getPosition:function(){return this.options.position},setPosition:function(t){var e=this._map;return e&&e.removeControl(this),this.options.position=t,e&&e.addControl(this),this},getContainer:function(){return this._container},addTo:function(t){this.remove(),this._map=t;var e=this._container=this.onAdd(t),i=this.getPosition(),n=t._controlCorners[i];return A(e,"leaflet-control"),i.indexOf("bottom")!==-1?n.insertBefore(e,n.firstChild):n.appendChild(e),this._map.on("unload",this.remove,this),this},remove:function(){return this._map?(q(this._container),this.onRemove&&this.onRemove(this._map),this._map.off("unload",this.remove,this),this._map=null,this):this},_refocusOnMap:function(t){this._map&&t&&t.screenX>0&&t.screenY>0&&this._map.getContainer().focus()}}),te=function(t){return new ft(t)};H.include({addControl:function(t){return t.addTo(this),this},removeControl:function(t){return t.remove(),this},_initControlPos:function(){var t=this._controlCorners={},e="leaflet-",i=this._controlContainer=G("div",e+"control-container",this._container);function n(o,r){var u=e+o+" "+e+r;t[o+r]=G("div",u,i)}n("top","left"),n("top","right"),n("bottom","left"),n("bottom","right")},_clearControlPos:function(){for(var t in this._controlCorners)q(this._controlCorners[t]);q(this._controlContainer),delete this._controlCorners,delete this._controlContainer}});var Vi=ft.extend({options:{collapsed:!0,position:"topright",autoZIndex:!0,hideSingleBase:!1,sortLayers:!1,sortFunction:function(t,e,i,n){return i1,this._baseLayersList.style.display=t?"":"none"),this._separator.style.display=e&&t?"":"none",this},_onLayerChange:function(t){this._handlingClick||this._update();var e=this._getLayer(l(t.target)),i=e.overlay?t.type==="add"?"overlayadd":"overlayremove":t.type==="add"?"baselayerchange":null;i&&this._map.fire(i,e)},_createRadioElement:function(t,e){var i='",n=document.createElement("div");return n.innerHTML=i,n.firstChild},_addItem:function(t){var e=document.createElement("label"),i=this._map.hasLayer(t.layer),n;t.overlay?(n=document.createElement("input"),n.type="checkbox",n.className="leaflet-control-layers-selector",n.defaultChecked=i):n=this._createRadioElement("leaflet-base-layers_"+l(this),i),this._layerControlInputs.push(n),n.layerId=l(t.layer),I(n,"click",this._onInputClick,this);var o=document.createElement("span");o.innerHTML=" "+t.name;var r=document.createElement("span");e.appendChild(r),r.appendChild(n),r.appendChild(o);var u=t.overlay?this._overlaysList:this._baseLayersList;return u.appendChild(e),this._checkDisabledLayers(),e},_onInputClick:function(){if(!this._preventClick){var t=this._layerControlInputs,e,i,n=[],o=[];this._handlingClick=!0;for(var r=t.length-1;r>=0;r--)e=t[r],i=this._getLayer(e.layerId).layer,e.checked?n.push(i):e.checked||o.push(i);for(r=0;r=0;o--)e=t[o],i=this._getLayer(e.layerId).layer,e.disabled=i.options.minZoom!==void 0&&ni.options.maxZoom},_expandIfNotCollapsed:function(){return this._map&&!this.options.collapsed&&this.expand(),this},_expandSafely:function(){var t=this._section;this._preventClick=!0,I(t,"click",Q),this.expand();var e=this;setTimeout(function(){W(t,"click",Q),e._preventClick=!1})}}),xo=function(t,e,i){return new Vi(t,e,i)},Xe=ft.extend({options:{position:"topleft",zoomInText:'',zoomInTitle:"Zoom in",zoomOutText:'',zoomOutTitle:"Zoom out"},onAdd:function(t){var e="leaflet-control-zoom",i=G("div",e+" leaflet-bar"),n=this.options;return this._zoomInButton=this._createButton(n.zoomInText,n.zoomInTitle,e+"-in",i,this._zoomIn),this._zoomOutButton=this._createButton(n.zoomOutText,n.zoomOutTitle,e+"-out",i,this._zoomOut),this._updateDisabled(),t.on("zoomend zoomlevelschange",this._updateDisabled,this),i},onRemove:function(t){t.off("zoomend zoomlevelschange",this._updateDisabled,this)},disable:function(){return this._disabled=!0,this._updateDisabled(),this},enable:function(){return this._disabled=!1,this._updateDisabled(),this},_zoomIn:function(t){!this._disabled&&this._map._zoomthis._map.getMinZoom()&&this._map.zoomOut(this._map.options.zoomDelta*(t.shiftKey?3:1))},_createButton:function(t,e,i,n,o){var r=G("a",i,n);return r.innerHTML=t,r.href="#",r.title=e,r.setAttribute("role","button"),r.setAttribute("aria-label",e),Qt(r),I(r,"click",It),I(r,"click",o,this),I(r,"click",this._refocusOnMap,this),r},_updateDisabled:function(){var t=this._map,e="leaflet-disabled";K(this._zoomInButton,e),K(this._zoomOutButton,e),this._zoomInButton.setAttribute("aria-disabled","false"),this._zoomOutButton.setAttribute("aria-disabled","false"),(this._disabled||t._zoom===t.getMinZoom())&&(A(this._zoomOutButton,e),this._zoomOutButton.setAttribute("aria-disabled","true")),(this._disabled||t._zoom===t.getMaxZoom())&&(A(this._zoomInButton,e),this._zoomInButton.setAttribute("aria-disabled","true"))}});H.mergeOptions({zoomControl:!0}),H.addInitHook(function(){this.options.zoomControl&&(this.zoomControl=new Xe,this.addControl(this.zoomControl))});var wo=function(t){return new Xe(t)},qi=ft.extend({options:{position:"bottomleft",maxWidth:100,metric:!0,imperial:!0},onAdd:function(t){var e="leaflet-control-scale",i=G("div",e),n=this.options;return this._addScales(n,e+"-line",i),t.on(n.updateWhenIdle?"moveend":"move",this._update,this),t.whenReady(this._update,this),i},onRemove:function(t){t.off(this.options.updateWhenIdle?"moveend":"move",this._update,this)},_addScales:function(t,e,i){t.metric&&(this._mScale=G("div",e,i)),t.imperial&&(this._iScale=G("div",e,i))},_update:function(){var t=this._map,e=t.getSize().y/2,i=t.distance(t.containerPointToLatLng([0,e]),t.containerPointToLatLng([this.options.maxWidth,e]));this._updateScales(i)},_updateScales:function(t){this.options.metric&&t&&this._updateMetric(t),this.options.imperial&&t&&this._updateImperial(t)},_updateMetric:function(t){var e=this._getRoundNum(t),i=e<1e3?e+" m":e/1e3+" km";this._updateScale(this._mScale,i,e/t)},_updateImperial:function(t){var e=t*3.2808399,i,n,o;e>5280?(i=e/5280,n=this._getRoundNum(i),this._updateScale(this._iScale,n+" mi",n/i)):(o=this._getRoundNum(e),this._updateScale(this._iScale,o+" ft",o/e))},_updateScale:function(t,e,i){t.style.width=Math.round(this.options.maxWidth*i)+"px",t.innerHTML=e},_getRoundNum:function(t){var e=Math.pow(10,(Math.floor(t)+"").length-1),i=t/e;return i=i>=10?10:i>=5?5:i>=3?3:i>=2?2:1,e*i}}),Po=function(t){return new qi(t)},bo='',Je=ft.extend({options:{position:"bottomright",prefix:''+(k.inlineSvg?bo+" ":"")+"Leaflet"},initialize:function(t){P(this,t),this._attributions={}},onAdd:function(t){t.attributionControl=this,this._container=G("div","leaflet-control-attribution"),Qt(this._container);for(var e in t._layers)t._layers[e].getAttribution&&this.addAttribution(t._layers[e].getAttribution());return this._update(),t.on("layeradd",this._addAttribution,this),this._container},onRemove:function(t){t.off("layeradd",this._addAttribution,this)},_addAttribution:function(t){t.layer.getAttribution&&(this.addAttribution(t.layer.getAttribution()),t.layer.once("remove",function(){this.removeAttribution(t.layer.getAttribution())},this))},setPrefix:function(t){return this.options.prefix=t,this._update(),this},addAttribution:function(t){return t?(this._attributions[t]||(this._attributions[t]=0),this._attributions[t]++,this._update(),this):this},removeAttribution:function(t){return t?(this._attributions[t]&&(this._attributions[t]--,this._update()),this):this},_update:function(){if(this._map){var t=[];for(var e in this._attributions)this._attributions[e]&&t.push(e);var i=[];this.options.prefix&&i.push(this.options.prefix),t.length&&i.push(t.join(", ")),this._container.innerHTML=i.join(' ')}}});H.mergeOptions({attributionControl:!0}),H.addInitHook(function(){this.options.attributionControl&&new Je().addTo(this)});var Co=function(t){return new Je(t)};ft.Layers=Vi,ft.Zoom=Xe,ft.Scale=qi,ft.Attribution=Je,te.layers=xo,te.zoom=wo,te.scale=Po,te.attribution=Co;var gt=yt.extend({initialize:function(t){this._map=t},enable:function(){return this._enabled?this:(this._enabled=!0,this.addHooks(),this)},disable:function(){return this._enabled?(this._enabled=!1,this.removeHooks(),this):this},enabled:function(){return!!this._enabled}});gt.addTo=function(t,e){return t.addHandler(e,this),this};var To={Events:rt},Ki=k.touch?"touchstart mousedown":"mousedown",St=Wt.extend({options:{clickTolerance:3},initialize:function(t,e,i,n){P(this,n),this._element=t,this._dragStartTarget=e||t,this._preventOutline=i},enable:function(){this._enabled||(I(this._dragStartTarget,Ki,this._onDown,this),this._enabled=!0)},disable:function(){this._enabled&&(St._dragging===this&&this.finishDrag(!0),W(this._dragStartTarget,Ki,this._onDown,this),this._enabled=!1,this._moved=!1)},_onDown:function(t){if(this._enabled&&(this._moved=!1,!Re(this._element,"leaflet-zoom-anim"))){if(t.touches&&t.touches.length!==1){St._dragging===this&&this.finishDrag();return}if(!(St._dragging||t.shiftKey||t.which!==1&&t.button!==1&&!t.touches)&&(St._dragging=this,this._preventOutline&&Ue(this._element),Fe(),Yt(),!this._moving)){this.fire("down");var e=t.touches?t.touches[0]:t,i=Hi(this._element);this._startPoint=new Z(e.clientX,e.clientY),this._startPos=Et(this._element),this._parentScale=We(i);var n=t.type==="mousedown";I(document,n?"mousemove":"touchmove",this._onMove,this),I(document,n?"mouseup":"touchend touchcancel",this._onUp,this)}}},_onMove:function(t){if(this._enabled){if(t.touches&&t.touches.length>1){this._moved=!0;return}var e=t.touches&&t.touches.length===1?t.touches[0]:t,i=new Z(e.clientX,e.clientY)._subtract(this._startPoint);!i.x&&!i.y||Math.abs(i.x)+Math.abs(i.y)r&&(u=c,r=p);r>i&&(e[u]=1,ti(t,e,i,n,u),ti(t,e,i,u,o))}function Oo(t,e){for(var i=[t[0]],n=1,o=0,r=t.length;ne&&(i.push(t[n]),o=n);return oe.max.x&&(i|=2),t.ye.max.y&&(i|=8),i}function zo(t,e){var i=e.x-t.x,n=e.y-t.y;return i*i+n*n}function ee(t,e,i,n){var o=e.x,r=e.y,u=i.x-o,c=i.y-r,p=u*u+c*c,y;return p>0&&(y=((t.x-o)*u+(t.y-r)*c)/p,y>1?(o=i.x,r=i.y):y>0&&(o+=u*y,r+=c*y)),u=t.x-o,c=t.y-r,n?u*u+c*c:new Z(o,r)}function lt(t){return!U(t[0])||typeof t[0][0]!="object"&&typeof t[0][0]<"u"}function en(t){return console.warn("Deprecated use of _flat, please use L.LineUtil.isFlat instead."),lt(t)}function nn(t,e){var i,n,o,r,u,c,p,y;if(!t||t.length===0)throw new Error("latlngs not passed");lt(t)||(console.warn("latlngs are not flat! Only the first ring will be used"),t=t[0]);var M=D([0,0]),O=$(t),R=O.getNorthWest().distanceTo(O.getSouthWest())*O.getNorthEast().distanceTo(O.getNorthWest());R<1700&&(M=Qe(t));var et=t.length,J=[];for(i=0;in){p=(r-n)/o,y=[c.x-p*(c.x-u.x),c.y-p*(c.y-u.y)];break}var st=e.unproject(E(y));return D([st.lat+M.lat,st.lng+M.lng])}var Eo={__proto__:null,simplify:Xi,pointToSegmentDistance:Ji,closestPointOnSegment:So,clipSegment:tn,_getEdgeIntersection:pe,_getBitCode:Bt,_sqClosestPointOnSegment:ee,isFlat:lt,_flat:en,polylineCenter:nn},ei={project:function(t){return new Z(t.lng,t.lat)},unproject:function(t){return new j(t.y,t.x)},bounds:new V([-180,-90],[180,90])},ii={R:6378137,R_MINOR:6356752314245179e-9,bounds:new V([-2003750834279e-5,-1549657073972e-5],[2003750834279e-5,1876465623138e-5]),project:function(t){var e=Math.PI/180,i=this.R,n=t.lat*e,o=this.R_MINOR/i,r=Math.sqrt(1-o*o),u=r*Math.sin(n),c=Math.tan(Math.PI/4-n/2)/Math.pow((1-u)/(1+u),r/2);return n=-i*Math.log(Math.max(c,1e-10)),new Z(t.lng*e*i,n)},unproject:function(t){for(var e=180/Math.PI,i=this.R,n=this.R_MINOR/i,o=Math.sqrt(1-n*n),r=Math.exp(-t.y/i),u=Math.PI/2-2*Math.atan(r),c=0,p=.1,y;c<15&&Math.abs(p)>1e-7;c++)y=o*Math.sin(u),y=Math.pow((1-y)/(1+y),o/2),p=Math.PI/2-2*Math.atan(r*y)-u,u+=p;return new j(u*e,t.x*e/i)}},Zo={__proto__:null,LonLat:ei,Mercator:ii,SphericalMercator:Se},Io=b({},Mt,{code:"EPSG:3395",projection:ii,transformation:function(){var t=.5/(Math.PI*ii.R);return Vt(t,.5,-t,.5)}()}),on=b({},Mt,{code:"EPSG:4326",projection:ei,transformation:Vt(1/180,1,-1/180,.5)}),Bo=b({},Lt,{projection:ei,transformation:Vt(1,0,-1,0),scale:function(t){return Math.pow(2,t)},zoom:function(t){return Math.log(t)/Math.LN2},distance:function(t,e){var i=e.lng-t.lng,n=e.lat-t.lat;return Math.sqrt(i*i+n*n)},infinite:!0});Lt.Earth=Mt,Lt.EPSG3395=Io,Lt.EPSG3857=Oe,Lt.EPSG900913=Hn,Lt.EPSG4326=on,Lt.Simple=Bo;var _t=Wt.extend({options:{pane:"overlayPane",attribution:null,bubblingMouseEvents:!0},addTo:function(t){return t.addLayer(this),this},remove:function(){return this.removeFrom(this._map||this._mapToAdd)},removeFrom:function(t){return t&&t.removeLayer(this),this},getPane:function(t){return this._map.getPane(t?this.options[t]||t:this.options.pane)},addInteractiveTarget:function(t){return this._map._targets[l(t)]=this,this},removeInteractiveTarget:function(t){return delete this._map._targets[l(t)],this},getAttribution:function(){return this.options.attribution},_layerAdd:function(t){var e=t.target;if(e.hasLayer(this)){if(this._map=e,this._zoomAnimated=e._zoomAnimated,this.getEvents){var i=this.getEvents();e.on(i,this),this.once("remove",function(){e.off(i,this)},this)}this.onAdd(e),this.fire("add"),e.fire("layeradd",{layer:this})}}});H.include({addLayer:function(t){if(!t._layerAdd)throw new Error("The provided object is not a Layer.");var e=l(t);return this._layers[e]?this:(this._layers[e]=t,t._mapToAdd=this,t.beforeAdd&&t.beforeAdd(this),this.whenReady(t._layerAdd,t),this)},removeLayer:function(t){var e=l(t);return this._layers[e]?(this._loaded&&t.onRemove(this),delete this._layers[e],this._loaded&&(this.fire("layerremove",{layer:t}),t.fire("remove")),t._map=t._mapToAdd=null,this):this},hasLayer:function(t){return l(t)in this._layers},eachLayer:function(t,e){for(var i in this._layers)t.call(e,this._layers[i]);return this},_addLayers:function(t){t=t?U(t)?t:[t]:[];for(var e=0,i=t.length;ethis._layersMaxZoom&&this.setZoom(this._layersMaxZoom),this.options.minZoom===void 0&&this._layersMinZoom&&this.getZoom()=2&&e[0]instanceof j&&e[0].equals(e[i-1])&&e.pop(),e},_setLatLngs:function(t){wt.prototype._setLatLngs.call(this,t),lt(this._latlngs)&&(this._latlngs=[this._latlngs])},_defaultShape:function(){return lt(this._latlngs[0])?this._latlngs[0]:this._latlngs[0][0]},_clipPoints:function(){var t=this._renderer._bounds,e=this.options.weight,i=new Z(e,e);if(t=new V(t.min.subtract(i),t.max.add(i)),this._parts=[],!(!this._pxBounds||!this._pxBounds.intersects(t))){if(this.options.noClip){this._parts=this._rings;return}for(var n=0,o=this._rings.length,r;nt.y!=o.y>t.y&&t.x<(o.x-n.x)*(t.y-n.y)/(o.y-n.y)+n.x&&(e=!e);return e||wt.prototype._containsPoint.call(this,t,!0)}});function jo(t,e){return new Ft(t,e)}var Pt=xt.extend({initialize:function(t,e){P(this,e),this._layers={},t&&this.addData(t)},addData:function(t){var e=U(t)?t:t.features,i,n,o;if(e){for(i=0,n=e.length;i0&&o.push(o[0].slice()),o}function Gt(t,e){return t.feature?b({},t.feature,{geometry:e}):xe(e)}function xe(t){return t.type==="Feature"||t.type==="FeatureCollection"?t:{type:"Feature",properties:{},geometry:t}}var ri={toGeoJSON:function(t){return Gt(this,{type:"Point",coordinates:si(this.getLatLng(),t)})}};me.include(ri),ni.include(ri),ge.include(ri),wt.include({toGeoJSON:function(t){var e=!lt(this._latlngs),i=Le(this._latlngs,e?1:0,!1,t);return Gt(this,{type:(e?"Multi":"")+"LineString",coordinates:i})}}),Ft.include({toGeoJSON:function(t){var e=!lt(this._latlngs),i=e&&!lt(this._latlngs[0]),n=Le(this._latlngs,i?2:e?1:0,!0,t);return e||(n=[n]),Gt(this,{type:(i?"Multi":"")+"Polygon",coordinates:n})}}),Dt.include({toMultiPoint:function(t){var e=[];return this.eachLayer(function(i){e.push(i.toGeoJSON(t).geometry.coordinates)}),Gt(this,{type:"MultiPoint",coordinates:e})},toGeoJSON:function(t){var e=this.feature&&this.feature.geometry&&this.feature.geometry.type;if(e==="MultiPoint")return this.toMultiPoint(t);var i=e==="GeometryCollection",n=[];return this.eachLayer(function(o){if(o.toGeoJSON){var r=o.toGeoJSON(t);if(i)n.push(r.geometry);else{var u=xe(r);u.type==="FeatureCollection"?n.push.apply(n,u.features):n.push(u)}}}),i?Gt(this,{geometries:n,type:"GeometryCollection"}):{type:"FeatureCollection",features:n}}});function an(t,e){return new Pt(t,e)}var Uo=an,we=_t.extend({options:{opacity:1,alt:"",interactive:!1,crossOrigin:!1,errorOverlayUrl:"",zIndex:1,className:""},initialize:function(t,e,i){this._url=t,this._bounds=$(e),P(this,i)},onAdd:function(){this._image||(this._initImage(),this.options.opacity<1&&this._updateOpacity()),this.options.interactive&&(A(this._image,"leaflet-interactive"),this.addInteractiveTarget(this._image)),this.getPane().appendChild(this._image),this._reset()},onRemove:function(){q(this._image),this.options.interactive&&this.removeInteractiveTarget(this._image)},setOpacity:function(t){return this.options.opacity=t,this._image&&this._updateOpacity(),this},setStyle:function(t){return t.opacity&&this.setOpacity(t.opacity),this},bringToFront:function(){return this._map&&Nt(this._image),this},bringToBack:function(){return this._map&&Rt(this._image),this},setUrl:function(t){return this._url=t,this._image&&(this._image.src=t),this},setBounds:function(t){return this._bounds=$(t),this._map&&this._reset(),this},getEvents:function(){var t={zoom:this._reset,viewreset:this._reset};return this._zoomAnimated&&(t.zoomanim=this._animateZoom),t},setZIndex:function(t){return this.options.zIndex=t,this._updateZIndex(),this},getBounds:function(){return this._bounds},getElement:function(){return this._image},_initImage:function(){var t=this._url.tagName==="IMG",e=this._image=t?this._url:G("img");if(A(e,"leaflet-image-layer"),this._zoomAnimated&&A(e,"leaflet-zoom-animated"),this.options.className&&A(e,this.options.className),e.onselectstart=d,e.onmousemove=d,e.onload=a(this.fire,this,"load"),e.onerror=a(this._overlayOnError,this,"error"),(this.options.crossOrigin||this.options.crossOrigin==="")&&(e.crossOrigin=this.options.crossOrigin===!0?"":this.options.crossOrigin),this.options.zIndex&&this._updateZIndex(),t){this._url=e.src;return}e.src=this._url,e.alt=this.options.alt},_animateZoom:function(t){var e=this._map.getZoomScale(t.zoom),i=this._map._latLngBoundsToNewLayerBounds(this._bounds,t.zoom,t.center).min;zt(this._image,i,e)},_reset:function(){var t=this._image,e=new V(this._map.latLngToLayerPoint(this._bounds.getNorthWest()),this._map.latLngToLayerPoint(this._bounds.getSouthEast())),i=e.getSize();Y(t,e.min),t.style.width=i.x+"px",t.style.height=i.y+"px"},_updateOpacity:function(){ut(this._image,this.options.opacity)},_updateZIndex:function(){this._image&&this.options.zIndex!==void 0&&this.options.zIndex!==null&&(this._image.style.zIndex=this.options.zIndex)},_overlayOnError:function(){this.fire("error");var t=this.options.errorOverlayUrl;t&&this._url!==t&&(this._url=t,this._image.src=t)},getCenter:function(){return this._bounds.getCenter()}}),Wo=function(t,e,i){return new we(t,e,i)},hn=we.extend({options:{autoplay:!0,loop:!0,keepAspectRatio:!0,muted:!1,playsInline:!0},_initImage:function(){var t=this._url.tagName==="VIDEO",e=this._image=t?this._url:G("video");if(A(e,"leaflet-image-layer"),this._zoomAnimated&&A(e,"leaflet-zoom-animated"),this.options.className&&A(e,this.options.className),e.onselectstart=d,e.onmousemove=d,e.onloadeddata=a(this.fire,this,"load"),t){for(var i=e.getElementsByTagName("source"),n=[],o=0;o0?n:[e.src];return}U(this._url)||(this._url=[this._url]),!this.options.keepAspectRatio&&Object.prototype.hasOwnProperty.call(e.style,"objectFit")&&(e.style.objectFit="fill"),e.autoplay=!!this.options.autoplay,e.loop=!!this.options.loop,e.muted=!!this.options.muted,e.playsInline=!!this.options.playsInline;for(var r=0;ro?(e.height=o+"px",A(t,r)):K(t,r),this._containerWidth=this._container.offsetWidth},_animateZoom:function(t){var e=this._map._latLngToNewLayerPoint(this._latlng,t.zoom,t.center),i=this._getAnchor();Y(this._container,e.add(i))},_adjustPan:function(){if(this.options.autoPan){if(this._map._panAnim&&this._map._panAnim.stop(),this._autopanning){this._autopanning=!1;return}var t=this._map,e=parseInt($t(this._container,"marginBottom"),10)||0,i=this._container.offsetHeight+e,n=this._containerWidth,o=new Z(this._containerLeft,-i-this._containerBottom);o._add(Et(this._container));var r=t.layerPointToContainerPoint(o),u=E(this.options.autoPanPadding),c=E(this.options.autoPanPaddingTopLeft||u),p=E(this.options.autoPanPaddingBottomRight||u),y=t.getSize(),M=0,O=0;r.x+n+p.x>y.x&&(M=r.x+n-y.x+p.x),r.x-M-c.x<0&&(M=r.x-c.x),r.y+i+p.y>y.y&&(O=r.y+i-y.y+p.y),r.y-O-c.y<0&&(O=r.y-c.y),(M||O)&&(this.options.keepInView&&(this._autopanning=!0),t.fire("autopanstart").panBy([M,O]))}},_getAnchor:function(){return E(this._source&&this._source._getPopupAnchor?this._source._getPopupAnchor():[0,0])}}),Ko=function(t,e){return new Pe(t,e)};H.mergeOptions({closePopupOnClick:!0}),H.include({openPopup:function(t,e,i){return this._initOverlay(Pe,t,e,i).openOn(this),this},closePopup:function(t){return t=arguments.length?t:this._popup,t&&t.close(),this}}),_t.include({bindPopup:function(t,e){return this._popup=this._initOverlay(Pe,this._popup,t,e),this._popupHandlersAdded||(this.on({click:this._openPopup,keypress:this._onKeyPress,remove:this.closePopup,move:this._movePopup}),this._popupHandlersAdded=!0),this},unbindPopup:function(){return this._popup&&(this.off({click:this._openPopup,keypress:this._onKeyPress,remove:this.closePopup,move:this._movePopup}),this._popupHandlersAdded=!1,this._popup=null),this},openPopup:function(t){return this._popup&&(this instanceof xt||(this._popup._source=this),this._popup._prepareOpen(t||this._latlng)&&this._popup.openOn(this._map)),this},closePopup:function(){return this._popup&&this._popup.close(),this},togglePopup:function(){return this._popup&&this._popup.toggle(this),this},isPopupOpen:function(){return this._popup?this._popup.isOpen():!1},setPopupContent:function(t){return this._popup&&this._popup.setContent(t),this},getPopup:function(){return this._popup},_openPopup:function(t){if(!(!this._popup||!this._map)){It(t);var e=t.layer||t.target;if(this._popup._source===e&&!(e instanceof kt)){this._map.hasLayer(this._popup)?this.closePopup():this.openPopup(t.latlng);return}this._popup._source=e,this.openPopup(t.latlng)}},_movePopup:function(t){this._popup.setLatLng(t.latlng)},_onKeyPress:function(t){t.originalEvent.keyCode===13&&this._openPopup(t)}});var be=vt.extend({options:{pane:"tooltipPane",offset:[0,0],direction:"auto",permanent:!1,sticky:!1,opacity:.9},onAdd:function(t){vt.prototype.onAdd.call(this,t),this.setOpacity(this.options.opacity),t.fire("tooltipopen",{tooltip:this}),this._source&&(this.addEventParent(this._source),this._source.fire("tooltipopen",{tooltip:this},!0))},onRemove:function(t){vt.prototype.onRemove.call(this,t),t.fire("tooltipclose",{tooltip:this}),this._source&&(this.removeEventParent(this._source),this._source.fire("tooltipclose",{tooltip:this},!0))},getEvents:function(){var t=vt.prototype.getEvents.call(this);return this.options.permanent||(t.preclick=this.close),t},_initLayout:function(){var t="leaflet-tooltip",e=t+" "+(this.options.className||"")+" leaflet-zoom-"+(this._zoomAnimated?"animated":"hide");this._contentNode=this._container=G("div",e),this._container.setAttribute("role","tooltip"),this._container.setAttribute("id","leaflet-tooltip-"+l(this))},_updateLayout:function(){},_adjustPan:function(){},_setPosition:function(t){var e,i,n=this._map,o=this._container,r=n.latLngToContainerPoint(n.getCenter()),u=n.layerPointToContainerPoint(t),c=this.options.direction,p=o.offsetWidth,y=o.offsetHeight,M=E(this.options.offset),O=this._getAnchor();c==="top"?(e=p/2,i=y):c==="bottom"?(e=p/2,i=0):c==="center"?(e=p/2,i=y/2):c==="right"?(e=0,i=y/2):c==="left"?(e=p,i=y/2):u.xthis.options.maxZoom||in?this._retainParent(o,r,u,n):!1)},_retainChildren:function(t,e,i,n){for(var o=2*t;o<2*t+2;o++)for(var r=2*e;r<2*e+2;r++){var u=new Z(o,r);u.z=i+1;var c=this._tileCoordsToKey(u),p=this._tiles[c];if(p&&p.active){p.retain=!0;continue}else p&&p.loaded&&(p.retain=!0);i+1this.options.maxZoom||this.options.minZoom!==void 0&&o1){this._setView(t,i);return}for(var O=o.min.y;O<=o.max.y;O++)for(var R=o.min.x;R<=o.max.x;R++){var et=new Z(R,O);if(et.z=this._tileZoom,!!this._isValidTile(et)){var J=this._tiles[this._tileCoordsToKey(et)];J?J.current=!0:u.push(et)}}if(u.sort(function(st,Ut){return st.distanceTo(r)-Ut.distanceTo(r)}),u.length!==0){this._loading||(this._loading=!0,this.fire("loading"));var ct=document.createDocumentFragment();for(R=0;Ri.max.x)||!e.wrapLat&&(t.yi.max.y))return!1}if(!this.options.bounds)return!0;var n=this._tileCoordsToBounds(t);return $(this.options.bounds).overlaps(n)},_keyToBounds:function(t){return this._tileCoordsToBounds(this._keyToTileCoords(t))},_tileCoordsToNwSe:function(t){var e=this._map,i=this.getTileSize(),n=t.scaleBy(i),o=n.add(i),r=e.unproject(n,t.z),u=e.unproject(o,t.z);return[r,u]},_tileCoordsToBounds:function(t){var e=this._tileCoordsToNwSe(t),i=new ot(e[0],e[1]);return this.options.noWrap||(i=this._map.wrapLatLngBounds(i)),i},_tileCoordsToKey:function(t){return t.x+":"+t.y+":"+t.z},_keyToTileCoords:function(t){var e=t.split(":"),i=new Z(+e[0],+e[1]);return i.z=+e[2],i},_removeTile:function(t){var e=this._tiles[t];e&&(q(e.el),delete this._tiles[t],this.fire("tileunload",{tile:e.el,coords:this._keyToTileCoords(t)}))},_initTile:function(t){A(t,"leaflet-tile");var e=this.getTileSize();t.style.width=e.x+"px",t.style.height=e.y+"px",t.onselectstart=d,t.onmousemove=d,k.ielt9&&this.options.opacity<1&&ut(t,this.options.opacity)},_addTile:function(t,e){var i=this._getTilePos(t),n=this._tileCoordsToKey(t),o=this.createTile(this._wrapCoords(t),a(this._tileReady,this,t));this._initTile(o),this.createTile.length<2&&it(a(this._tileReady,this,t,null,o)),Y(o,i),this._tiles[n]={el:o,coords:t,current:!0},e.appendChild(o),this.fire("tileloadstart",{tile:o,coords:t})},_tileReady:function(t,e,i){e&&this.fire("tileerror",{error:e,tile:i,coords:t});var n=this._tileCoordsToKey(t);i=this._tiles[n],i&&(i.loaded=+new Date,this._map._fadeAnimated?(ut(i.el,0),ht(this._fadeFrame),this._fadeFrame=it(this._updateOpacity,this)):(i.active=!0,this._pruneTiles()),e||(A(i.el,"leaflet-tile-loaded"),this.fire("tileload",{tile:i.el,coords:t})),this._noTilesToLoad()&&(this._loading=!1,this.fire("load"),k.ielt9||!this._map._fadeAnimated?it(this._pruneTiles,this):setTimeout(a(this._pruneTiles,this),250)))},_getTilePos:function(t){return t.scaleBy(this.getTileSize()).subtract(this._level.origin)},_wrapCoords:function(t){var e=new Z(this._wrapX?g(t.x,this._wrapX):t.x,this._wrapY?g(t.y,this._wrapY):t.y);return e.z=t.z,e},_pxBoundsToTileRange:function(t){var e=this.getTileSize();return new V(t.min.unscaleBy(e).floor(),t.max.unscaleBy(e).ceil().subtract([1,1]))},_noTilesToLoad:function(){for(var t in this._tiles)if(!this._tiles[t].loaded)return!1;return!0}});function Xo(t){return new ne(t)}var jt=ne.extend({options:{minZoom:0,maxZoom:18,subdomains:"abc",errorTileUrl:"",zoomOffset:0,tms:!1,zoomReverse:!1,detectRetina:!1,crossOrigin:!1,referrerPolicy:!1},initialize:function(t,e){this._url=t,e=P(this,e),e.detectRetina&&k.retina&&e.maxZoom>0?(e.tileSize=Math.floor(e.tileSize/2),e.zoomReverse?(e.zoomOffset--,e.minZoom=Math.min(e.maxZoom,e.minZoom+1)):(e.zoomOffset++,e.maxZoom=Math.max(e.minZoom,e.maxZoom-1)),e.minZoom=Math.max(0,e.minZoom)):e.zoomReverse?e.minZoom=Math.min(e.maxZoom,e.minZoom):e.maxZoom=Math.max(e.minZoom,e.maxZoom),typeof e.subdomains=="string"&&(e.subdomains=e.subdomains.split("")),this.on("tileunload",this._onTileRemove)},setUrl:function(t,e){return this._url===t&&e===void 0&&(e=!0),this._url=t,e||this.redraw(),this},createTile:function(t,e){var i=document.createElement("img");return I(i,"load",a(this._tileOnLoad,this,e,i)),I(i,"error",a(this._tileOnError,this,e,i)),(this.options.crossOrigin||this.options.crossOrigin==="")&&(i.crossOrigin=this.options.crossOrigin===!0?"":this.options.crossOrigin),typeof this.options.referrerPolicy=="string"&&(i.referrerPolicy=this.options.referrerPolicy),i.alt="",i.src=this.getTileUrl(t),i},getTileUrl:function(t){var e={r:k.retina?"@2x":"",s:this._getSubdomain(t),x:t.x,y:t.y,z:this._getZoomForUrl()};if(this._map&&!this._map.options.crs.infinite){var i=this._globalTileRange.max.y-t.y;this.options.tms&&(e.y=i),e["-y"]=i}return B(this._url,b(e,this.options))},_tileOnLoad:function(t,e){k.ielt9?setTimeout(a(t,this,null,e),0):t(null,e)},_tileOnError:function(t,e,i){var n=this.options.errorTileUrl;n&&e.getAttribute("src")!==n&&(e.src=n),t(i,e)},_onTileRemove:function(t){t.tile.onload=null},_getZoomForUrl:function(){var t=this._tileZoom,e=this.options.maxZoom,i=this.options.zoomReverse,n=this.options.zoomOffset;return i&&(t=e-t),t+n},_getSubdomain:function(t){var e=Math.abs(t.x+t.y)%this.options.subdomains.length;return this.options.subdomains[e]},_abortLoading:function(){var t,e;for(t in this._tiles)if(this._tiles[t].coords.z!==this._tileZoom&&(e=this._tiles[t].el,e.onload=d,e.onerror=d,!e.complete)){e.src=tt;var i=this._tiles[t].coords;q(e),delete this._tiles[t],this.fire("tileabort",{tile:e,coords:i})}},_removeTile:function(t){var e=this._tiles[t];if(e)return e.el.setAttribute("src",tt),ne.prototype._removeTile.call(this,t)},_tileReady:function(t,e,i){if(!(!this._map||i&&i.getAttribute("src")===tt))return ne.prototype._tileReady.call(this,t,e,i)}});function cn(t,e){return new jt(t,e)}var dn=jt.extend({defaultWmsParams:{service:"WMS",request:"GetMap",layers:"",styles:"",format:"image/jpeg",transparent:!1,version:"1.1.1"},options:{crs:null,uppercase:!1},initialize:function(t,e){this._url=t;var i=b({},this.defaultWmsParams);for(var n in e)n in this.options||(i[n]=e[n]);e=P(this,e);var o=e.detectRetina&&k.retina?2:1,r=this.getTileSize();i.width=r.x*o,i.height=r.y*o,this.wmsParams=i},onAdd:function(t){this._crs=this.options.crs||t.options.crs,this._wmsVersion=parseFloat(this.wmsParams.version);var e=this._wmsVersion>=1.3?"crs":"srs";this.wmsParams[e]=this._crs.code,jt.prototype.onAdd.call(this,t)},getTileUrl:function(t){var e=this._tileCoordsToNwSe(t),i=this._crs,n=nt(i.project(e[0]),i.project(e[1])),o=n.min,r=n.max,u=(this._wmsVersion>=1.3&&this._crs===on?[o.y,o.x,r.y,r.x]:[o.x,o.y,r.x,r.y]).join(","),c=jt.prototype.getTileUrl.call(this,t);return c+F(this.wmsParams,c,this.options.uppercase)+(this.options.uppercase?"&BBOX=":"&bbox=")+u},setParams:function(t,e){return b(this.wmsParams,t),e||this.redraw(),this}});function Jo(t,e){return new dn(t,e)}jt.WMS=dn,cn.wms=Jo;var bt=_t.extend({options:{padding:.1},initialize:function(t){P(this,t),l(this),this._layers=this._layers||{}},onAdd:function(){this._container||(this._initContainer(),A(this._container,"leaflet-zoom-animated")),this.getPane().appendChild(this._container),this._update(),this.on("update",this._updatePaths,this)},onRemove:function(){this.off("update",this._updatePaths,this),this._destroyContainer()},getEvents:function(){var t={viewreset:this._reset,zoom:this._onZoom,moveend:this._update,zoomend:this._onZoomEnd};return this._zoomAnimated&&(t.zoomanim=this._onAnimZoom),t},_onAnimZoom:function(t){this._updateTransform(t.center,t.zoom)},_onZoom:function(){this._updateTransform(this._map.getCenter(),this._map.getZoom())},_updateTransform:function(t,e){var i=this._map.getZoomScale(e,this._zoom),n=this._map.getSize().multiplyBy(.5+this.options.padding),o=this._map.project(this._center,e),r=n.multiplyBy(-i).add(o).subtract(this._map._getNewPixelOrigin(t,e));k.any3d?zt(this._container,r,i):Y(this._container,r)},_reset:function(){this._update(),this._updateTransform(this._center,this._zoom);for(var t in this._layers)this._layers[t]._reset()},_onZoomEnd:function(){for(var t in this._layers)this._layers[t]._project()},_updatePaths:function(){for(var t in this._layers)this._layers[t]._update()},_update:function(){var t=this.options.padding,e=this._map.getSize(),i=this._map.containerPointToLayerPoint(e.multiplyBy(-t)).round();this._bounds=new V(i,i.add(e.multiplyBy(1+t*2)).round()),this._center=this._map.getCenter(),this._zoom=this._map.getZoom()}}),fn=bt.extend({options:{tolerance:0},getEvents:function(){var t=bt.prototype.getEvents.call(this);return t.viewprereset=this._onViewPreReset,t},_onViewPreReset:function(){this._postponeUpdatePaths=!0},onAdd:function(){bt.prototype.onAdd.call(this),this._draw()},_initContainer:function(){var t=this._container=document.createElement("canvas");I(t,"mousemove",this._onMouseMove,this),I(t,"click dblclick mousedown mouseup contextmenu",this._onClick,this),I(t,"mouseout",this._handleMouseOut,this),t._leaflet_disable_events=!0,this._ctx=t.getContext("2d")},_destroyContainer:function(){ht(this._redrawRequest),delete this._ctx,q(this._container),W(this._container),delete this._container},_updatePaths:function(){if(!this._postponeUpdatePaths){var t;this._redrawBounds=null;for(var e in this._layers)t=this._layers[e],t._update();this._redraw()}},_update:function(){if(!(this._map._animatingZoom&&this._bounds)){bt.prototype._update.call(this);var t=this._bounds,e=this._container,i=t.getSize(),n=k.retina?2:1;Y(e,t.min),e.width=n*i.x,e.height=n*i.y,e.style.width=i.x+"px",e.style.height=i.y+"px",k.retina&&this._ctx.scale(2,2),this._ctx.translate(-t.min.x,-t.min.y),this.fire("update")}},_reset:function(){bt.prototype._reset.call(this),this._postponeUpdatePaths&&(this._postponeUpdatePaths=!1,this._updatePaths())},_initPath:function(t){this._updateDashArray(t),this._layers[l(t)]=t;var e=t._order={layer:t,prev:this._drawLast,next:null};this._drawLast&&(this._drawLast.next=e),this._drawLast=e,this._drawFirst=this._drawFirst||this._drawLast},_addPath:function(t){this._requestRedraw(t)},_removePath:function(t){var e=t._order,i=e.next,n=e.prev;i?i.prev=n:this._drawLast=n,n?n.next=i:this._drawFirst=i,delete t._order,delete this._layers[l(t)],this._requestRedraw(t)},_updatePath:function(t){this._extendRedrawBounds(t),t._project(),t._update(),this._requestRedraw(t)},_updateStyle:function(t){this._updateDashArray(t),this._requestRedraw(t)},_updateDashArray:function(t){if(typeof t.options.dashArray=="string"){var e=t.options.dashArray.split(/[, ]+/),i=[],n,o;for(o=0;o')}}catch{}return function(t){return document.createElement("<"+t+' xmlns="urn:schemas-microsoft.com:vml" class="lvml">')}}(),Qo={_initContainer:function(){this._container=G("div","leaflet-vml-container")},_update:function(){this._map._animatingZoom||(bt.prototype._update.call(this),this.fire("update"))},_initPath:function(t){var e=t._container=oe("shape");A(e,"leaflet-vml-shape "+(this.options.className||"")),e.coordsize="1 1",t._path=oe("path"),e.appendChild(t._path),this._updateStyle(t),this._layers[l(t)]=t},_addPath:function(t){var e=t._container;this._container.appendChild(e),t.options.interactive&&t.addInteractiveTarget(e)},_removePath:function(t){var e=t._container;q(e),t.removeInteractiveTarget(e),delete this._layers[l(t)]},_updateStyle:function(t){var e=t._stroke,i=t._fill,n=t.options,o=t._container;o.stroked=!!n.stroke,o.filled=!!n.fill,n.stroke?(e||(e=t._stroke=oe("stroke")),o.appendChild(e),e.weight=n.weight+"px",e.color=n.color,e.opacity=n.opacity,n.dashArray?e.dashStyle=U(n.dashArray)?n.dashArray.join(" "):n.dashArray.replace(/( *, *)/g," "):e.dashStyle="",e.endcap=n.lineCap.replace("butt","flat"),e.joinstyle=n.lineJoin):e&&(o.removeChild(e),t._stroke=null),n.fill?(i||(i=t._fill=oe("fill")),o.appendChild(i),i.color=n.fillColor||n.color,i.opacity=n.fillOpacity):i&&(o.removeChild(i),t._fill=null)},_updateCircle:function(t){var e=t._point.round(),i=Math.round(t._radius),n=Math.round(t._radiusY||i);this._setPath(t,t._empty()?"M0 0":"AL "+e.x+","+e.y+" "+i+","+n+" 0,"+65535*360)},_setPath:function(t,e){t._path.v=e},_bringToFront:function(t){Nt(t._container)},_bringToBack:function(t){Rt(t._container)}},Ce=k.vml?oe:gi,se=bt.extend({_initContainer:function(){this._container=Ce("svg"),this._container.setAttribute("pointer-events","none"),this._rootGroup=Ce("g"),this._container.appendChild(this._rootGroup)},_destroyContainer:function(){q(this._container),W(this._container),delete this._container,delete this._rootGroup,delete this._svgSize},_update:function(){if(!(this._map._animatingZoom&&this._bounds)){bt.prototype._update.call(this);var t=this._bounds,e=t.getSize(),i=this._container;(!this._svgSize||!this._svgSize.equals(e))&&(this._svgSize=e,i.setAttribute("width",e.x),i.setAttribute("height",e.y)),Y(i,t.min),i.setAttribute("viewBox",[t.min.x,t.min.y,e.x,e.y].join(" ")),this.fire("update")}},_initPath:function(t){var e=t._path=Ce("path");t.options.className&&A(e,t.options.className),t.options.interactive&&A(e,"leaflet-interactive"),this._updateStyle(t),this._layers[l(t)]=t},_addPath:function(t){this._rootGroup||this._initContainer(),this._rootGroup.appendChild(t._path),t.addInteractiveTarget(t._path)},_removePath:function(t){q(t._path),t.removeInteractiveTarget(t._path),delete this._layers[l(t)]},_updatePath:function(t){t._project(),t._update()},_updateStyle:function(t){var e=t._path,i=t.options;e&&(i.stroke?(e.setAttribute("stroke",i.color),e.setAttribute("stroke-opacity",i.opacity),e.setAttribute("stroke-width",i.weight),e.setAttribute("stroke-linecap",i.lineCap),e.setAttribute("stroke-linejoin",i.lineJoin),i.dashArray?e.setAttribute("stroke-dasharray",i.dashArray):e.removeAttribute("stroke-dasharray"),i.dashOffset?e.setAttribute("stroke-dashoffset",i.dashOffset):e.removeAttribute("stroke-dashoffset")):e.setAttribute("stroke","none"),i.fill?(e.setAttribute("fill",i.fillColor||i.color),e.setAttribute("fill-opacity",i.fillOpacity),e.setAttribute("fill-rule",i.fillRule||"evenodd")):e.setAttribute("fill","none"))},_updatePoly:function(t,e){this._setPath(t,vi(t._parts,e))},_updateCircle:function(t){var e=t._point,i=Math.max(Math.round(t._radius),1),n=Math.max(Math.round(t._radiusY),1)||i,o="a"+i+","+n+" 0 1,0 ",r=t._empty()?"M0 0":"M"+(e.x-i)+","+e.y+o+i*2+",0 "+o+-i*2+",0 ";this._setPath(t,r)},_setPath:function(t,e){t._path.setAttribute("d",e)},_bringToFront:function(t){Nt(t._path)},_bringToBack:function(t){Rt(t._path)}});k.vml&&se.include(Qo);function pn(t){return k.svg||k.vml?new se(t):null}H.include({getRenderer:function(t){var e=t.options.renderer||this._getPaneRenderer(t.options.pane)||this.options.renderer||this._renderer;return e||(e=this._renderer=this._createRenderer()),this.hasLayer(e)||this.addLayer(e),e},_getPaneRenderer:function(t){if(t==="overlayPane"||t===void 0)return!1;var e=this._paneRenderers[t];return e===void 0&&(e=this._createRenderer({pane:t}),this._paneRenderers[t]=e),e},_createRenderer:function(t){return this.options.preferCanvas&&_n(t)||pn(t)}});var mn=Ft.extend({initialize:function(t,e){Ft.prototype.initialize.call(this,this._boundsToLatLngs(t),e)},setBounds:function(t){return this.setLatLngs(this._boundsToLatLngs(t))},_boundsToLatLngs:function(t){return t=$(t),[t.getSouthWest(),t.getNorthWest(),t.getNorthEast(),t.getSouthEast()]}});function ts(t,e){return new mn(t,e)}se.create=Ce,se.pointsToPath=vi,Pt.geometryToLayer=ve,Pt.coordsToLatLng=oi,Pt.coordsToLatLngs=ye,Pt.latLngToCoords=si,Pt.latLngsToCoords=Le,Pt.getFeature=Gt,Pt.asFeature=xe,H.mergeOptions({boxZoom:!0});var gn=gt.extend({initialize:function(t){this._map=t,this._container=t._container,this._pane=t._panes.overlayPane,this._resetStateTimeout=0,t.on("unload",this._destroy,this)},addHooks:function(){I(this._container,"mousedown",this._onMouseDown,this)},removeHooks:function(){W(this._container,"mousedown",this._onMouseDown,this)},moved:function(){return this._moved},_destroy:function(){q(this._pane),delete this._pane},_resetState:function(){this._resetStateTimeout=0,this._moved=!1},_clearDeferredResetState:function(){this._resetStateTimeout!==0&&(clearTimeout(this._resetStateTimeout),this._resetStateTimeout=0)},_onMouseDown:function(t){if(!t.shiftKey||t.which!==1&&t.button!==1)return!1;this._clearDeferredResetState(),this._resetState(),Yt(),Fe(),this._startPoint=this._map.mouseEventToContainerPoint(t),I(document,{contextmenu:It,mousemove:this._onMouseMove,mouseup:this._onMouseUp,keydown:this._onKeyDown},this)},_onMouseMove:function(t){this._moved||(this._moved=!0,this._box=G("div","leaflet-zoom-box",this._container),A(this._container,"leaflet-crosshair"),this._map.fire("boxzoomstart")),this._point=this._map.mouseEventToContainerPoint(t);var e=new V(this._point,this._startPoint),i=e.getSize();Y(this._box,e.min),this._box.style.width=i.x+"px",this._box.style.height=i.y+"px"},_finish:function(){this._moved&&(q(this._box),K(this._container,"leaflet-crosshair")),Xt(),Ge(),W(document,{contextmenu:It,mousemove:this._onMouseMove,mouseup:this._onMouseUp,keydown:this._onKeyDown},this)},_onMouseUp:function(t){if(!(t.which!==1&&t.button!==1)&&(this._finish(),!!this._moved)){this._clearDeferredResetState(),this._resetStateTimeout=setTimeout(a(this._resetState,this),0);var e=new ot(this._map.containerPointToLatLng(this._startPoint),this._map.containerPointToLatLng(this._point));this._map.fitBounds(e).fire("boxzoomend",{boxZoomBounds:e})}},_onKeyDown:function(t){t.keyCode===27&&(this._finish(),this._clearDeferredResetState(),this._resetState())}});H.addInitHook("addHandler","boxZoom",gn),H.mergeOptions({doubleClickZoom:!0});var vn=gt.extend({addHooks:function(){this._map.on("dblclick",this._onDoubleClick,this)},removeHooks:function(){this._map.off("dblclick",this._onDoubleClick,this)},_onDoubleClick:function(t){var e=this._map,i=e.getZoom(),n=e.options.zoomDelta,o=t.originalEvent.shiftKey?i-n:i+n;e.options.doubleClickZoom==="center"?e.setZoom(o):e.setZoomAround(t.containerPoint,o)}});H.addInitHook("addHandler","doubleClickZoom",vn),H.mergeOptions({dragging:!0,inertia:!0,inertiaDeceleration:3400,inertiaMaxSpeed:1/0,easeLinearity:.2,worldCopyJump:!1,maxBoundsViscosity:0});var yn=gt.extend({addHooks:function(){if(!this._draggable){var t=this._map;this._draggable=new St(t._mapPane,t._container),this._draggable.on({dragstart:this._onDragStart,drag:this._onDrag,dragend:this._onDragEnd},this),this._draggable.on("predrag",this._onPreDragLimit,this),t.options.worldCopyJump&&(this._draggable.on("predrag",this._onPreDragWrap,this),t.on("zoomend",this._onZoomEnd,this),t.whenReady(this._onZoomEnd,this))}A(this._map._container,"leaflet-grab leaflet-touch-drag"),this._draggable.enable(),this._positions=[],this._times=[]},removeHooks:function(){K(this._map._container,"leaflet-grab"),K(this._map._container,"leaflet-touch-drag"),this._draggable.disable()},moved:function(){return this._draggable&&this._draggable._moved},moving:function(){return this._draggable&&this._draggable._moving},_onDragStart:function(){var t=this._map;if(t._stop(),this._map.options.maxBounds&&this._map.options.maxBoundsViscosity){var e=$(this._map.options.maxBounds);this._offsetLimit=nt(this._map.latLngToContainerPoint(e.getNorthWest()).multiplyBy(-1),this._map.latLngToContainerPoint(e.getSouthEast()).multiplyBy(-1).add(this._map.getSize())),this._viscosity=Math.min(1,Math.max(0,this._map.options.maxBoundsViscosity))}else this._offsetLimit=null;t.fire("movestart").fire("dragstart"),t.options.inertia&&(this._positions=[],this._times=[])},_onDrag:function(t){if(this._map.options.inertia){var e=this._lastTime=+new Date,i=this._lastPos=this._draggable._absPos||this._draggable._newPos;this._positions.push(i),this._times.push(e),this._prunePositions(e)}this._map.fire("move",t).fire("drag",t)},_prunePositions:function(t){for(;this._positions.length>1&&t-this._times[0]>50;)this._positions.shift(),this._times.shift()},_onZoomEnd:function(){var t=this._map.getSize().divideBy(2),e=this._map.latLngToLayerPoint([0,0]);this._initialWorldOffset=e.subtract(t).x,this._worldWidth=this._map.getPixelWorldBounds().getSize().x},_viscousLimit:function(t,e){return t-(t-e)*this._viscosity},_onPreDragLimit:function(){if(!(!this._viscosity||!this._offsetLimit)){var t=this._draggable._newPos.subtract(this._draggable._startPos),e=this._offsetLimit;t.xe.max.x&&(t.x=this._viscousLimit(t.x,e.max.x)),t.y>e.max.y&&(t.y=this._viscousLimit(t.y,e.max.y)),this._draggable._newPos=this._draggable._startPos.add(t)}},_onPreDragWrap:function(){var t=this._worldWidth,e=Math.round(t/2),i=this._initialWorldOffset,n=this._draggable._newPos.x,o=(n-e+i)%t+e-i,r=(n+e+i)%t-e-i,u=Math.abs(o+i)0?r:-r))-e;this._delta=0,this._startTime=null,u&&(t.options.scrollWheelZoom==="center"?t.setZoom(e+u):t.setZoomAround(this._lastMousePos,e+u))}});H.addInitHook("addHandler","scrollWheelZoom",xn);var es=600;H.mergeOptions({tapHold:k.touchNative&&k.safari&&k.mobile,tapTolerance:15});var wn=gt.extend({addHooks:function(){I(this._map._container,"touchstart",this._onDown,this)},removeHooks:function(){W(this._map._container,"touchstart",this._onDown,this)},_onDown:function(t){if(clearTimeout(this._holdTimeout),t.touches.length===1){var e=t.touches[0];this._startPos=this._newPos=new Z(e.clientX,e.clientY),this._holdTimeout=setTimeout(a(function(){this._cancel(),this._isTapValid()&&(I(document,"touchend",Q),I(document,"touchend touchcancel",this._cancelClickPrevent),this._simulateEvent("contextmenu",e))},this),es),I(document,"touchend touchcancel contextmenu",this._cancel,this),I(document,"touchmove",this._onMove,this)}},_cancelClickPrevent:function t(){W(document,"touchend",Q),W(document,"touchend touchcancel",t)},_cancel:function(){clearTimeout(this._holdTimeout),W(document,"touchend touchcancel contextmenu",this._cancel,this),W(document,"touchmove",this._onMove,this)},_onMove:function(t){var e=t.touches[0];this._newPos=new Z(e.clientX,e.clientY)},_isTapValid:function(){return this._newPos.distanceTo(this._startPos)<=this._map.options.tapTolerance},_simulateEvent:function(t,e){var i=new MouseEvent(t,{bubbles:!0,cancelable:!0,view:window,screenX:e.screenX,screenY:e.screenY,clientX:e.clientX,clientY:e.clientY});i._simulated=!0,e.target.dispatchEvent(i)}});H.addInitHook("addHandler","tapHold",wn),H.mergeOptions({touchZoom:k.touch,bounceAtZoomLimits:!0});var Pn=gt.extend({addHooks:function(){A(this._map._container,"leaflet-touch-zoom"),I(this._map._container,"touchstart",this._onTouchStart,this)},removeHooks:function(){K(this._map._container,"leaflet-touch-zoom"),W(this._map._container,"touchstart",this._onTouchStart,this)},_onTouchStart:function(t){var e=this._map;if(!(!t.touches||t.touches.length!==2||e._animatingZoom||this._zooming)){var i=e.mouseEventToContainerPoint(t.touches[0]),n=e.mouseEventToContainerPoint(t.touches[1]);this._centerPoint=e.getSize()._divideBy(2),this._startLatLng=e.containerPointToLatLng(this._centerPoint),e.options.touchZoom!=="center"&&(this._pinchStartLatLng=e.containerPointToLatLng(i.add(n)._divideBy(2))),this._startDist=i.distanceTo(n),this._startZoom=e.getZoom(),this._moved=!1,this._zooming=!0,e._stop(),I(document,"touchmove",this._onTouchMove,this),I(document,"touchend touchcancel",this._onTouchEnd,this),Q(t)}},_onTouchMove:function(t){if(!(!t.touches||t.touches.length!==2||!this._zooming)){var e=this._map,i=e.mouseEventToContainerPoint(t.touches[0]),n=e.mouseEventToContainerPoint(t.touches[1]),o=i.distanceTo(n)/this._startDist;if(this._zoom=e.getScaleZoom(o,this._startZoom),!e.options.bounceAtZoomLimits&&(this._zoome.getMaxZoom()&&o>1)&&(this._zoom=e._limitZoom(this._zoom)),e.options.touchZoom==="center"){if(this._center=this._startLatLng,o===1)return}else{var r=i._add(n)._divideBy(2)._subtract(this._centerPoint);if(o===1&&r.x===0&&r.y===0)return;this._center=e.unproject(e.project(this._pinchStartLatLng,this._zoom).subtract(r),this._zoom)}this._moved||(e._moveStart(!0,!1),this._moved=!0),ht(this._animRequest);var u=a(e._move,e,this._center,this._zoom,{pinch:!0,round:!1},void 0);this._animRequest=it(u,this,!0),Q(t)}},_onTouchEnd:function(){if(!this._moved||!this._zooming){this._zooming=!1;return}this._zooming=!1,ht(this._animRequest),W(document,"touchmove",this._onTouchMove,this),W(document,"touchend touchcancel",this._onTouchEnd,this),this._map.options.zoomAnimation?this._map._animateZoom(this._center,this._map._limitZoom(this._zoom),!0,this._map.options.zoomSnap):this._map._resetView(this._center,this._map._limitZoom(this._zoom))}});H.addInitHook("addHandler","touchZoom",Pn),H.BoxZoom=gn,H.DoubleClickZoom=vn,H.Drag=yn,H.Keyboard=Ln,H.ScrollWheelZoom=xn,H.TapHold=wn,H.TouchZoom=Pn,f.Bounds=V,f.Browser=k,f.CRS=Lt,f.Canvas=fn,f.Circle=ni,f.CircleMarker=ge,f.Class=yt,f.Control=ft,f.DivIcon=ln,f.DivOverlay=vt,f.DomEvent=yo,f.DomUtil=go,f.Draggable=St,f.Evented=Wt,f.FeatureGroup=xt,f.GeoJSON=Pt,f.GridLayer=ne,f.Handler=gt,f.Icon=Ht,f.ImageOverlay=we,f.LatLng=j,f.LatLngBounds=ot,f.Layer=_t,f.LayerGroup=Dt,f.LineUtil=Eo,f.Map=H,f.Marker=me,f.Mixin=To,f.Path=kt,f.Point=Z,f.PolyUtil=Mo,f.Polygon=Ft,f.Polyline=wt,f.Popup=Pe,f.PosAnimation=Wi,f.Projection=Zo,f.Rectangle=mn,f.Renderer=bt,f.SVG=se,f.SVGOverlay=un,f.TileLayer=jt,f.Tooltip=be,f.Transformation=ke,f.Util=Rn,f.VideoOverlay=hn,f.bind=a,f.bounds=nt,f.canvas=_n,f.circle=Fo,f.circleMarker=Ho,f.control=te,f.divIcon=Yo,f.extend=b,f.featureGroup=No,f.geoJSON=an,f.geoJson=Uo,f.gridLayer=Xo,f.icon=Ro,f.imageOverlay=Wo,f.latLng=D,f.latLngBounds=$,f.layerGroup=Ao,f.map=Lo,f.marker=Do,f.point=E,f.polygon=jo,f.polyline=Go,f.popup=Ko,f.rectangle=ts,f.setOptions=P,f.stamp=l,f.svg=pn,f.svgOverlay=qo,f.tileLayer=cn,f.tooltip=$o,f.transformation=Vt,f.version=T,f.videoOverlay=Vo;var is=window.L;f.noConflict=function(){return window.L=is,this},window.L=f})}(re,re.exports)),re.exports}var li=Cs();const dt=ds(li);function Ts(x,w,f){return Object.freeze({instance:x,context:w,container:f})}function Ms(x,w){return w==null?function(T,b){const s=N.useRef();return s.current||(s.current=x(T,b)),s}:function(T,b){const s=N.useRef();s.current||(s.current=x(T,b));const a=N.useRef(T),{instance:h}=s.current;return N.useEffect(function(){a.current!==T&&(w(h,T,a.current),a.current=T)},[h,T,b]),s}}function Ss(x,w){N.useEffect(function(){return(w.layerContainer??w.map).addLayer(x.instance),function(){var s;(s=w.layerContainer)==null||s.removeLayer(x.instance),w.map.removeLayer(x.instance)}},[w,x])}function ks(x){return function(f){const T=An(),b=x(Nn(f,T),T);return vs(T.map,f.attribution),Ps(b.current,f.eventHandlers),Ss(b.current,T),b}}function Os(x,w){const f=Ms(x,w),T=ks(f);return ws(T)}function zs(x,w,f){const{opacity:T,zIndex:b}=w;T!=null&&T!==f.opacity&&x.setOpacity(T),b!=null&&b!==f.zIndex&&x.setZIndex(b)}function ci(){return An().map}function hi(){return hi=Object.assign||function(x){for(var w=1;w(m==null?void 0:m.map)??null,[m]);const P=N.useCallback(S=>{if(S!==null&&m===null){const B=new li.Map(S,g);f!=null&&_!=null?B.setView(f,_):x!=null&&B.fitBounds(x,w),l!=null&&B.whenReady(l),C(Ls(B))}},[]);N.useEffect(()=>()=>{m==null||m.map.remove()},[m]);const F=m?Mn.createElement(xs,{value:m},T):a??null;return Mn.createElement("div",hi({},v,{ref:P}),F)}const Zs=N.forwardRef(Es),Is=Os(function({url:w,...f},T){const b=new li.TileLayer(w,Nn(f,T));return Ts(b,T)},function(w,f,T){zs(w,f,T);const{url:b}=f;b!=null&&b!==T.url&&w.setUrl(b)});var ae={exports:{}},Bs=ae.exports,kn;function As(){return kn||(kn=1,function(x,w){(function(f,T){T(w)})(Bs,function(f){var T=L.MarkerClusterGroup=L.FeatureGroup.extend({options:{maxClusterRadius:80,iconCreateFunction:null,clusterPane:L.Marker.prototype.options.pane,spiderfyOnEveryZoom:!1,spiderfyOnMaxZoom:!0,showCoverageOnHover:!0,zoomToBoundsOnClick:!0,singleMarkerMode:!1,disableClusteringAtZoom:null,removeOutsideVisibleBounds:!0,animate:!0,animateAddingMarkers:!1,spiderfyShapePositions:null,spiderfyDistanceMultiplier:1,spiderLegPolylineOptions:{weight:1.5,color:"#222",opacity:.5},chunkedLoading:!1,chunkInterval:200,chunkDelay:50,chunkProgress:null,polygonOptions:{}},initialize:function(s){L.Util.setOptions(this,s),this.options.iconCreateFunction||(this.options.iconCreateFunction=this._defaultIconCreateFunction),this._featureGroup=L.featureGroup(),this._featureGroup.addEventParent(this),this._nonPointGroup=L.featureGroup(),this._nonPointGroup.addEventParent(this),this._inZoomAnimation=0,this._needsClustering=[],this._needsRemoving=[],this._currentShownBounds=null,this._queue=[],this._childMarkerEventHandlers={dragstart:this._childMarkerDragStart,move:this._childMarkerMoved,dragend:this._childMarkerDragEnd};var a=L.DomUtil.TRANSITION&&this.options.animate;L.extend(this,a?this._withAnimation:this._noAnimation),this._markerCluster=a?L.MarkerCluster:L.MarkerClusterNonAnimated},addLayer:function(s){if(s instanceof L.LayerGroup)return this.addLayers([s]);if(!s.getLatLng)return this._nonPointGroup.addLayer(s),this.fire("layeradd",{layer:s}),this;if(!this._map)return this._needsClustering.push(s),this.fire("layeradd",{layer:s}),this;if(this.hasLayer(s))return this;this._unspiderfy&&this._unspiderfy(),this._addLayer(s,this._maxZoom),this.fire("layeradd",{layer:s}),this._topClusterLevel._recalculateBounds(),this._refreshClustersIcons();var a=s,h=this._zoom;if(s.__parent)for(;a.__parent._zoom>=h;)a=a.__parent;return this._currentShownBounds.contains(a.getLatLng())&&(this.options.animateAddingMarkers?this._animationAddLayer(s,a):this._animationAddLayerNonAnimated(s,a)),this},removeLayer:function(s){return s instanceof L.LayerGroup?this.removeLayers([s]):s.getLatLng?this._map?s.__parent?(this._unspiderfy&&(this._unspiderfy(),this._unspiderfyLayer(s)),this._removeLayer(s,!0),this.fire("layerremove",{layer:s}),this._topClusterLevel._recalculateBounds(),this._refreshClustersIcons(),s.off(this._childMarkerEventHandlers,this),this._featureGroup.hasLayer(s)&&(this._featureGroup.removeLayer(s),s.clusterShow&&s.clusterShow()),this):this:(!this._arraySplice(this._needsClustering,s)&&this.hasLayer(s)&&this._needsRemoving.push({layer:s,latlng:s._latlng}),this.fire("layerremove",{layer:s}),this):(this._nonPointGroup.removeLayer(s),this.fire("layerremove",{layer:s}),this)},addLayers:function(s,a){if(!L.Util.isArray(s))return this.addLayer(s);var h=this._featureGroup,l=this._nonPointGroup,_=this.options.chunkedLoading,g=this.options.chunkInterval,d=this.options.chunkProgress,v=s.length,m=0,C=!0,P;if(this._map){var F=new Date().getTime(),S=L.bind(function(){var U=new Date().getTime();for(this._map&&this._unspiderfy&&this._unspiderfy();mg)break}if(P=s[m],P instanceof L.LayerGroup){C&&(s=s.slice(),C=!1),this._extractNonGroupLayers(P,s),v=s.length;continue}if(!P.getLatLng){l.addLayer(P),a||this.fire("layeradd",{layer:P});continue}if(!this.hasLayer(P)&&(this._addLayer(P,this._maxZoom),a||this.fire("layeradd",{layer:P}),P.__parent&&P.__parent.getChildCount()===2)){var tt=P.__parent.getAllChildMarkers(),Tt=tt[0]===P?tt[1]:tt[0];h.removeLayer(Tt)}}d&&d(m,v,new Date().getTime()-F),m===v?(this._topClusterLevel._recalculateBounds(),this._refreshClustersIcons(),this._topClusterLevel._recursivelyAddChildrenToMap(null,this._zoom,this._currentShownBounds)):setTimeout(S,this.options.chunkDelay)},this);S()}else for(var B=this._needsClustering;m=0;a--)s.extend(this._needsClustering[a].getLatLng());return s.extend(this._nonPointGroup.getBounds()),s},eachLayer:function(s,a){var h=this._needsClustering.slice(),l=this._needsRemoving,_,g,d;for(this._topClusterLevel&&this._topClusterLevel.getAllChildMarkers(h),g=h.length-1;g>=0;g--){for(_=!0,d=l.length-1;d>=0;d--)if(l[d].layer===h[g]){_=!1;break}_&&s.call(a,h[g])}this._nonPointGroup.eachLayer(s,a)},getLayers:function(){var s=[];return this.eachLayer(function(a){s.push(a)}),s},getLayer:function(s){var a=null;return s=parseInt(s,10),this.eachLayer(function(h){L.stamp(h)===s&&(a=h)}),a},hasLayer:function(s){if(!s)return!1;var a,h=this._needsClustering;for(a=h.length-1;a>=0;a--)if(h[a]===s)return!0;for(h=this._needsRemoving,a=h.length-1;a>=0;a--)if(h[a].layer===s)return!1;return!!(s.__parent&&s.__parent._group===this)||this._nonPointGroup.hasLayer(s)},zoomToShowLayer:function(s,a){var h=this._map;typeof a!="function"&&(a=function(){});var l=function(){(h.hasLayer(s)||h.hasLayer(s.__parent))&&!this._inZoomAnimation&&(this._map.off("moveend",l,this),this.off("animationend",l,this),h.hasLayer(s)?a():s.__parent._icon&&(this.once("spiderfied",a,this),s.__parent.spiderfy()))};s._icon&&this._map.getBounds().contains(s.getLatLng())?a():s.__parent._zoom=0;h--)if(s[h]===a)return s.splice(h,1),!0},_removeFromGridUnclustered:function(s,a){for(var h=this._map,l=this._gridUnclustered,_=Math.floor(this._map.getMinZoom());a>=_&&l[a].removeObject(s,h.project(s.getLatLng(),a));a--);},_childMarkerDragStart:function(s){s.target.__dragStart=s.target._latlng},_childMarkerMoved:function(s){if(!this._ignoreMove&&!s.target.__dragStart){var a=s.target._popup&&s.target._popup.isOpen();this._moveChild(s.target,s.oldLatLng,s.latlng),a&&s.target.openPopup()}},_moveChild:function(s,a,h){s._latlng=a,this.removeLayer(s),s._latlng=h,this.addLayer(s)},_childMarkerDragEnd:function(s){var a=s.target.__dragStart;delete s.target.__dragStart,a&&this._moveChild(s.target,a,s.target._latlng)},_removeLayer:function(s,a,h){var l=this._gridClusters,_=this._gridUnclustered,g=this._featureGroup,d=this._map,v=Math.floor(this._map.getMinZoom());a&&this._removeFromGridUnclustered(s,this._maxZoom);var m=s.__parent,C=m._markers,P;for(this._arraySplice(C,s);m&&(m._childCount--,m._boundsNeedUpdate=!0,!(m._zoom"+a+"",className:"marker-cluster"+h,iconSize:new L.Point(40,40)})},_bindEvents:function(){var s=this._map,a=this.options.spiderfyOnMaxZoom,h=this.options.showCoverageOnHover,l=this.options.zoomToBoundsOnClick,_=this.options.spiderfyOnEveryZoom;(a||l||_)&&this.on("clusterclick clusterkeypress",this._zoomOrSpiderfy,this),h&&(this.on("clustermouseover",this._showCoverage,this),this.on("clustermouseout",this._hideCoverage,this),s.on("zoomend",this._hideCoverage,this))},_zoomOrSpiderfy:function(s){var a=s.layer,h=a;if(!(s.type==="clusterkeypress"&&s.originalEvent&&s.originalEvent.keyCode!==13)){for(;h._childClusters.length===1;)h=h._childClusters[0];h._zoom===this._maxZoom&&h._childCount===a._childCount&&this.options.spiderfyOnMaxZoom?a.spiderfy():this.options.zoomToBoundsOnClick&&a.zoomToBounds(),this.options.spiderfyOnEveryZoom&&a.spiderfy(),s.originalEvent&&s.originalEvent.keyCode===13&&this._map._container.focus()}},_showCoverage:function(s){var a=this._map;this._inZoomAnimation||(this._shownPolygon&&a.removeLayer(this._shownPolygon),s.layer.getChildCount()>2&&s.layer!==this._spiderfied&&(this._shownPolygon=new L.Polygon(s.layer.getConvexHull(),this.options.polygonOptions),a.addLayer(this._shownPolygon)))},_hideCoverage:function(){this._shownPolygon&&(this._map.removeLayer(this._shownPolygon),this._shownPolygon=null)},_unbindEvents:function(){var s=this.options.spiderfyOnMaxZoom,a=this.options.showCoverageOnHover,h=this.options.zoomToBoundsOnClick,l=this.options.spiderfyOnEveryZoom,_=this._map;(s||h||l)&&this.off("clusterclick clusterkeypress",this._zoomOrSpiderfy,this),a&&(this.off("clustermouseover",this._showCoverage,this),this.off("clustermouseout",this._hideCoverage,this),_.off("zoomend",this._hideCoverage,this))},_zoomEnd:function(){this._map&&(this._mergeSplitClusters(),this._zoom=Math.round(this._map._zoom),this._currentShownBounds=this._getExpandedVisibleBounds())},_moveEnd:function(){if(!this._inZoomAnimation){var s=this._getExpandedVisibleBounds();this._topClusterLevel._recursivelyRemoveChildrenFromMap(this._currentShownBounds,Math.floor(this._map.getMinZoom()),this._zoom,s),this._topClusterLevel._recursivelyAddChildrenToMap(null,Math.round(this._map._zoom),s),this._currentShownBounds=s}},_generateInitialClusters:function(){var s=Math.ceil(this._map.getMaxZoom()),a=Math.floor(this._map.getMinZoom()),h=this.options.maxClusterRadius,l=h;typeof h!="function"&&(l=function(){return h}),this.options.disableClusteringAtZoom!==null&&(s=this.options.disableClusteringAtZoom-1),this._maxZoom=s,this._gridClusters={},this._gridUnclustered={};for(var _=s;_>=a;_--)this._gridClusters[_]=new L.DistanceGrid(l(_)),this._gridUnclustered[_]=new L.DistanceGrid(l(_));this._topClusterLevel=new this._markerCluster(this,a-1)},_addLayer:function(s,a){var h=this._gridClusters,l=this._gridUnclustered,_=Math.floor(this._map.getMinZoom()),g,d;for(this.options.singleMarkerMode&&this._overrideMarkerIcon(s),s.on(this._childMarkerEventHandlers,this);a>=_;a--){g=this._map.project(s.getLatLng(),a);var v=h[a].getNearObject(g);if(v){v._addChild(s),s.__parent=v;return}if(v=l[a].getNearObject(g),v){var m=v.__parent;m&&this._removeLayer(v,!1);var C=new this._markerCluster(this,a,v,s);h[a].addObject(C,this._map.project(C._cLatLng,a)),v.__parent=C,s.__parent=C;var P=C;for(d=a-1;d>m._zoom;d--)P=new this._markerCluster(this,d,P),h[d].addObject(P,this._map.project(v.getLatLng(),d));m._addChild(P),this._removeFromGridUnclustered(v,a);return}l[a].addObject(s,g)}this._topClusterLevel._addChild(s),s.__parent=this._topClusterLevel},_refreshClustersIcons:function(){this._featureGroup.eachLayer(function(s){s instanceof L.MarkerCluster&&s._iconNeedsUpdate&&s._updateIcon()})},_enqueue:function(s){this._queue.push(s),this._queueTimeout||(this._queueTimeout=setTimeout(L.bind(this._processQueue,this),300))},_processQueue:function(){for(var s=0;ss?(this._animationStart(),this._animationZoomOut(this._zoom,s)):this._moveEnd()},_getExpandedVisibleBounds:function(){if(this.options.removeOutsideVisibleBounds){if(L.Browser.mobile)return this._checkBoundsMaxLat(this._map.getBounds())}else return this._mapBoundsInfinite;return this._checkBoundsMaxLat(this._map.getBounds().pad(1))},_checkBoundsMaxLat:function(s){var a=this._maxLat;return a!==void 0&&(s.getNorth()>=a&&(s._northEast.lat=1/0),s.getSouth()<=-a&&(s._southWest.lat=-1/0)),s},_animationAddLayerNonAnimated:function(s,a){if(a===s)this._featureGroup.addLayer(s);else if(a._childCount===2){a._addToMap();var h=a.getAllChildMarkers();this._featureGroup.removeLayer(h[0]),this._featureGroup.removeLayer(h[1])}else a._updateIcon()},_extractNonGroupLayers:function(s,a){var h=s.getLayers(),l=0,_;for(a=a||[];l=0;g--)C=m[g],h.contains(C._latlng)||l.removeLayer(C)}),this._forceLayout(),this._topClusterLevel._recursivelyBecomeVisible(h,a),l.eachLayer(function(d){!(d instanceof L.MarkerCluster)&&d._icon&&d.clusterShow()}),this._topClusterLevel._recursively(h,s,a,function(d){d._recursivelyRestoreChildPositions(a)}),this._ignoreMove=!1,this._enqueue(function(){this._topClusterLevel._recursively(h,s,_,function(d){l.removeLayer(d),d.clusterShow()}),this._animationEnd()})},_animationZoomOut:function(s,a){this._animationZoomOutSingle(this._topClusterLevel,s-1,a),this._topClusterLevel._recursivelyAddChildrenToMap(null,a,this._getExpandedVisibleBounds()),this._topClusterLevel._recursivelyRemoveChildrenFromMap(this._currentShownBounds,Math.floor(this._map.getMinZoom()),s,this._getExpandedVisibleBounds())},_animationAddLayer:function(s,a){var h=this,l=this._featureGroup;l.addLayer(s),a!==s&&(a._childCount>2?(a._updateIcon(),this._forceLayout(),this._animationStart(),s._setPos(this._map.latLngToLayerPoint(a.getLatLng())),s.clusterHide(),this._enqueue(function(){l.removeLayer(s),s.clusterShow(),h._animationEnd()})):(this._forceLayout(),h._animationStart(),h._animationZoomOutSingle(a,this._map.getMaxZoom(),this._zoom)))}},_animationZoomOutSingle:function(s,a,h){var l=this._getExpandedVisibleBounds(),_=Math.floor(this._map.getMinZoom());s._recursivelyAnimateChildrenInAndAddSelfToMap(l,_,a+1,h);var g=this;this._forceLayout(),s._recursivelyBecomeVisible(l,h),this._enqueue(function(){if(s._childCount===1){var d=s._markers[0];this._ignoreMove=!0,d.setLatLng(d.getLatLng()),this._ignoreMove=!1,d.clusterShow&&d.clusterShow()}else s._recursively(l,h,_,function(v){v._recursivelyRemoveChildrenFromMap(l,_,a+1)});g._animationEnd()})},_animationEnd:function(){this._map&&(this._map._mapPane.className=this._map._mapPane.className.replace(" leaflet-cluster-anim","")),this._inZoomAnimation--,this.fire("animationend")},_forceLayout:function(){L.Util.falseFn(document.body.offsetWidth)}}),L.markerClusterGroup=function(s){return new L.MarkerClusterGroup(s)};var b=L.MarkerCluster=L.Marker.extend({options:L.Icon.prototype.options,initialize:function(s,a,h,l){L.Marker.prototype.initialize.call(this,h?h._cLatLng||h.getLatLng():new L.LatLng(0,0),{icon:this,pane:s.options.clusterPane}),this._group=s,this._zoom=a,this._markers=[],this._childClusters=[],this._childCount=0,this._iconNeedsUpdate=!0,this._boundsNeedUpdate=!0,this._bounds=new L.LatLngBounds,h&&this._addChild(h),l&&this._addChild(l)},getAllChildMarkers:function(s,a){s=s||[];for(var h=this._childClusters.length-1;h>=0;h--)this._childClusters[h].getAllChildMarkers(s,a);for(var l=this._markers.length-1;l>=0;l--)a&&this._markers[l].__dragStart||s.push(this._markers[l]);return s},getChildCount:function(){return this._childCount},zoomToBounds:function(s){for(var a=this._childClusters.slice(),h=this._group._map,l=h.getBoundsZoom(this._bounds),_=this._zoom+1,g=h.getZoom(),d;a.length>0&&l>_;){_++;var v=[];for(d=0;d_?this._group._map.setView(this._latlng,_):l<=g?this._group._map.setView(this._latlng,g+1):this._group._map.fitBounds(this._bounds,s)},getBounds:function(){var s=new L.LatLngBounds;return s.extend(this._bounds),s},_updateIcon:function(){this._iconNeedsUpdate=!0,this._icon&&this.setIcon(this)},createIcon:function(){return this._iconNeedsUpdate&&(this._iconObj=this._group.options.iconCreateFunction(this),this._iconNeedsUpdate=!1),this._iconObj.createIcon()},createShadow:function(){return this._iconObj.createShadow()},_addChild:function(s,a){this._iconNeedsUpdate=!0,this._boundsNeedUpdate=!0,this._setClusterCenter(s),s instanceof L.MarkerCluster?(a||(this._childClusters.push(s),s.__parent=this),this._childCount+=s._childCount):(a||this._markers.push(s),this._childCount++),this.__parent&&this.__parent._addChild(s,!0)},_setClusterCenter:function(s){this._cLatLng||(this._cLatLng=s._cLatLng||s._latlng)},_resetBounds:function(){var s=this._bounds;s._southWest&&(s._southWest.lat=1/0,s._southWest.lng=1/0),s._northEast&&(s._northEast.lat=-1/0,s._northEast.lng=-1/0)},_recalculateBounds:function(){var s=this._markers,a=this._childClusters,h=0,l=0,_=this._childCount,g,d,v,m;if(_!==0){for(this._resetBounds(),g=0;g=0;g--)d=_[g],d._icon&&(d._setPos(a),d.clusterHide())},function(l){var _=l._childClusters,g,d;for(g=_.length-1;g>=0;g--)d=_[g],d._icon&&(d._setPos(a),d.clusterHide())})},_recursivelyAnimateChildrenInAndAddSelfToMap:function(s,a,h,l){this._recursively(s,l,a,function(_){_._recursivelyAnimateChildrenIn(s,_._group._map.latLngToLayerPoint(_.getLatLng()).round(),h),_._isSingleParent()&&h-1===l?(_.clusterShow(),_._recursivelyRemoveChildrenFromMap(s,a,h)):_.clusterHide(),_._addToMap()})},_recursivelyBecomeVisible:function(s,a){this._recursively(s,this._group._map.getMinZoom(),a,null,function(h){h.clusterShow()})},_recursivelyAddChildrenToMap:function(s,a,h){this._recursively(h,this._group._map.getMinZoom()-1,a,function(l){if(a!==l._zoom)for(var _=l._markers.length-1;_>=0;_--){var g=l._markers[_];h.contains(g._latlng)&&(s&&(g._backupLatlng=g.getLatLng(),g.setLatLng(s),g.clusterHide&&g.clusterHide()),l._group._featureGroup.addLayer(g))}},function(l){l._addToMap(s)})},_recursivelyRestoreChildPositions:function(s){for(var a=this._markers.length-1;a>=0;a--){var h=this._markers[a];h._backupLatlng&&(h.setLatLng(h._backupLatlng),delete h._backupLatlng)}if(s-1===this._zoom)for(var l=this._childClusters.length-1;l>=0;l--)this._childClusters[l]._restorePosition();else for(var _=this._childClusters.length-1;_>=0;_--)this._childClusters[_]._recursivelyRestoreChildPositions(s)},_restorePosition:function(){this._backupLatlng&&(this.setLatLng(this._backupLatlng),delete this._backupLatlng)},_recursivelyRemoveChildrenFromMap:function(s,a,h,l){var _,g;this._recursively(s,a-1,h-1,function(d){for(g=d._markers.length-1;g>=0;g--)_=d._markers[g],(!l||!l.contains(_._latlng))&&(d._group._featureGroup.removeLayer(_),_.clusterShow&&_.clusterShow())},function(d){for(g=d._childClusters.length-1;g>=0;g--)_=d._childClusters[g],(!l||!l.contains(_._latlng))&&(d._group._featureGroup.removeLayer(_),_.clusterShow&&_.clusterShow())})},_recursively:function(s,a,h,l,_){var g=this._childClusters,d=this._zoom,v,m;if(a<=d&&(l&&l(this),_&&d===h&&_(this)),d=0;v--)m=g[v],m._boundsNeedUpdate&&m._recalculateBounds(),s.intersects(m._bounds)&&m._recursively(s,a,h,l,_)},_isSingleParent:function(){return this._childClusters.length>0&&this._childClusters[0]._childCount===this._childCount}});L.Marker.include({clusterHide:function(){var s=this.options.opacity;return this.setOpacity(0),this.options.opacity=s,this},clusterShow:function(){return this.setOpacity(this.options.opacity)}}),L.DistanceGrid=function(s){this._cellSize=s,this._sqCellSize=s*s,this._grid={},this._objectPoint={}},L.DistanceGrid.prototype={addObject:function(s,a){var h=this._getCoord(a.x),l=this._getCoord(a.y),_=this._grid,g=_[l]=_[l]||{},d=g[h]=g[h]||[],v=L.Util.stamp(s);this._objectPoint[v]=a,d.push(s)},updateObject:function(s,a){this.removeObject(s),this.addObject(s,a)},removeObject:function(s,a){var h=this._getCoord(a.x),l=this._getCoord(a.y),_=this._grid,g=_[l]=_[l]||{},d=g[h]=g[h]||[],v,m;for(delete this._objectPoint[L.Util.stamp(s)],v=0,m=d.length;v=0;g--){if(d=a[g],v=this.getDistant(d,s),v>0)_.push(d);else continue;v>h&&(h=v,l=d)}return{maxPoint:l,newPoints:_}},buildConvexHull:function(s,a){var h=[],l=this.findMostDistantPointFromBaseLine(s,a);return l.maxPoint?(h=h.concat(this.buildConvexHull([s[0],l.maxPoint],l.newPoints)),h=h.concat(this.buildConvexHull([l.maxPoint,s[1]],l.newPoints)),h):[s[0]]},getConvexHull:function(s){var a=!1,h=!1,l=!1,_=!1,g=null,d=null,v=null,m=null,C=null,P=null,F;for(F=s.length-1;F>=0;F--){var S=s[F];(a===!1||S.lat>a)&&(g=S,a=S.lat),(h===!1||S.latl)&&(v=S,l=S.lng),(_===!1||S.lng<_)&&(m=S,_=S.lng)}h!==a?(P=d,C=g):(P=m,C=v);var B=[].concat(this.buildConvexHull([P,C],s),this.buildConvexHull([C,P],s));return B}}}(),L.MarkerCluster.include({getConvexHull:function(){var s=this.getAllChildMarkers(),a=[],h,l;for(l=s.length-1;l>=0;l--)h=s[l].getLatLng(),a.push(h);return L.QuickHull.getConvexHull(a)}}),L.MarkerCluster.include({_2PI:Math.PI*2,_circleFootSeparation:25,_circleStartAngle:0,_spiralFootSeparation:28,_spiralLengthStart:11,_spiralLengthFactor:5,_circleSpiralSwitchover:9,spiderfy:function(){if(!(this._group._spiderfied===this||this._group._inZoomAnimation)){var s=this.getAllChildMarkers(null,!0),a=this._group,h=a._map,l=h.latLngToLayerPoint(this._latlng),_;this._group._unspiderfy(),this._group._spiderfied=this,this._group.options.spiderfyShapePositions?_=this._group.options.spiderfyShapePositions(s.length,l):s.length>=this._circleSpiralSwitchover?_=this._generatePointsSpiral(s.length,l):(l.y+=10,_=this._generatePointsCircle(s.length,l)),this._animationSpiderfy(s,_)}},unspiderfy:function(s){this._group._inZoomAnimation||(this._animationUnspiderfy(s),this._group._spiderfied=null)},_generatePointsCircle:function(s,a){var h=this._group.options.spiderfyDistanceMultiplier*this._circleFootSeparation*(2+s),l=h/this._2PI,_=this._2PI/s,g=[],d,v;for(l=Math.max(l,35),g.length=s,d=0;d=0;m--)m=0;g--)_=l[g],h.removeLayer(_),_._preSpiderfyLatlng&&(_.setLatLng(_._preSpiderfyLatlng),delete _._preSpiderfyLatlng),_.setZIndexOffset&&_.setZIndexOffset(0),_._spiderLeg&&(a.removeLayer(_._spiderLeg),delete _._spiderLeg);s.fire("unspiderfied",{cluster:this,markers:l}),s._ignoreMove=!1,s._spiderfied=null}}),L.MarkerClusterNonAnimated=L.MarkerCluster.extend({_animationSpiderfy:function(s,a){var h=this._group,l=h._map,_=h._featureGroup,g=this._group.options.spiderLegPolylineOptions,d,v,m,C;for(h._ignoreMove=!0,d=0;d=0;F--)tt=_.layerPointToLatLng(a[F]),S=s[F],S._preSpiderfyLatlng=S._latlng,S.setLatLng(tt),S.clusterShow&&S.clusterShow(),m&&(B=S._spiderLeg,U=B._path,U.style.strokeDashoffset=0,B.setStyle({opacity:P}));this.setOpacity(.3),l._ignoreMove=!1,setTimeout(function(){l._animationEnd(),l.fire("spiderfied",{cluster:h,markers:s})},200)},_animationUnspiderfy:function(s){var a=this,h=this._group,l=h._map,_=h._featureGroup,g=s?l._latLngToNewLayerPoint(this._latlng,s.zoom,s.center):l.latLngToLayerPoint(this._latlng),d=this.getAllChildMarkers(null,!0),v=L.Path.SVG,m,C,P,F,S,B;for(h._ignoreMove=!0,h._animationStart(),this.setOpacity(1),C=d.length-1;C>=0;C--)m=d[C],m._preSpiderfyLatlng&&(m.closePopup(),m.setLatLng(m._preSpiderfyLatlng),delete m._preSpiderfyLatlng,B=!0,m._setPos&&(m._setPos(g),B=!1),m.clusterHide&&(m.clusterHide(),B=!1),B&&_.removeLayer(m),v&&(P=m._spiderLeg,F=P._path,S=F.getTotalLength()+.1,F.style.strokeDashoffset=S,P.setStyle({opacity:0})));h._ignoreMove=!1,setTimeout(function(){var U=0;for(C=d.length-1;C>=0;C--)m=d[C],m._spiderLeg&&U++;for(C=d.length-1;C>=0;C--)m=d[C],m._spiderLeg&&(m.clusterShow&&m.clusterShow(),m.setZIndexOffset&&m.setZIndexOffset(0),U>1&&_.removeLayer(m),l.removeLayer(m._spiderLeg),delete m._spiderLeg);h._animationEnd(),h.fire("unspiderfied",{cluster:a,markers:d})},200)}}),L.MarkerClusterGroup.include({_spiderfied:null,unspiderfy:function(){this._unspiderfy.apply(this,arguments)},_spiderfierOnAdd:function(){this._map.on("click",this._unspiderfyWrapper,this),this._map.options.zoomAnimation&&this._map.on("zoomstart",this._unspiderfyZoomStart,this),this._map.on("zoomend",this._noanimationUnspiderfy,this),L.Browser.touch||this._map.getRenderer(this)},_spiderfierOnRemove:function(){this._map.off("click",this._unspiderfyWrapper,this),this._map.off("zoomstart",this._unspiderfyZoomStart,this),this._map.off("zoomanim",this._unspiderfyZoomAnim,this),this._map.off("zoomend",this._noanimationUnspiderfy,this),this._noanimationUnspiderfy()},_unspiderfyZoomStart:function(){this._map&&this._map.on("zoomanim",this._unspiderfyZoomAnim,this)},_unspiderfyZoomAnim:function(s){L.DomUtil.hasClass(this._map._mapPane,"leaflet-touching")||(this._map.off("zoomanim",this._unspiderfyZoomAnim,this),this._unspiderfy(s))},_unspiderfyWrapper:function(){this._unspiderfy()},_unspiderfy:function(s){this._spiderfied&&this._spiderfied.unspiderfy(s)},_noanimationUnspiderfy:function(){this._spiderfied&&this._spiderfied._noanimationUnspiderfy()},_unspiderfyLayer:function(s){s._spiderLeg&&(this._featureGroup.removeLayer(s),s.clusterShow&&s.clusterShow(),s.setZIndexOffset&&s.setZIndexOffset(0),this._map.removeLayer(s._spiderLeg),delete s._spiderLeg)}}),L.MarkerClusterGroup.include({refreshClusters:function(s){return s?s instanceof L.MarkerClusterGroup?s=s._topClusterLevel.getAllChildMarkers():s instanceof L.LayerGroup?s=s._layers:s instanceof L.MarkerCluster?s=s.getAllChildMarkers():s instanceof L.Marker&&(s=[s]):s=this._topClusterLevel.getAllChildMarkers(),this._flagParentsIconsNeedUpdate(s),this._refreshClustersIcons(),this.options.singleMarkerMode&&this._refreshSingleMarkerModeMarkers(s),this},_flagParentsIconsNeedUpdate:function(s){var a,h;for(a in s)for(h=s[a].__parent;h;)h._iconNeedsUpdate=!0,h=h.__parent},_refreshSingleMarkerModeMarkers:function(s){var a,h;for(a in s)h=s[a],this.hasLayer(h)&&h.setIcon(this._overrideMarkerIcon(h))}}),L.Marker.include({refreshIconOptions:function(s,a){var h=this.options.icon;return L.setOptions(h,s),this.setIcon(h),a&&this.__parent&&this.__parent._group.refreshClusters(this),this}}),f.MarkerClusterGroup=T,f.MarkerCluster=b,Object.defineProperty(f,"__esModule",{value:!0})})}(ae,ae.exports)),ae.exports}As();const On={healthy:"#10b981",warning:"#f59e0b",critical:"#ef4444"},zn={healthy:"background: #d1fae5; color: #065f46;",warning:"background: #fef3c7; color: #92400e;",critical:"background: #fee2e2; color: #991b1b;"},ui={small:{dimension:"30px",fontSize:"12px",threshold:0},medium:{dimension:"40px",fontSize:"14px",threshold:10},large:{dimension:"50px",fontSize:"16px",threshold:50}},Ns={size:20,emoji:"🌳",className:"custom-tree-marker"},Rs={backgroundColor:"#10b981",className:"custom-cluster-icon",iconSize:[30,30]},En=(x="healthy")=>{const w=On[x]||On.healthy,{size:f,emoji:T,className:b}=Ns;return dt.divIcon({className:b,html:Ds(w,f,T),iconSize:[f,f],iconAnchor:[f/2,f/2]})},Ds=(x,w,f)=>` +
+ ${f} +
+`,Hs=x=>{const w=Fs(x),{dimension:f,fontSize:T}=ui[w],{backgroundColor:b,className:s,iconSize:a}=Rs;return dt.divIcon({html:Gs(b,f,T,x),className:s,iconSize:a})},Fs=x=>x>ui.large.threshold?"large":x>ui.medium.threshold?"medium":"small",Gs=(x,w,f,T)=>` +
${T}
+`,Zn=x=>{const w=zn[x.health]||zn.healthy;return js(x,w)},js=(x,w)=>` +
+

${x.name}

+

Species: ${x.species}

+

Height: ${x.height}m

+

Health: + ${x.health} +

+
+`,Us=({trees:x,onTreeClick:w,zoom:f})=>{const T=ci();return N.useEffect(()=>{if(!(!T||!x.length)){if(T.eachLayer(b=>{(b instanceof dt.MarkerClusterGroup||b instanceof dt.LayerGroup)&&T.removeLayer(b)}),f<12){const b=dt.markerClusterGroup({chunkedLoading:!0,spiderfyOnMaxZoom:!0,showCoverageOnHover:!1,zoomToBoundsOnClick:!0,maxClusterRadius:50,iconCreateFunction:s=>{const a=s.getChildCount();return Hs(a)}});x.forEach(s=>{const a=dt.marker([s.lat,s.lng],{icon:En(s.health)});a.bindPopup(Zn(s)),a.on("click",()=>w(s)),b.addLayer(a)}),T.addLayer(b)}else{const b=dt.layerGroup();x.forEach(s=>{const a=dt.marker([s.lat,s.lng],{icon:En(s.health)});a.bindPopup(Zn(s)),a.on("click",()=>w(s)),b.addLayer(a)}),b.addTo(T)}return()=>{T.eachLayer(b=>{(b instanceof dt.MarkerClusterGroup||b instanceof dt.LayerGroup)&&T.removeLayer(b)})}}},[T,x,w,f]),null},Ws=({onMapReady:x,onMapError:w})=>{const f=ci();return N.useEffect(()=>{const T=()=>{x&&x()},b=s=>{console.error("Map error:",s),w&&w(s)};return f.whenReady(()=>{T()}),f.on("tileerror",b),()=>{f.off("tileerror",b)}},[f,x,w]),null},Vs=({onZoomChange:x})=>{const w=ci();return N.useEffect(()=>{const f=()=>{x&&x(w.getZoom())};return w.on("zoomend",f),()=>{w.off("zoomend",f)}},[w]),null},qs=Ot.div` + height: 600px; + width: 100%; + border-radius: 0.75rem; + overflow: hidden; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; + position: relative; +`;Ot.div` + background: white; + padding: 1rem 1.5rem; + border-bottom: 1px solid #e5e7eb; + display: flex; + justify-content: space-between; + align-items: center; +`;Ot.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`;Ot.div` + display: flex; + gap: 0.5rem; +`;Ot.button` + padding: 0.5rem; + background: white; + border: 1px solid #d1d5db; + border-radius: 0.375rem; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background: #f9fafb; + border-color: #9ca3af; + } + + &:focus { + outline: none; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } +`;const Ks=Ot.div` + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: rgba(255, 255, 255, 0.9); + display: flex; + align-items: center; + justify-content: center; + z-index: 1000; + border-radius: 0.75rem; +`,$s=Ot.div` + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: rgba(255, 255, 255, 0.95); + display: flex; + align-items: center; + justify-content: center; + z-index: 1000; + border-radius: 0.75rem; +`,Ys=Ot.div` + text-align: center; + padding: 2rem; + max-width: 300px; +`;delete dt.Icon.Default.prototype._getIconUrl;dt.Icon.Default.mergeOptions({iconRetinaUrl:"https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.7.1/images/marker-icon-2x.png",iconUrl:"https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.7.1/images/marker-icon.png",shadowUrl:"https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.7.1/images/marker-shadow.png"});const Xs=({trees:x=[],onTreeSelect:w,loading:f=!1,error:T=null})=>{const[b,s]=N.useState(5),[a,h]=N.useState(null),[l,_]=N.useState(!0),[g,d]=N.useState(null),v=x,m=N.useCallback(B=>{h(B),w&&w(B)},[w]),C=N.useCallback(B=>{s(B)},[]),P=N.useCallback(()=>{_(!1),d(null)},[]),F=N.useCallback(B=>{_(!1),d((B==null?void 0:B.message)||"Failed to load map")},[]),S=N.useCallback(()=>{_(!0),d(null),setTimeout(()=>{_(!1)},1e3)},[]);return z.jsxs("div",{children:[z.jsxs(qs,{children:[z.jsxs(Zs,{center:[59.3293,18.0686],zoom:5,style:{height:"100%",width:"100%"},children:[z.jsx(Is,{url:"https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png",attribution:'© OpenStreetMap contributors'}),z.jsx(Ws,{onMapReady:P,onMapError:F}),z.jsx(Vs,{onZoomChange:C}),z.jsx(Us,{trees:v,onTreeClick:m,zoom:b})]}),(l||f)&&z.jsx(Ks,{children:z.jsxs("div",{className:"text-center",children:[z.jsx(In,{}),z.jsx("p",{className:"mt-2 text-sm text-gray-600",children:"Loading map..."})]})}),(g||T)&&z.jsx($s,{children:z.jsxs(Ys,{children:[z.jsx("svg",{className:"w-12 h-12 text-red-500 mx-auto mb-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:z.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 8v4m0 4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"})}),z.jsx("h4",{className:"text-lg font-semibold text-gray-900 mb-2",children:"Map Error"}),z.jsx("p",{className:"text-sm text-gray-600 mb-4",children:g||T}),z.jsx("button",{onClick:S,className:"px-4 py-2 bg-green-600 text-white rounded-lg hover:bg-green-700 transition-colors text-sm",children:"Retry"})]})})]}),z.jsxs("div",{className:"mt-4 p-4 bg-white rounded-lg border border-gray-200",role:"region","aria-label":"Map legend",children:[z.jsx("h4",{className:"font-semibold text-gray-900 mb-2",children:"Legend"}),z.jsxs("div",{className:"flex flex-wrap gap-4 text-sm text-gray-900",children:[z.jsxs("div",{className:"flex items-center",children:[z.jsx("div",{className:"w-4 h-4 bg-green-500 rounded-full mr-2"}),z.jsx("span",{children:"Healthy Trees"})]}),z.jsxs("div",{className:"flex items-center",children:[z.jsx("div",{className:"w-4 h-4 bg-yellow-500 rounded-full mr-2"}),z.jsx("span",{children:"Warning"})]}),z.jsxs("div",{className:"flex items-center",children:[z.jsx("div",{className:"w-4 h-4 bg-red-500 rounded-full mr-2"}),z.jsx("span",{children:"Critical"})]}),z.jsx("div",{className:"flex items-center",children:z.jsxs("span",{children:["Showing ",v.length," of ",x.length," trees"]})}),z.jsxs("div",{className:"flex items-center",children:[z.jsxs("span",{"aria-live":"polite",children:["Zoom level: ",b]}),b<12&&z.jsx("span",{className:"ml-4 text-xs",children:"(Clustering enabled)"})]})]})]})]})},Js=()=>{const[x,w]=N.useState(null),[f,T]=N.useState(!1),b=N.useCallback(h=>{w(h),T(!0)},[]),s=N.useCallback(()=>{w(null),T(!1)},[]),a=N.useCallback(()=>{w(null),T(!1)},[]);return{selectedTree:x,isModalOpen:f,handleTreeSelect:b,handleCloseTreeDetail:s,clearSelection:a}},rr=()=>{const{sidebarOpen:x,toggleSidebar:w,closeSidebar:f}=fs(),{selectedTree:T,isModalOpen:b,handleTreeSelect:s,handleCloseTreeDetail:a}=Js(),[h,l]=N.useState([]),[_,g]=N.useState(!0),[d,v]=N.useState(null);return N.useEffect(()=>{(async()=>{g(!0),v(null);try{const C=await gs.getAll();let P=[];C.data&&Array.isArray(C.data.trees)?P=C.data.trees:C.data&&Array.isArray(C.data)?P=C.data:Array.isArray(C)&&(P=C);const F=P.map(S=>{var U,at,tt,Tt;const B=((at=(U=S.location)==null?void 0:U.coordinates)==null?void 0:at.length)===2;return{...S,lat:B?S.location.coordinates[1]:null,lng:B?S.location.coordinates[0]:null,name:S.treeId||S.name||`Tree ${S._id}`,health:S.health||((Tt=(tt=S.measurements)==null?void 0:tt[0])==null?void 0:Tt.healthStatus)||"unknown"}}).filter(S=>S.lat!==null&&S.lng!==null);l(F)}catch(C){console.error("Failed to fetch trees:",C),v(C.message||"Failed to load trees"),l([])}finally{g(!1)}})()},[]),z.jsxs("div",{className:"min-h-screen bg-gray-50 dark:bg-gray-900 flex flex-col",children:[z.jsx(_s,{onToggleSidebar:w}),z.jsx(ps,{isOpen:x,onClose:f}),z.jsx("main",{className:"flex-1 p-4 md:p-6 lg:p-8 lg:ml-64",children:z.jsxs("div",{className:"max-w-7xl mx-auto",children:[z.jsx("div",{className:"mb-8",children:z.jsx("div",{className:"flex justify-between items-center mb-4",children:z.jsxs("div",{children:[z.jsx("h2",{className:"text-3xl font-bold text-gray-900 dark:text-white mb-2",children:"Forest Map"}),z.jsx("p",{className:"text-gray-600 dark:text-gray-300",children:"Explore your forests and individual trees with interactive mapping."})]})})}),z.jsx(ms,{tree:T,isOpen:b,onClose:a}),_?z.jsx("div",{className:"flex justify-center items-center h-96 bg-white dark:bg-gray-800 rounded-lg shadow",children:z.jsx(In,{text:"Loading trees..."})}):z.jsx(Xs,{trees:h,onTreeSelect:s,loading:_,error:d})]})})]})};export{rr as MapPage}; diff --git a/frontend/dist/assets/Navbar-BjYzKaxh.js b/frontend/dist/assets/Navbar-BjYzKaxh.js new file mode 100644 index 0000000000..737e1506b6 --- /dev/null +++ b/frontend/dist/assets/Navbar-BjYzKaxh.js @@ -0,0 +1 @@ +import{j as e,L as a,r as n}from"./index-tEqMizXb.js";import{u as m}from"./useKeyboardNavigation-DT9lybaQ.js";import{D as d}from"./DarkModeToggle-ZfYhBz0e.js";import{I as g}from"./IconButton-DmJ5sToe.js";const h=({className:r="",...t})=>e.jsx("svg",{className:r,fill:"none",stroke:"currentColor",viewBox:"0 0 24 24","aria-hidden":"true",...t,children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4 6h16M4 12h16M4 18h16"})}),f=({className:r="",...t})=>e.jsx("svg",{className:r,fill:"none",stroke:"currentColor",viewBox:"0 0 24 24","aria-hidden":"true",...t,children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M6 18L18 6M6 6l12 12"})}),b=({className:r=""})=>e.jsx(a,{to:"/",className:`text-2xl font-extrabold text-green-600 dark:text-green-400 tracking-tight ${r}`,children:"Nanwa"}),c=()=>{const r="text-gray-700 dark:text-gray-300 hover:text-green-600 dark:hover:text-green-400 transition-colors px-4 py-2 text-base font-medium focus-ring rounded-md";return e.jsxs(e.Fragment,{children:[e.jsx("a",{href:"#features",className:r,children:"Features"}),e.jsx("a",{href:"#about",className:r,children:"About"}),e.jsx(a,{to:"/login",className:r,children:"Login"}),e.jsx(a,{to:"/register",className:"ml-2 bg-green-600 dark:bg-green-500 text-white px-4 py-2 rounded-lg hover:bg-green-700 dark:hover:bg-green-600 transition-colors text-base font-medium focus-ring",children:"Get Started"})]})},p=({isOpen:r})=>r?e.jsx("div",{className:"md:hidden border-t border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-900 px-4 pb-4 space-y-1",role:"menu","aria-label":"Mobile menu",children:e.jsx(c,{})}):null,N=()=>{const[r,t]=n.useState(!1),o=n.useRef(null),{containerRef:x,focusFirst:i}=m({onEscape:()=>t(!1),trapFocus:r,autoFocus:!1}),l=()=>{t(!r)};n.useEffect(()=>{setTimeout(r?()=>{i()}:()=>{var s;(s=o.current)==null||s.focus()},100)},[r,i]);const u=s=>{(s.key==="Enter"||s.key===" ")&&(s.preventDefault(),l())};return e.jsxs("nav",{className:"fixed top-0 left-0 right-0 z-50 bg-white/80 dark:bg-gray-900/80 backdrop-blur-sm border-b border-gray-200 dark:border-gray-700",role:"navigation","aria-label":"Main",children:[e.jsx("div",{className:"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8",children:e.jsxs("div",{className:"flex justify-between items-center h-16",children:[e.jsx(b,{}),e.jsxs("div",{className:"hidden md:flex items-center gap-4",children:[e.jsx(c,{}),e.jsx(d,{size:"sm"})]}),e.jsxs("div",{className:"md:hidden flex items-center gap-2",children:[e.jsx(d,{size:"sm"}),e.jsx(g,{ref:o,"aria-label":"Toggle menu","aria-expanded":r,"aria-controls":"mobile-menu",className:"text-gray-700 dark:text-gray-300 hover:text-green-600 dark:hover:text-green-400",onClick:l,onKeyDown:u,children:r?e.jsx(f,{className:"h-6 w-6"}):e.jsx(h,{className:"h-6 w-6"})})]})]})}),e.jsx("div",{ref:x,id:"mobile-menu",children:e.jsx(p,{isOpen:r})})]})};export{N}; diff --git a/frontend/dist/assets/OverviewDashboardPage-ChIgynnA.js b/frontend/dist/assets/OverviewDashboardPage-ChIgynnA.js new file mode 100644 index 0000000000..908bd3ce0f --- /dev/null +++ b/frontend/dist/assets/OverviewDashboardPage-ChIgynnA.js @@ -0,0 +1 @@ +import{j as e,r as c,b as H,L as P}from"./index-tEqMizXb.js";import{C as y,a as N,b as C,c as ie,d as Q,e as le}from"./ChartComponents-BKyCVLLP.js";import{c as Z,R as _,C as ce,T as J,a as ee,X as re,Y as te,B as de,L as he,b as xe,d as U,e as se,E as K}from"./EnhancedStatCard-CVgqadrW.js";import{t as ae,G as ue}from"./GlobalFilters-B5C4gkva.js";import{P as me,a as ge,B as je}from"./PieChart-DRbmPJpB.js";import{u as ve,D as pe,a as be}from"./useSidebarState-CJJaSDFr.js";import"./vendor-BtP0CW_r.js";import"./DarkModeToggle-ZfYhBz0e.js";import"./IconButton-DmJ5sToe.js";const q=({icon:s,title:r,value:o,color:l="green"})=>{const j={green:"bg-green-100 dark:bg-green-900/30 text-green-600 dark:text-green-400",blue:"bg-blue-100 dark:bg-blue-900/30 text-blue-600 dark:text-blue-400",yellow:"bg-yellow-100 dark:bg-yellow-900/30 text-yellow-600 dark:text-yellow-400",purple:"bg-purple-100 dark:bg-purple-900/30 text-purple-600 dark:text-purple-400"};return e.jsx("div",{className:"bg-white dark:bg-gray-800 p-6 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700",children:e.jsxs("div",{className:"flex items-center",children:[e.jsx("div",{className:`p-2 rounded-lg ${j[l]}`,children:s}),e.jsxs("div",{className:"ml-4",children:[e.jsx("p",{className:"text-sm font-medium text-gray-600 dark:text-gray-300",children:r}),e.jsx("p",{className:"text-2xl font-bold text-gray-900 dark:text-white",children:o})]})]})})},oe=(s="line")=>c.useMemo(()=>({...{margin:{top:5,right:30,left:20,bottom:5},gridProps:{strokeDasharray:"3 3",stroke:"#f3f4f6"},axisProps:{stroke:"#6b7280",fontSize:12}},...{line:{strokeWidth:3,dot:{strokeWidth:2,r:4},activeDot:{r:6,strokeWidth:2}},bar:{radius:[4,4,0,0]},pie:{innerRadius:60,outerRadius:80,paddingAngle:5}}[s]}),[s]),fe=5*60*1e3,ke=(s,r)=>{const o=s.name||"unknown",l=JSON.stringify(r);return`chart_${o}_${btoa(l)}`},ye=s=>{try{const r=sessionStorage.getItem(s);if(!r)return null;const{data:o,timestamp:l}=JSON.parse(r);return Date.now()-l>fe?(sessionStorage.removeItem(s),null):o}catch(r){return console.warn("Cache read error:",r),null}},Ne=(s,r)=>{try{const o={data:r,timestamp:Date.now()};sessionStorage.setItem(s,JSON.stringify(o))}catch(o){console.warn("Cache write error:",o)}},ne=(s,r={},o={})=>{var h,b;const[l,j]=c.useState(null),[t,m]=c.useState(!o.skip),[f,x]=c.useState(null),{skip:v=!1}=o,d=c.useMemo(()=>ae(r),[(h=r==null?void 0:r.dateRange)==null?void 0:h.startDate,(b=r==null?void 0:r.dateRange)==null?void 0:b.endDate,r==null?void 0:r.selectedForests,r==null?void 0:r.species]),n=c.useCallback(async()=>{try{m(!0),x(null);const u=ke(s,d),p=ye(u);if(p){j(p),m(!1);return}const k=(await s(d)).data;Ne(u,k),j(k)}catch(u){console.error("Error fetching chart data:",u),x(u.message||"Failed to fetch chart data")}finally{m(!1)}},[s,d]);c.useEffect(()=>{v||n()},[n,v]);const g=c.useCallback(()=>{n()},[n]);return{data:l,loading:t,error:f,refresh:g}},Ce=(s={})=>ne(Z.getSurvivalRate,s),we=(s={})=>ne(Z.getCO2Absorption,s),Se=({filters:s={},dashboardData:r=null})=>{var g,h,b;const{data:o,loading:l,error:j}=Ce(r?{}:s),t=((g=r==null?void 0:r.charts)==null?void 0:g.survivalRate)||o,m=((h=t==null?void 0:t.chartData)==null?void 0:h.length)>0&&((b=t.chartData[t.chartData.length-1])==null?void 0:b.survivalRate)||0,f=r?!1:l,x=r?!1:j,v={survived:m,lost:100-m},d=oe("pie"),n=[{name:"Survived",value:v.survived,color:"#10b981"},{name:"Lost",value:v.lost,color:"#ef4444"}];return f?e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"Tree Survival Rate"})}),e.jsx("div",{className:"flex justify-center items-center h-48",children:e.jsx(H,{size:"32px",text:"Loading chart data..."})})]}):x?e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"Tree Survival Rate"})}),e.jsx("div",{className:"flex justify-center items-center h-48",children:e.jsxs("div",{className:"text-center",children:[e.jsx("p",{className:"text-red-600 mb-2",children:"Error loading chart data"}),e.jsx("p",{className:"text-sm text-gray-500",children:x})]})})]}):e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"Tree Survival Rate"})}),e.jsx(ie,{value:v.survived,label:"Survival Rate"}),e.jsx(_,{width:"100%",height:"100%",children:e.jsxs(me,{children:[e.jsx(ge,{data:n,cx:"50%",cy:"50%",innerRadius:d.innerRadius,outerRadius:d.outerRadius,paddingAngle:d.paddingAngle,dataKey:"value",children:n.map((u,p)=>e.jsx(ce,{fill:u.color},`cell-${p}`))}),e.jsx(J,{content:e.jsx(Q,{valueFormatter:u=>`${u.toFixed(1)}%`})})]})}),e.jsx(le,{data:n})]})},Le=c.memo(Se),Ae=({filters:s={},dashboardData:r=null})=>{var n,g;const o={...s,groupBy:"year"},{data:l,loading:j,error:t}=we(r?{}:o),m=((n=r==null?void 0:r.charts)==null?void 0:n.co2Absorption)||l,f=((g=m==null?void 0:m.chartData)==null?void 0:g.map(h=>({year:h.period,co2:h.totalAbsorption||0})))||[],x=oe("bar"),v=r?!1:j,d=r?!1:t;return v?e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"CO₂ Absorption by Year"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx(H,{size:"32px",text:"Loading chart data..."})})]}):d?e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"CO₂ Absorption by Year"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsxs("div",{className:"text-center",children:[e.jsx("p",{className:"text-red-600 mb-2",children:"Error loading chart data"}),e.jsx("p",{className:"text-sm text-gray-500",children:d})]})})]}):e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"CO₂ Absorption Over Time"})}),e.jsx(_,{width:"100%",height:"100%",children:e.jsxs(je,{data:f,margin:x.margin,children:[e.jsx(ee,{...x.gridProps}),e.jsx(re,{dataKey:"year",...x.axisProps}),e.jsx(te,{...x.axisProps,label:{value:"CO₂ (tons)",angle:-90,position:"insideLeft",fontSize:12}}),e.jsx(J,{content:e.jsx(Q,{valueFormatter:h=>`CO₂ Absorption: ${h.toFixed(1)} tons`})}),e.jsx(de,{dataKey:"co2",fill:"#8b5cf6",radius:x.radius})]})})]})},Fe=c.memo(Ae),Ve=({filters:s={},dashboardData:r=null})=>{const[o,l]=c.useState([]),[j,t]=c.useState(!r),[m,f]=c.useState(null);c.useEffect(()=>{var n;if((n=r==null?void 0:r.charts)!=null&&n.forestValue){l(r.charts.forestValue.chartData),t(!1);return}r||(async()=>{var g,h,b,u,p,S,k,A,F,V,D,E,R,M,O,$,B,I,W,z;try{t(!0),f(null);const a=(await se.getStats(s)).data,T=[{period:"2020",currentValue:(((h=(g=a==null?void 0:a.investor)==null?void 0:g.portfolio)==null?void 0:h.totalCurrentValue)||0)*.7,acquisitionCost:((u=(b=a==null?void 0:a.investor)==null?void 0:b.portfolio)==null?void 0:u.totalAcquisitionCost)||0},{period:"2021",currentValue:(((S=(p=a==null?void 0:a.investor)==null?void 0:p.portfolio)==null?void 0:S.totalCurrentValue)||0)*.8,acquisitionCost:((A=(k=a==null?void 0:a.investor)==null?void 0:k.portfolio)==null?void 0:A.totalAcquisitionCost)||0},{period:"2022",currentValue:(((V=(F=a==null?void 0:a.investor)==null?void 0:F.portfolio)==null?void 0:V.totalCurrentValue)||0)*.9,acquisitionCost:((E=(D=a==null?void 0:a.investor)==null?void 0:D.portfolio)==null?void 0:E.totalAcquisitionCost)||0},{period:"2023",currentValue:(((M=(R=a==null?void 0:a.investor)==null?void 0:R.portfolio)==null?void 0:M.totalCurrentValue)||0)*.95,acquisitionCost:(($=(O=a==null?void 0:a.investor)==null?void 0:O.portfolio)==null?void 0:$.totalAcquisitionCost)||0},{period:"2024",currentValue:((I=(B=a==null?void 0:a.investor)==null?void 0:B.portfolio)==null?void 0:I.totalCurrentValue)||0,acquisitionCost:((z=(W=a==null?void 0:a.investor)==null?void 0:W.portfolio)==null?void 0:z.totalAcquisitionCost)||0}];l(T)}catch(L){console.error("Error fetching forest value data:",L),f(L.message||"Failed to load forest value data")}finally{t(!1)}})()},[s,r]);const x=d=>d?new Intl.NumberFormat("sv-SE",{style:"currency",currency:"SEK",minimumFractionDigits:0,maximumFractionDigits:0}).format(d):"0",v=({active:d,payload:n,label:g})=>d&&n&&n.length?e.jsxs("div",{className:"bg-white dark:bg-gray-800 p-3 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg",children:[e.jsx("p",{className:"text-sm font-medium text-gray-900 dark:text-white",children:`Year: ${g}`}),n.map((h,b)=>e.jsx("p",{className:"text-sm",style:{color:h.color},children:`${h.dataKey==="currentValue"?"Current Value":"Acquisition Cost"}: ${x(h.value)}`},b)),n.length===2&&e.jsxs("p",{className:"text-sm font-medium text-green-600 dark:text-green-400 mt-1",children:["Appreciation: ",x(n[0].value-n[1].value)]})]}):null;return j?e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"Forest Value Appreciation"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx(H,{size:"32px",text:"Loading value data..."})})]}):m?e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"Forest Value Appreciation"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsxs("div",{className:"text-center",children:[e.jsx("p",{className:"text-red-600 mb-2",children:"Error loading chart data"}),e.jsx("p",{className:"text-sm text-gray-500",children:m})]})})]}):!o||o.length===0?e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"Forest Value Appreciation"})}),e.jsx("div",{className:"flex justify-center items-center h-72",children:e.jsx("p",{className:"text-gray-500",children:"No forest value data available"})})]}):e.jsxs(y,{children:[e.jsx(N,{children:e.jsx(C,{children:"Forest Value Appreciation"})}),e.jsx(_,{width:"100%",height:300,children:e.jsxs(he,{data:o,margin:{top:5,right:30,left:20,bottom:5},children:[e.jsx(ee,{strokeDasharray:"3 3",className:"opacity-30"}),e.jsx(re,{dataKey:"period",className:"text-xs",tick:{fontSize:12}}),e.jsx(te,{className:"text-xs",tick:{fontSize:12},tickFormatter:x}),e.jsx(J,{content:e.jsx(v,{})}),e.jsx(xe,{}),e.jsx(U,{type:"monotone",dataKey:"acquisitionCost",stroke:"#8b5cf6",strokeWidth:2,name:"Acquisition Cost",dot:{r:4}}),e.jsx(U,{type:"monotone",dataKey:"currentValue",stroke:"#10b981",strokeWidth:2,name:"Current Value",dot:{r:4}})]})})]})},De=10*60*1e3,Ee=s=>{const r=JSON.stringify(s);return`dashboard_stats_${btoa(r)}`},Re=s=>{try{const r=sessionStorage.getItem(s);if(!r)return null;const{data:o,timestamp:l}=JSON.parse(r);return{data:o,timestamp:l,isExpired:Date.now()-l>De}}catch(r){return console.warn("Dashboard cache read error:",r),null}},Me=(s,r)=>{try{const o={data:r,timestamp:Date.now()};sessionStorage.setItem(s,JSON.stringify(o))}catch(o){console.warn("Dashboard cache write error:",o)}},Oe=(s={})=>{var g,h;const[r,o]=c.useState(null),[l,j]=c.useState(!0),[t,m]=c.useState(!1),[f,x]=c.useState(null),v=c.useMemo(()=>ae(s),[(g=s==null?void 0:s.dateRange)==null?void 0:g.startDate,(h=s==null?void 0:s.dateRange)==null?void 0:h.endDate,s==null?void 0:s.selectedForests,s==null?void 0:s.species]),d=c.useCallback(async(b=!0)=>{try{const u=Ee(v),p=Re(u);if(p&&b){if(o(p.data),j(!1),m(p.isExpired),!p.isExpired){x(null);return}}else j(!0);x(null);const k=(await se.getEnhancedStats(v)).data;Me(u,k),o(k),m(!1)}catch(u){console.error("Error fetching dashboard stats:",u),x(u.message||"Failed to fetch dashboard statistics")}finally{j(!1)}},[v]);c.useEffect(()=>{d(!0)},[d]);const n=c.useCallback(()=>{d(!1)},[d]);return{stats:r,loading:l,isStale:t,error:f,refresh:n}},He=()=>{var k,A,F,V,D,E,R,M,O,$,B,I,W,z,L,a,T,Y,G,X;const{sidebarOpen:s,toggleSidebar:r,closeSidebar:o}=ve(),[l,j]=c.useState({}),{stats:t,loading:m,error:f,isStale:x,refresh:v}=Oe(l),d=c.useCallback(i=>{j(i)},[]),n=i=>i?i.toLocaleString():"0",g=i=>i?`${i.toFixed(1)}%`:"0%",h=i=>i?`${i.toFixed(1)}m`:"0m",b=i=>i?`${i.toFixed(1)}t`:"0t",u=(i,w="SEK")=>i?new Intl.NumberFormat("sv-SE",{style:"currency",currency:w,minimumFractionDigits:0,maximumFractionDigits:0}).format(i):"0",p=(i,w="SEK")=>i?i>=1e6?`${(i/1e6).toFixed(1)}M ${w}`:i>=1e3?`${(i/1e3).toFixed(1)}K ${w}`:u(i,w):"0",S=(i,w=100)=>i?`${i.toFixed(1)}/${w}`:"0";return e.jsxs("div",{className:"min-h-screen bg-gray-50 dark:bg-gray-900 flex flex-col",children:[e.jsx(pe,{onToggleSidebar:r}),e.jsx(be,{isOpen:s,onClose:o}),e.jsx("main",{className:"flex-1 p-4 md:p-6 lg:p-8 lg:ml-64",children:e.jsxs("div",{className:"max-w-7xl mx-auto",children:[e.jsx("div",{className:"mb-8",children:e.jsx("div",{className:"flex justify-between items-center mb-4",children:e.jsxs("div",{children:[e.jsxs("div",{className:"flex items-center gap-3 mb-2",children:[e.jsx("h2",{className:"text-3xl font-bold text-gray-900 dark:text-white",children:"Forest Overview"}),x&&e.jsxs("div",{className:"flex items-center gap-1 text-xs text-blue-600 dark:text-blue-400 bg-blue-50 dark:bg-blue-900/30 px-2 py-1 rounded",children:[e.jsxs("svg",{className:"w-3 h-3 animate-spin",fill:"none",viewBox:"0 0 24 24",children:[e.jsx("circle",{className:"opacity-25",cx:"12",cy:"12",r:"10",stroke:"currentColor",strokeWidth:"4"}),e.jsx("path",{className:"opacity-75",fill:"currentColor",d:"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"})]}),"Updating..."]})]}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300",children:"Monitor your forests with real-time insights. Navigate to specialized dashboards for detailed analysis."})]})})}),e.jsx(ue,{onFiltersChange:d}),m?e.jsx("div",{className:"flex justify-center items-center py-12",children:e.jsx(H,{text:"Loading dashboard overview..."})}):f?e.jsxs("div",{className:"bg-red-50 border border-red-200 rounded-lg p-4 mb-8",children:[e.jsxs("p",{className:"text-red-600",children:["Error loading dashboard statistics: ",f]}),e.jsx("button",{onClick:v,className:"mt-2 text-red-600 hover:text-red-800 underline",children:"Try again"})]}):e.jsxs(e.Fragment,{children:[e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Core Metrics"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6",children:[e.jsx(q,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 3v1m0 16v1m9-9h-1M4 12H3m15.364 6.364l-.707-.707M6.343 6.343l-.707-.707m12.728 0l-.707.707M6.343 17.657l-.707.707M16 12a4 4 0 11-8 0 4 4 0 018 0z"})}),title:"Total Trees",value:n((k=t==null?void 0:t.overview)==null?void 0:k.totalTrees),color:"green"}),e.jsx(q,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"})}),title:"Survival Rate",value:g((A=t==null?void 0:t.overview)==null?void 0:A.survivalRate),color:"blue"}),e.jsx(q,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M7 12l3-3 3 3 4-4M8 21l4-4 4 4M3 4h18M4 4h16v12a1 1 0 01-1 1H5a1 1 0 01-1-1V4z"})}),title:"Avg Height",value:h((F=t==null?void 0:t.height)==null?void 0:F.average),color:"yellow"}),e.jsx(q,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M13 10V3L4 14h7v7l9-11h-7z"})}),title:"CO₂ Absorbed",value:b((V=t==null?void 0:t.co2)==null?void 0:V.totalAbsorption),color:"purple"})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsxs("div",{className:"flex justify-between items-center mb-4",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white",children:"Financial Summary"}),e.jsx(P,{to:"/dashboard/financial",className:"text-blue-600 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-300 font-medium text-sm",children:"View Full Financial Dashboard →"})]}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-6",children:[e.jsx(K,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 8c-1.657 0-3 .895-3 2s1.343 2 3 2 3 .895 3 2-1.343 2-3 2m0-8c1.11 0 2.08.402 2.599 1M12 8V7m0 1v8m0 0v1m0-1c-1.11 0-2.08-.402-2.599-1"})}),title:"Portfolio Value",value:p((E=(D=t==null?void 0:t.investor)==null?void 0:D.portfolio)==null?void 0:E.totalCurrentValue),subtitle:`${((M=(R=t==null?void 0:t.investor)==null?void 0:R.portfolio)==null?void 0:M.forestCount)||0} forests`,color:"green"}),e.jsx(K,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M13 7h8m0 0v8m0-8l-8 8-4-4-6 6"})}),title:"Average ROI",value:g(($=(O=t==null?void 0:t.investor)==null?void 0:O.roi)==null?void 0:$.averageROI),subtitle:`${p((I=(B=t==null?void 0:t.investor)==null?void 0:B.roi)==null?void 0:I.totalValueGain)} total gain`,color:"blue"})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsxs("div",{className:"flex justify-between items-center mb-4",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white",children:"Ecological Summary"}),e.jsx(P,{to:"/dashboard/ecological",className:"text-green-600 hover:text-green-800 dark:text-green-400 dark:hover:text-green-300 font-medium text-sm",children:"View Full Ecological Dashboard →"})]}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-6",children:[e.jsx(K,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4.318 6.318a4.5 4.5 0 000 6.364L12 20.364l7.682-7.682a4.5 4.5 0 00-6.364-6.364L12 7.636l-1.318-1.318a4.5 4.5 0 00-6.364 0z"})}),title:"Biodiversity Index",value:S((z=(W=t==null?void 0:t.manager)==null?void 0:W.biodiversity)==null?void 0:z.averageIndex),subtitle:`${n((a=(L=t==null?void 0:t.manager)==null?void 0:L.biodiversity)==null?void 0:a.totalSpecies)} species tracked`,color:"green"}),e.jsx(K,{icon:e.jsx("svg",{className:"h-6 w-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-2.5L13.732 4c-.77-.833-1.94-.833-2.71 0L4.104 16.5c-.77.833.192 2.5 1.732 2.5z"})}),title:"Trees at Risk",value:n((Y=(T=t==null?void 0:t.manager)==null?void 0:T.treesAtRisk)==null?void 0:Y.total),subtitle:`${n((X=(G=t==null?void 0:t.manager)==null?void 0:G.treesAtRisk)==null?void 0:X.critical)} critical`,color:"red"})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Detailed Analysis"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 gap-6",children:[e.jsx(P,{to:"/dashboard/financial",className:"block",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow-md hover:shadow-lg transition-shadow duration-200 p-6 border border-gray-200 dark:border-gray-700 hover:border-blue-300 dark:hover:border-blue-600",children:[e.jsxs("div",{className:"flex items-center mb-4",children:[e.jsx("div",{className:"p-2 bg-blue-100 dark:bg-blue-900 rounded-lg mr-4",children:e.jsx("svg",{className:"h-6 w-6 text-blue-600 dark:text-blue-400",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M13 7h8m0 0v8m0-8l-8 8-4-4-6 6"})})}),e.jsx("h4",{className:"text-lg font-semibold text-gray-900 dark:text-white",children:"Financial Dashboard"})]}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300 mb-4",children:"Detailed investment analytics including ROI, carbon credit revenue, maintenance costs, and species economic performance."}),e.jsxs("div",{className:"flex items-center text-blue-600 dark:text-blue-400 font-medium",children:["View Financial Analytics",e.jsx("svg",{className:"h-4 w-4 ml-2",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 5l7 7-7 7"})})]})]})}),e.jsx(P,{to:"/dashboard/ecological",className:"block",children:e.jsxs("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow-md hover:shadow-lg transition-shadow duration-200 p-6 border border-gray-200 dark:border-gray-700 hover:border-green-300 dark:hover:border-green-600",children:[e.jsxs("div",{className:"flex items-center mb-4",children:[e.jsx("div",{className:"p-2 bg-green-100 dark:bg-green-900 rounded-lg mr-4",children:e.jsx("svg",{className:"h-6 w-6 text-green-600 dark:text-green-400",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4.318 6.318a4.5 4.5 0 000 6.364L12 20.364l7.682-7.682a4.5 4.5 0 00-6.364-6.364L12 7.636l-1.318-1.318a4.5 4.5 0 00-6.364 0z"})})}),e.jsx("h4",{className:"text-lg font-semibold text-gray-900 dark:text-white",children:"Ecological Dashboard"})]}),e.jsx("p",{className:"text-gray-600 dark:text-gray-300 mb-4",children:"Comprehensive environmental impact analysis including biodiversity trends, ecological benefits, and conservation metrics."}),e.jsxs("div",{className:"flex items-center text-green-600 dark:text-green-400 font-medium",children:["View Ecological Analytics",e.jsx("svg",{className:"h-4 w-4 ml-2",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 5l7 7-7 7"})})]})]})})]})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Key Performance Indicators"}),e.jsxs("div",{className:"grid grid-cols-1 lg:grid-cols-2 xl:grid-cols-3 gap-6",children:[e.jsx("div",{className:"col-span-1",children:e.jsx(Le,{filters:l,dashboardData:t})}),e.jsx("div",{className:"col-span-1",children:e.jsx(Ve,{filters:l,dashboardData:t})}),e.jsx("div",{className:"col-span-1 xl:col-span-1",children:e.jsx(Fe,{filters:l,dashboardData:t})})]})]}),e.jsxs("div",{className:"mb-8",children:[e.jsx("h3",{className:"text-lg font-semibold text-gray-900 dark:text-white mb-4",children:"Recent Activity"}),e.jsx("div",{className:"bg-white dark:bg-gray-800 rounded-lg shadow-md p-6 border border-gray-200 dark:border-gray-700",children:e.jsxs("div",{className:"text-center py-8",children:[e.jsx("svg",{className:"h-12 w-12 text-gray-400 mx-auto mb-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 5H7a2 2 0 00-2 2v10a2 2 0 002 2h8a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-3 7h3m-3 4h3m-6-4h.01M9 16h.01"})}),e.jsx("p",{className:"text-gray-500 dark:text-gray-400",children:"Recent activities and alerts will appear here"})]})})]})]})})]})};export{He as OverviewDashboardPage}; diff --git a/frontend/dist/assets/Pagination-EPc_5dQ6.js b/frontend/dist/assets/Pagination-EPc_5dQ6.js new file mode 100644 index 0000000000..5fdd6c0c78 --- /dev/null +++ b/frontend/dist/assets/Pagination-EPc_5dQ6.js @@ -0,0 +1,70 @@ +import{d as o}from"./index-tEqMizXb.js";o.div` + font-size: 0.875rem; + color: #6b7280; +`;o.button` + padding: 0.5rem 0.75rem; + background: white; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + cursor: pointer; + transition: all 0.2s; + + &:hover:not(:disabled) { + background: #f9fafb; + border-color: #9ca3af; + } + + &:disabled { + opacity: 0.5; + cursor: not-allowed; + } + + &.active { + background: #10b981; + color: white; + border-color: #10b981; + } + + &:focus-visible { + outline: none; + box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.45); + } +`;o.nav` + padding: 1rem 1.5rem; + background: #f9fafb; + border-top: 1px solid #e5e7eb; + display: flex; + justify-content: space-between; + align-items: center; +`;o.ul` + display: flex; + gap: 0.5rem; + align-items: center; + list-style: none; + margin: 0; + padding: 0; +`;o.button` + padding: 0.5rem 0.75rem; + background: white; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + cursor: pointer; + transition: all 0.2s; + + &:hover:not(:disabled) { + background: #f9fafb; + border-color: #9ca3af; + } + + &:disabled { + opacity: 0.5; + cursor: not-allowed; + } + + &:focus-visible { + outline: none; + box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.45); + } +`; diff --git a/frontend/dist/assets/PasswordInput-8zSydSJy.js b/frontend/dist/assets/PasswordInput-8zSydSJy.js new file mode 100644 index 0000000000..c08ae3ecc6 --- /dev/null +++ b/frontend/dist/assets/PasswordInput-8zSydSJy.js @@ -0,0 +1 @@ +import{j as e,r as g}from"./index-tEqMizXb.js";const y=({label:r,id:a,error:o,required:d=!1,children:n,className:l=""})=>e.jsxs("div",{className:l,children:[e.jsxs("label",{htmlFor:a,className:"block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2",children:[r,d&&e.jsx("span",{className:"text-red-500 ml-1",children:"*"})]}),n,o&&e.jsx("p",{id:`${a}-error`,className:"mt-1 text-sm text-red-600 dark:text-red-400",children:o})]}),w=({id:r,name:a,value:o,onChange:d,onBlur:n,className:l="",placeholder:c="",disabled:i=!1,error:s="",label:x="Password",required:m=!1,...h})=>{const[t,p]=g.useState(!1),u=()=>{p(!t)};return e.jsxs("div",{className:"relative",children:[e.jsxs("label",{htmlFor:r,className:"form-label block mb-2 text-gray-700 dark:text-gray-300",children:[x,m&&e.jsx("span",{className:"text-red-500 ml-1",children:"*"})]}),e.jsx("input",{type:t?"text":"password",id:r,name:a,value:o,onChange:d,onBlur:n,className:`w-full px-3 py-2 pr-10 border rounded-lg focus:outline-none focus:ring-2 focus:ring-green-500 focus:border-transparent bg-white dark:bg-gray-700 text-gray-900 dark:text-white placeholder-gray-500 dark:placeholder-gray-400 ${s?"border-red-300 dark:border-red-600":"border-gray-300 dark:border-gray-600"} ${l}`,placeholder:c,disabled:i,"aria-describedby":s?`${r}-error`:void 0,"aria-invalid":!!s,...h}),e.jsx("button",{type:"button",onClick:u,className:"absolute inset-y-0 right-0 pr-3 flex items-center",disabled:i,"aria-label":t?"Hide password":"Show password","aria-pressed":t,children:t?e.jsx(j,{className:"h-5 w-5 text-gray-400"}):e.jsx(k,{className:"h-5 w-5 text-gray-400"})}),s&&e.jsx("p",{id:`${r}-error`,className:"mt-2 text-sm text-red-600 dark:text-red-400",children:s})]})},k=({className:r})=>e.jsxs("svg",{className:r,fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:[e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M15 12a3 3 0 11-6 0 3 3 0 016 0z"}),e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z"})]}),j=({className:r})=>e.jsx("svg",{className:r,fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M13.875 18.825A10.05 10.05 0 0112 19c-4.478 0-8.268-2.943-9.543-7a9.97 9.97 0 011.563-3.029m5.858.908a3 3 0 114.243 4.243M9.878 9.878l4.242 4.242M9.878 9.878L3 3m6.878 6.878L21 21"})});export{y as F,w as P}; diff --git a/frontend/dist/assets/PieChart-DRbmPJpB.js b/frontend/dist/assets/PieChart-DRbmPJpB.js new file mode 100644 index 0000000000..284414e3dd --- /dev/null +++ b/frontend/dist/assets/PieChart-DRbmPJpB.js @@ -0,0 +1 @@ +import{r as i,c as U}from"./index-tEqMizXb.js";import{h as y,a4 as Z,a5 as be,a6 as Pe,a7 as I,a8 as Ee,a9 as Oe,aa as se,ab as Se,ac as Te,I as S,ad as xe,ae as Re,af as De,ag as Ce,ah as Ie,ai as je,aj as we,ak as Ke,p as ce,al as Me,am as ke,O as k,v as ue,x as Q,an as Ne,q as X,y as _e,C as de,G as Le,D as N,ao as ae,ap as W,aq as F,ar as Be,as as V,K as $e,M as Ge,at as We,N as te,P as Ve,Q as ze,U as Fe,V as qe,W as Ue,Z as Ze,_ as Qe,S as Xe,a1 as Ye,au as He,z as Je,a2 as ea,a3 as ve,u as aa,av as ta,aw as re,ax as ra,ay as na,az as ia,aA as la,aB as oa}from"./EnhancedStatCard-CVgqadrW.js";var me=e=>e.graphicalItems.polarItems,sa=y([Z,be],Pe),Y=y([me,I,sa],Ee),ca=y([Y],Oe),H=y([ca,se],Se),ua=y([H,I,Y],Te),da=y([H,I,Y],(e,a,t)=>t.length>0?e.flatMap(r=>t.flatMap(n=>{var l,o=S(r,(l=a.dataKey)!==null&&l!==void 0?l:n.dataKey);return{value:o,errorDomain:[]}})).filter(Boolean):(a==null?void 0:a.dataKey)!=null?e.map(r=>({value:S(r,a.dataKey),errorDomain:[]})):e.map(r=>({value:r,errorDomain:[]}))),ne=()=>{},va=y([I,xe,ne,da,ne],Re),fe=y([I,De,H,ua,Ce,Z,va],Ie),ma=y([fe,I,je],we);y([I,fe,ma,Z],Ke);function ie(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);a&&(r=r.filter(function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable})),t.push.apply(t,r)}return t}function le(e){for(var a=1;aa,ya=[],ee=(e,a,t)=>(t==null?void 0:t.length)===0?ya:t,pe=y([se,J,ee],(e,a,t)=>{var{chartData:r}=e,n;if((a==null?void 0:a.data)!=null&&a.data.length>0?n=a.data:n=r,(!n||!n.length)&&t!=null&&(n=t.map(l=>le(le({},a.presentationProps),l.props))),n!=null)return n}),ha=y([pe,J,ee],(e,a,t)=>{if(e!=null)return e.map((r,n)=>{var l,o=S(r,a.nameKey,a.name),d;return t!=null&&(l=t[n])!==null&&l!==void 0&&(l=l.props)!==null&&l!==void 0&&l.fill?d=t[n].props.fill:typeof r=="object"&&r!=null&&"fill"in r?d=r.fill:d=a.fill,{value:ce(o,a.dataKey),color:d,payload:r,type:a.legendType}})}),Aa=y([me,J],(e,a)=>{if(e.some(t=>t.type==="pie"&&a.dataKey===t.dataKey&&a.data===t.data))return a}),ba=y([pe,Aa,ee,Me],(e,a,t,r)=>{if(!(a==null||e==null))return ka({offset:r,pieSettings:a,displayedData:e,cells:t})}),Pa=["onMouseEnter","onClick","onMouseLeave"];function Ea(e,a){if(e==null)return{};var t,r,n=Oa(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(r=0;rk(e,!1),[e]),t=i.useMemo(()=>ue(e.children,de),[e.children]),r=i.useMemo(()=>({name:e.name,nameKey:e.nameKey,tooltipType:e.tooltipType,data:e.data,dataKey:e.dataKey,cx:e.cx,cy:e.cy,startAngle:e.startAngle,endAngle:e.endAngle,minAngle:e.minAngle,paddingAngle:e.paddingAngle,innerRadius:e.innerRadius,outerRadius:e.outerRadius,cornerRadius:e.cornerRadius,legendType:e.legendType,fill:e.fill,presentationProps:a}),[e.cornerRadius,e.cx,e.cy,e.data,e.dataKey,e.endAngle,e.innerRadius,e.minAngle,e.name,e.nameKey,e.outerRadius,e.paddingAngle,e.startAngle,e.tooltipType,e.legendType,e.fill,a]),n=Q(l=>ha(l,r,t));return i.createElement(Ne,{legendPayload:n})}function Ra(e){var{dataKey:a,nameKey:t,sectors:r,stroke:n,strokeWidth:l,fill:o,name:d,hide:h,tooltipType:v}=e;return{dataDefinedOnItem:r==null?void 0:r.map(f=>f.tooltipPayload),positions:r==null?void 0:r.map(f=>f.tooltipPosition),settings:{stroke:n,strokeWidth:l,fill:o,dataKey:a,nameKey:t,name:ce(d,a),hide:h,type:v,color:o,unit:""}}}var Da=(e,a)=>e>a?"start":etypeof a=="function"?a(e):V(a,t,t*.8),Ia=(e,a,t)=>{var{top:r,left:n,width:l,height:o}=a,d=Be(l,o),h=n+V(e.cx,l,l/2),v=r+V(e.cy,o,o/2),f=V(e.innerRadius,d,0),A=Ca(t,e.outerRadius,d),s=e.maxRadius||Math.sqrt(l*l+o*o)/2;return{cx:h,cy:v,innerRadius:f,outerRadius:A,maxRadius:s}},ja=(e,a)=>{var t=W(a-e),r=Math.min(Math.abs(a-e),360);return t*r},wa=(e,a)=>{if(i.isValidElement(e))return i.cloneElement(e,a);if(typeof e=="function")return e(a);var t=U("recharts-pie-label-line",typeof e!="boolean"?e.className:"");return i.createElement(Ye,C({},a,{type:"linear",className:t}))},Ka=(e,a,t)=>{if(i.isValidElement(e))return i.cloneElement(e,a);var r=t;if(typeof e=="function"&&(r=e(a),i.isValidElement(r)))return r;var n=U("recharts-pie-label-text",typeof e!="boolean"&&typeof e!="function"?e.className:"");return i.createElement(He,C({},a,{alignmentBaseline:"middle",className:n}),r)};function Ma(e){var{sectors:a,props:t,showLabels:r}=e,{label:n,labelLine:l,dataKey:o}=t;if(!r||!n||!a)return null;var d=k(t,!1),h=k(n,!1),v=k(l,!1),f=typeof n=="object"&&"offsetRadius"in n&&n.offsetRadius||20,A=a.map((s,g)=>{var c=(s.startAngle+s.endAngle)/2,m=F(s.cx,s.cy,s.outerRadius+f,c),E=u(u(u(u({},d),s),{},{stroke:"none"},h),{},{index:g,textAnchor:Da(m.x,s.cx)},m),x=u(u(u(u({},d),s),{},{fill:"none",stroke:s.fill},v),{},{index:g,points:[F(s.cx,s.cy,s.outerRadius,c),m],key:"line"});return i.createElement(N,{key:"label-".concat(s.startAngle,"-").concat(s.endAngle,"-").concat(s.midAngle,"-").concat(g)},l&&wa(l,x),Ka(n,E,S(s,o)))});return i.createElement(N,{className:"recharts-pie-labels"},A)}function ge(e){var{sectors:a,activeShape:t,inactiveShape:r,allOtherPieProps:n,showLabels:l}=e,o=Q(Ve),{onMouseEnter:d,onClick:h,onMouseLeave:v}=n,f=Ea(n,Pa),A=ze(d,n.dataKey),s=Fe(v),g=qe(h,n.dataKey);return a==null?null:i.createElement(i.Fragment,null,a.map((c,m)=>{if((c==null?void 0:c.startAngle)===0&&(c==null?void 0:c.endAngle)===0&&a.length!==1)return null;var E=t&&String(m)===o,x=o?r:null,w=E?t:x,R=u(u({},c),{},{stroke:c.stroke,tabIndex:-1,[Ze]:m,[Ue]:n.dataKey});return i.createElement(N,C({tabIndex:-1,className:"recharts-pie-sector"},Qe(f,c,m),{onMouseEnter:A(c,m),onMouseLeave:s(c,m),onClick:g(c,m),key:"sector-".concat(c==null?void 0:c.startAngle,"-").concat(c==null?void 0:c.endAngle,"-").concat(c.midAngle,"-").concat(m)}),i.createElement(Xe,C({option:w,isActive:E,shapeType:"sector"},R)))}),i.createElement(Ma,{sectors:a,props:n,showLabels:l}))}function ka(e){var a,{pieSettings:t,displayedData:r,cells:n,offset:l}=e,{cornerRadius:o,startAngle:d,endAngle:h,dataKey:v,nameKey:f,tooltipType:A}=t,s=Math.abs(t.minAngle),g=ja(d,h),c=Math.abs(g),m=r.length<=1?0:(a=t.paddingAngle)!==null&&a!==void 0?a:0,E=r.filter(p=>S(p,v,0)!==0).length,x=(c>=360?E:E-1)*m,w=c-E*s-x,R=r.reduce((p,b)=>{var P=S(b,v,0);return p+(ae(P)?P:0)},0),D;if(R>0){var K;D=r.map((p,b)=>{var P=S(p,v,0),j=S(p,f,b),O=Ia(t,l,p),_=(ae(P)?P:0)/R,T,M=u(u({},p),n&&n[b]&&n[b].props);b?T=K.endAngle+W(g)*m*(P!==0?1:0):T=d;var L=T+W(g)*((P!==0?s:0)+_*w),B=(T+L)/2,$=(O.innerRadius+O.outerRadius)/2,G=[{name:j,value:P,payload:M,dataKey:v,type:A}],Ae=F(O.cx,O.cy,$,B);return K=u(u(u(u({},t.presentationProps),{},{percent:_,cornerRadius:o,name:j,tooltipPayload:G,midAngle:B,middleRadius:$,tooltipPosition:Ae},M),O),{},{value:S(p,v),startAngle:T,endAngle:L,payload:M,paddingAngle:W(g)*m}),K})}return D}function Na(e){var{props:a,previousSectorsRef:t}=e,{sectors:r,isAnimationActive:n,animationBegin:l,animationDuration:o,animationEasing:d,activeShape:h,inactiveShape:v,onAnimationStart:f,onAnimationEnd:A}=a,s=$e(a,"recharts-pie-"),g=t.current,[c,m]=i.useState(!0),E=i.useCallback(()=>{typeof A=="function"&&A(),m(!1)},[A]),x=i.useCallback(()=>{typeof f=="function"&&f(),m(!0)},[f]);return i.createElement(Ge,{begin:l,duration:o,isActive:n,easing:d,from:{t:0},to:{t:1},onAnimationStart:x,onAnimationEnd:E,key:s},w=>{var{t:R}=w,D=[],K=r&&r[0],p=K.startAngle;return r.forEach((b,P)=>{var j=g&&g[P],O=P>0?We(b,"paddingAngle",0):0;if(j){var _=te(j.endAngle-j.startAngle,b.endAngle-b.startAngle),T=u(u({},b),{},{startAngle:p+O,endAngle:p+_(R)+O});D.push(T),p=T.endAngle}else{var{endAngle:M,startAngle:L}=b,B=te(0,M-L),$=B(R),G=u(u({},b),{},{startAngle:p+O,endAngle:p+$+O});D.push(G),p=G.endAngle}}),t.current=D,i.createElement(N,null,i.createElement(ge,{sectors:D,activeShape:h,inactiveShape:v,allOtherPieProps:a,showLabels:!c}))})}function _a(e){var{sectors:a,isAnimationActive:t,activeShape:r,inactiveShape:n}=e,l=i.useRef(null),o=l.current;return t&&a&&a.length&&(!o||o!==a)?i.createElement(Na,{props:e,previousSectorsRef:l}):i.createElement(ge,{sectors:a,activeShape:r,inactiveShape:n,allOtherPieProps:e,showLabels:!0})}function La(e){var{hide:a,className:t,rootTabIndex:r}=e,n=U("recharts-pie",t);return a?null:i.createElement(N,{tabIndex:r,className:n},i.createElement(_a,e))}var ye={animationBegin:400,animationDuration:1500,animationEasing:"ease",cx:"50%",cy:"50%",dataKey:"value",endAngle:360,fill:"#808080",hide:!1,innerRadius:0,isAnimationActive:!Le.isSsr,labelLine:!0,legendType:"rect",minAngle:0,nameKey:"name",outerRadius:"80%",paddingAngle:0,rootTabIndex:0,startAngle:0,stroke:"#fff"};function Ba(e){var a=X(e,ye),t=i.useMemo(()=>ue(e.children,de),[e.children]),r=k(a,!1),n=i.useMemo(()=>({name:a.name,nameKey:a.nameKey,tooltipType:a.tooltipType,data:a.data,dataKey:a.dataKey,cx:a.cx,cy:a.cy,startAngle:a.startAngle,endAngle:a.endAngle,minAngle:a.minAngle,paddingAngle:a.paddingAngle,innerRadius:a.innerRadius,outerRadius:a.outerRadius,cornerRadius:a.cornerRadius,legendType:a.legendType,fill:a.fill,presentationProps:r}),[a.cornerRadius,a.cx,a.cy,a.data,a.dataKey,a.endAngle,a.innerRadius,a.minAngle,a.name,a.nameKey,a.outerRadius,a.paddingAngle,a.startAngle,a.tooltipType,a.legendType,a.fill,r]),l=Q(o=>ba(o,n,t));return i.createElement(i.Fragment,null,i.createElement(_e,{fn:Ra,args:u(u({},a),{},{sectors:l})}),i.createElement(La,C({},a,{sectors:l})))}class he extends i.PureComponent{constructor(){super(...arguments),z(this,"id",Je("recharts-pie-"))}render(){return i.createElement(i.Fragment,null,i.createElement(ke,{data:this.props.data,dataKey:this.props.dataKey,hide:this.props.hide,angleAxisId:0,radiusAxisId:0,stackId:void 0,barSize:void 0,type:"pie"}),i.createElement(xa,this.props),i.createElement(Ba,this.props),this.props.children)}}z(he,"displayName","Pie");z(he,"defaultProps",ye);var $a=["axis","item"],Ha=i.forwardRef((e,a)=>i.createElement(ea,{chartName:"BarChart",defaultTooltipEventType:"axis",validateTooltipEventTypes:$a,tooltipPayloadSearcher:ve,categoricalChartProps:e,ref:a}));function Ga(e){var a=aa();return i.useEffect(()=>{a(ta(e))},[a,e]),null}var Wa=["width","height","layout"];function q(){return q=Object.assign?Object.assign.bind():function(e){for(var a=1;a{var t=X(e,Qa);return i.createElement(Ua,{chartName:"PieChart",defaultTooltipEventType:"item",validateTooltipEventTypes:Za,tooltipPayloadSearcher:ve,categoricalChartProps:t,ref:a})});export{Ha as B,Ja as P,he as a}; diff --git a/frontend/dist/assets/RegisterPage-DEfT8Iyu.js b/frontend/dist/assets/RegisterPage-DEfT8Iyu.js new file mode 100644 index 0000000000..25ebaf6bfb --- /dev/null +++ b/frontend/dist/assets/RegisterPage-DEfT8Iyu.js @@ -0,0 +1,27 @@ +import{j as r,d as c,r as l,u as b,L as x}from"./index-tEqMizXb.js";import{F as u,P as h}from"./PasswordInput-8zSydSJy.js";import"./vendor-BtP0CW_r.js";const j=c.div` + width: 100%; + height: 4px; + background-color: #e5e7eb; + border-radius: 2px; + overflow: hidden; + margin-top: 0.5rem; +`,y=c.div` + height: 100%; + background-color: ${e=>e.strength==="weak"?"#ef4444":e.strength==="medium"?"#f59e0b":e.strength==="strong"?"#10b981":"#e5e7eb"}; + width: ${e=>e.strength==="weak"?"33%":e.strength==="medium"?"66%":e.strength==="strong"?"100%":"0%"}; + transition: all 0.3s ease; +`,P=e=>{switch(e){case"weak":return"Weak password";case"medium":return"Medium strength password";case"strong":return"Strong password";default:return""}},v=({strength:e,show:s})=>s?r.jsxs("div",{className:"mt-2",children:[r.jsx(j,{children:r.jsx(y,{strength:e})}),r.jsx("p",{className:`text-xs mt-1 ${e==="weak"?"text-red-600":e==="medium"?"text-yellow-600":e==="strong"?"text-green-600":"text-gray-500"}`,children:P(e)})]}):null,N=e=>l.useMemo(()=>{if(!e)return{strength:"none",score:0};let s=0;return e.length>=8&&(s+=1),/[a-z]/.test(e)&&(s+=1),/[A-Z]/.test(e)&&(s+=1),/[0-9]/.test(e)&&(s+=1),/[^A-Za-z0-9]/.test(e)&&(s+=1),s<=2?{strength:"weak",score:s}:s<=3?{strength:"medium",score:s}:{strength:"strong",score:s}},[e]),S=()=>{const[e,s]=l.useState({email:"",password:"",confirmPassword:""}),[a,o]=l.useState({}),[i,d]=l.useState(!1),{register:f}=b(),g=N(e.password),w=()=>{const t={};return e.email?/\S+@\S+\.\S+/.test(e.email)||(t.email="Email is invalid"):t.email="Email is required",e.password?e.password.length<6?t.password="Password must be at least 6 characters":g.strength==="weak"&&(t.password="Password is too weak. Include uppercase, lowercase, numbers, and special characters."):t.password="Password is required",e.confirmPassword?e.password!==e.confirmPassword&&(t.confirmPassword="Passwords do not match"):t.confirmPassword="Please confirm your password",o(t),Object.keys(t).length===0};return{formData:e,errors:a,isSubmitting:i,passwordStrength:g,handleInputChange:t=>{const{name:n,value:p}=t.target;s(m=>({...m,[n]:p})),a[n]&&o(m=>({...m,[n]:""}))},handleSubmit:async t=>{if(t.preventDefault(),!!w()){d(!0);try{const n=await f(e.email,e.password,e.confirmPassword);n.success||o({general:n.error})}catch{o({general:"An unexpected error occurred. Please try again."})}finally{d(!1)}}}}},k=()=>{const{formData:e,errors:s,isSubmitting:a,passwordStrength:o,handleInputChange:i,handleSubmit:d}=S();return r.jsxs("form",{onSubmit:d,className:"space-y-6",children:[s.general&&r.jsx("div",{className:"bg-red-50 border border-red-200 text-red-700 px-4 py-3 rounded-lg",children:s.general}),r.jsx(u,{label:"Email address",id:"email",error:s.email,required:!0,children:r.jsx("input",{type:"email",id:"email",name:"email",value:e.email,onChange:i,className:`w-full px-3 py-2 border rounded-lg focus:outline-none focus:ring-2 focus:ring-green-500 focus:border-transparent ${s.email?"border-red-300":"border-gray-300"}`,placeholder:"Enter your email",disabled:a})}),r.jsxs(u,{label:"Password",id:"password",error:s.password,required:!0,children:[r.jsx(h,{id:"password",name:"password",value:e.password,onChange:i,error:s.password,placeholder:"Create a strong password",disabled:a}),r.jsx(v,{strength:o.strength,show:e.password.length>0})]}),r.jsx(u,{label:"Confirm password",id:"confirmPassword",error:s.confirmPassword,required:!0,children:r.jsx(h,{id:"confirmPassword",name:"confirmPassword",value:e.confirmPassword,onChange:i,error:s.confirmPassword,placeholder:"Confirm your password",disabled:a})}),r.jsx(C,{disabled:a}),r.jsx(F,{isSubmitting:a}),r.jsx(I,{})]})},C=({disabled:e})=>r.jsxs("div",{className:"flex items-center",children:[r.jsx("input",{id:"terms",name:"terms",type:"checkbox",required:!0,className:"h-4 w-4 text-green-600 focus:ring-green-500 border-gray-300 rounded",disabled:e}),r.jsxs("label",{htmlFor:"terms",className:"ml-2 block text-sm text-gray-900",children:["I agree to the"," ",r.jsx("a",{href:"#",className:"font-medium text-green-600 hover:text-green-500",children:"Terms of Service"})," ","and"," ",r.jsx("a",{href:"#",className:"font-medium text-green-600 hover:text-green-500",children:"Privacy Policy"})]})]}),F=({isSubmitting:e})=>r.jsx("button",{type:"submit",disabled:e,className:"w-full flex justify-center py-2 px-4 border border-transparent rounded-lg shadow-sm text-sm font-medium text-white bg-green-600 hover:bg-green-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-green-500 disabled:opacity-50 disabled:cursor-not-allowed",children:e?r.jsxs("div",{className:"flex items-center",children:[r.jsxs("svg",{className:"animate-spin -ml-1 mr-3 h-5 w-5 text-white",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24",children:[r.jsx("circle",{className:"opacity-25",cx:"12",cy:"12",r:"10",stroke:"currentColor",strokeWidth:"4"}),r.jsx("path",{className:"opacity-75",fill:"currentColor",d:"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"})]}),"Creating account..."]}):"Create account"}),I=()=>r.jsx("div",{className:"text-center",children:r.jsxs("p",{className:"text-sm text-gray-600",children:["Already have an account?"," ",r.jsx(x,{to:"/login",className:"font-medium text-green-600 hover:text-green-500",children:"Sign in"})]})}),E=c.div` + min-height: 100vh; + background: linear-gradient(135deg, #f0fdf4 0%, #dcfce7 100%); + display: flex; + align-items: center; + justify-content: center; + padding: 1rem; +`,R=c.div` + background: white; + border-radius: 1rem; + box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04); + padding: 2rem; + width: 100%; + max-width: 450px; +`,M=()=>r.jsx(E,{children:r.jsxs(R,{children:[r.jsx(q,{}),r.jsx(k,{})]})}),q=()=>r.jsxs("div",{className:"text-center mb-8",children:[r.jsx(x,{to:"/",className:"inline-block mb-6",children:r.jsx("h1",{className:"text-3xl font-bold text-green-600",children:"Nanwa"})}),r.jsx("h2",{className:"text-2xl font-bold text-gray-900 mb-2",children:"Create your account"}),r.jsx("p",{className:"text-gray-600",children:"Join Nanwa to start monitoring your trees"})]});export{M as RegisterPage}; diff --git a/frontend/dist/assets/TreeDetailModal-PvfFs-NG.js b/frontend/dist/assets/TreeDetailModal-PvfFs-NG.js new file mode 100644 index 0000000000..3592ca3463 --- /dev/null +++ b/frontend/dist/assets/TreeDetailModal-PvfFs-NG.js @@ -0,0 +1,207 @@ +import{d as o,r as x,m as c,j as r,o as j}from"./index-tEqMizXb.js";import{u as $}from"./useKeyboardNavigation-DT9lybaQ.js";const n=()=>{const e=document.documentElement.classList.contains("dark");return{containerBg:e?"#1f2937":"#ffffff",containerBorder:e?"#374151":"#e5e7eb",containerShadow:e?"0 1px 3px 0 rgba(0, 0, 0, 0.3), 0 1px 2px 0 rgba(0, 0, 0, 0.2)":"0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06)",headerBg:e?"#374151":"#f9fafb",headerBorder:e?"#4b5563":"#e5e7eb",headerText:e?"#f9fafb":"#111827",cellText:e?"#e5e7eb":"#111827",cellBorder:e?"#4b5563":"#f3f4f6",cellHover:e?"#374151":"#f9fafb",cellHeaderText:e?"#d1d5db":"#374151",cellHeaderHover:e?"#4b5563":"#f3f4f6",mutedText:e?"#9ca3af":"#6b7280",sortIcon:e?"#6b7280":"#9ca3af"}};o.div` + background: ${()=>n().containerBg}; + border-radius: 0.75rem; + box-shadow: ${()=>n().containerShadow}; + border: 1px solid ${()=>n().containerBorder}; + overflow: hidden; + transition: background-color 0.2s ease, border-color 0.2s ease; +`;o.div` + background: ${()=>n().headerBg}; + padding: 1.5rem; + border-bottom: 1px solid ${()=>n().headerBorder}; + display: flex; + justify-content: space-between; + align-items: center; + transition: background-color 0.2s ease, border-color 0.2s ease; +`;o.h3` + font-size: 1.125rem; + font-weight: 600; + color: ${()=>n().headerText}; + margin: 0; + transition: color 0.2s ease; +`;const M=o.table` + width: 100%; + border-collapse: collapse; +`;o.thead` + background: ${()=>n().headerBg}; + border-bottom: 1px solid ${()=>n().headerBorder}; + transition: background-color 0.2s ease, border-color 0.2s ease; +`;const g=o.tr` + border-bottom: 1px solid ${()=>n().cellBorder}; + transition: background-color 0.2s ease, border-color 0.2s ease; + + &:hover { + background: ${()=>n().cellHover}; + } + + &:last-child { + border-bottom: none; + } +`,h=o.th` + padding: 0.75rem 1rem; + text-align: left; + font-size: 0.875rem; + font-weight: 600; + color: ${()=>n().cellHeaderText}; + cursor: pointer; + user-select: none; + transition: background-color 0.2s ease, color 0.2s ease; + + &:hover { + background: ${()=>n().cellHeaderHover}; + } +`,m=o.td` + padding: 0.75rem 1rem; + font-size: 0.875rem; + color: ${()=>n().cellText}; + vertical-align: top; + transition: color 0.2s ease; +`;o.div` + text-align: center; + padding: 3rem; + color: ${()=>n().mutedText}; + transition: color 0.2s ease; +`;o.span` + margin-left: 0.5rem; + font-size: 0.75rem; + color: ${()=>n().sortIcon}; + transition: color 0.2s ease; +`;const D=[{date:"2024-01-15",height:1.5,diameter:7.3,health:"healthy"},{date:"2024-02-15",height:1.6,diameter:7.4,health:"healthy"},{date:"2024-03-15",height:1.7,diameter:7.5,health:"healthy"},{date:"2024-04-15",height:1.8,diameter:7.6,health:"healthy"},{date:"2024-05-15",height:1.9,diameter:7.7,health:"healthy"},{date:"2024-06-15",height:2,diameter:7.8,health:"healthy"},{date:"2024-07-15",height:2.1,diameter:7.9,health:"healthy"},{date:"2024-08-15",height:2.2,diameter:8,health:"healthy"},{date:"2024-09-15",height:2.3,diameter:8.1,health:"healthy"},{date:"2024-10-15",height:2.4,diameter:8.2,health:"healthy"}],y=(e,t)=>{const[a,s]=x.useState([]);return x.useEffect(()=>{e&&t&&s(D)},[e,t]),a},v=o.div` + position: fixed; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: rgba(0, 0, 0, 0.5); + display: flex; + align-items: center; + justify-content: center; + z-index: 10000; + padding: 1rem; + outline: none; +`,w=o.div` + background: ${e=>e.$isDarkMode?"#1f2937":"white"}; + border-radius: 0.75rem; + max-width: 800px; + width: 100%; + max-height: 90vh; + overflow-y: auto; + box-shadow: 0 25px 50px -12px rgba(0, 0, 0, 0.25); + position: relative; + outline: none; +`,T=o.div` + display: flex; + justify-content: space-between; + align-items: center; + padding: 1.5rem; + border-bottom: 1px solid ${e=>e.$isDarkMode?"#374151":"#e5e7eb"}; + background: ${e=>e.$isDarkMode?"#111827":"#f9fafb"}; + border-radius: 0.75rem 0.75rem 0 0; +`,I=o.h2` + font-size: 1.5rem; + font-weight: 700; + color: ${e=>e.$isDarkMode?"#f9fafb":"#111827"}; + margin: 0; +`,H=o.button` + padding: 0.5rem; + background: transparent; + border: none; + border-radius: 0.375rem; + cursor: pointer; + color: ${e=>e.$isDarkMode?"#9ca3af":"#6b7280"}; + transition: all 0.2s; + + &:hover { + background: ${e=>e.$isDarkMode?"#374151":"#f3f4f6"}; + color: ${e=>e.$isDarkMode?"#f9fafb":"#111827"}; + } + + &:focus-visible { + outline: none; + box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.45); + } +`,B=o.div` + padding: 1.5rem; +`,C=()=>r.jsx("svg",{className:"w-6 h-6",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:r.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M6 18L18 6M6 6l12 12"})}),L=({isOpen:e,onClose:t,title:a,children:s,containerRef:d})=>{const{isDarkMode:l}=c(),k=f=>{f.target===f.currentTarget&&t()};return e?r.jsx(v,{onClick:k,role:"presentation",children:r.jsxs(w,{ref:d,role:"dialog","aria-modal":"true","aria-labelledby":"modal-title",tabIndex:"-1",$isDarkMode:l,children:[r.jsxs(T,{$isDarkMode:l,children:[r.jsx(I,{id:"modal-title",$isDarkMode:l,children:a}),r.jsx(H,{onClick:t,"aria-label":"Close modal",type:"button",$isDarkMode:l,children:r.jsx(C,{})})]}),r.jsx(B,{children:s})]})}):null},z=o.div` + background: ${e=>e.$isDarkMode?"#374151":"#f9fafb"}; + border-radius: 0.75rem; + padding: 1.5rem; + border: 1px solid ${e=>e.$isDarkMode?"#4b5563":"#e5e7eb"}; +`,S=o.h3` + font-size: 1.125rem; + font-weight: 600; + color: ${e=>e.$isDarkMode?"#f9fafb":"#111827"}; + margin: 0 0 1rem 0; +`,E=o.div` + display: flex; + justify-content: space-between; + align-items: center; + padding: 0.5rem 0; + border-bottom: 1px solid ${e=>e.$isDarkMode?"#4b5563":"#e5e7eb"}; + + &:last-child { + border-bottom: none; + } +`,F=o.span` + color: ${e=>e.$isDarkMode?"#9ca3af":"#6b7280"}; + font-size: 0.875rem; +`,P=o.span` + font-weight: 500; + color: ${e=>e.$isDarkMode?"#f9fafb":"#111827"}; + font-size: 0.875rem; +`,u=({title:e,children:t})=>{const{isDarkMode:a}=c();return r.jsxs(z,{$isDarkMode:a,children:[r.jsx(S,{$isDarkMode:a,children:e}),t]})},i=({label:e,children:t})=>{const{isDarkMode:a}=c();return r.jsxs(E,{$isDarkMode:a,children:[r.jsxs(F,{$isDarkMode:a,children:[e,":"]}),r.jsx(P,{$isDarkMode:a,children:t})]})},G=o.span` + padding: 0.25rem 0.75rem; + border-radius: 9999px; + font-size: 0.75rem; + font-weight: 500; + + ${e=>{switch(e.health){case"healthy":return"background: #d1fae5; color: #065f46;";case"warning":return"background: #fef3c7; color: #92400e;";case"critical":return"background: #fee2e2; color: #991b1b;";default:return"background: #f3f4f6; color: #374151;"}}} +`,p=({health:e,children:t})=>r.jsx(G,{health:e,children:t||e}),N=({tree:e})=>r.jsxs(u,{title:"Basic Information",children:[r.jsx(i,{label:"Tree ID",children:e.name}),r.jsx(i,{label:"Species",children:e.species}),r.jsxs(i,{label:"Current Height",children:[e.height,"m"]}),r.jsx(i,{label:"Health Status",children:r.jsx(p,{health:e.health,children:e.health})}),r.jsx(i,{label:"Planted Date",children:"March 15, 2023"})]}),R=({tree:e})=>r.jsxs(u,{title:"Location & Contract",children:[r.jsx(i,{label:"Latitude",children:e.lat.toFixed(6)}),r.jsx(i,{label:"Longitude",children:e.lng.toFixed(6)}),r.jsxs(i,{label:"Forest",children:["Forest ",e.id<=4?"A":"B"]}),r.jsx(i,{label:"Contract Status",children:r.jsx("span",{className:"px-2 py-1 bg-green-100 text-green-800 rounded-full text-xs",children:"Active"})}),r.jsx(i,{label:"Last Inspection",children:"Jan 15, 2024"})]}),A=o.div` + margin-bottom: 2rem; +`,V=o.h3` + font-size: 1.125rem; + font-weight: 600; + color: ${e=>e.$isDarkMode?"#f9fafb":"#111827"}; + margin: 0 0 1rem 0; +`,W=o(M)` + border-radius: 0.5rem; + overflow: hidden; +`,J=o.thead``,K=({measurementHistory:e})=>{const{isDarkMode:t}=c(),a=[...e].sort((s,d)=>new Date(d.date)-new Date(s.date));return r.jsxs(A,{children:[r.jsx(V,{$isDarkMode:t,children:"Measurement History (Last 10 Entries)"}),r.jsxs(W,{children:[r.jsx(J,{children:r.jsxs(g,{children:[r.jsx(h,{children:"Date"}),r.jsx(h,{children:"Height (m)"}),r.jsx(h,{children:"Diameter (cm)"}),r.jsx(h,{children:"Health"})]})}),r.jsx("tbody",{children:a.slice(0,10).map((s,d)=>r.jsxs(g,{children:[r.jsx(m,{children:j(s.date)}),r.jsx(m,{children:s.height}),r.jsx(m,{children:s.diameter}),r.jsx(m,{children:r.jsx(p,{health:s.health,children:s.health})})]},d))})]})]})},q=o.div` + display: grid; + grid-template-columns: repeat(auto-fill, minmax(150px, 1fr)); + gap: 1rem; + margin-top: 1rem; +`,O=o.div` + background: ${e=>e.$isDarkMode?"#374151":"#f9fafb"}; + border-radius: 0.5rem; + padding: 1rem; + text-align: center; + border: 1px solid ${e=>e.$isDarkMode?"#4b5563":"#e5e7eb"}; +`,Q=o.div` + width: 100%; + height: 100px; + background: ${e=>e.$isDarkMode?"#4b5563":"#e5e7eb"}; + border-radius: 0.5rem; + display: flex; + align-items: center; + justify-content: center; + color: ${e=>e.$isDarkMode?"#9ca3af":"#6b7280"}; + font-size: 0.875rem; + margin-bottom: 0.5rem; +`,U=()=>r.jsx("svg",{className:"w-8 h-8",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:r.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z"})}),b=({caption:e})=>{const{isDarkMode:t}=c();return r.jsxs(O,{$isDarkMode:t,children:[r.jsx(Q,{$isDarkMode:t,role:"img","aria-label":`${e} placeholder image`,children:r.jsx(U,{})}),r.jsx("div",{style:{fontSize:"0.75rem",color:t?"#9ca3af":"#6b7280"},children:e})]})},X=()=>r.jsxs(q,{children:[r.jsx(b,{caption:"Latest Photo"}),r.jsx(b,{caption:"Growth Progress"}),r.jsx(b,{caption:"Planting Day"})]}),Y=o.div` + display: grid; + grid-template-columns: 1fr; + gap: 1.5rem; + margin-bottom: 2rem; + + @media (min-width: 768px) { + grid-template-columns: 1fr 1fr; + } +`,Z=o.div` + margin-bottom: 2rem; +`,_=o.h3` + font-size: 1.125rem; + font-weight: 600; + color: ${e=>e.$isDarkMode?"#f9fafb":"#111827"}; + margin: 0 0 1rem 0; +`,oe=({tree:e,isOpen:t,onClose:a})=>{const{containerRef:s}=$({onEscape:a,trapFocus:!0,autoFocus:!0}),{isDarkMode:d}=c(),l=y(e,t);return e?r.jsxs(L,{isOpen:t,onClose:a,title:e.name,containerRef:s,children:[r.jsxs(Y,{children:[r.jsx(N,{tree:e}),r.jsx(R,{tree:e})]}),r.jsx(K,{measurementHistory:l}),r.jsxs(Z,{children:[r.jsx(_,{$isDarkMode:d,children:"Tree Images"}),r.jsx(X,{})]})]}):null};export{oe as T}; diff --git a/frontend/dist/assets/index-TYKw_KzE.css b/frontend/dist/assets/index-TYKw_KzE.css new file mode 100644 index 0000000000..b7e39ae26f --- /dev/null +++ b/frontend/dist/assets/index-TYKw_KzE.css @@ -0,0 +1 @@ +*,:before,:after{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }*,:before,:after{box-sizing:border-box;border-width:0;border-style:solid;border-color:#e5e7eb}:before,:after{--tw-content: ""}html,:host{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:Inter,system-ui,sans-serif;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:JetBrains Mono,Menlo,Monaco,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dl,dd,h1,h2,h3,h4,h5,h6,hr,figure,p,pre{margin:0}fieldset{margin:0;padding:0}legend{padding:0}ol,ul,menu{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}button,[role=button]{cursor:pointer}:disabled{cursor:default}img,svg,video,canvas,audio,iframe,embed,object{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]:where(:not([hidden=until-found])){display:none}.heading-1{font-size:2.25rem;line-height:2.5rem;font-weight:700;line-height:1.25;letter-spacing:-.025em}@media (min-width: 768px){.heading-1{font-size:3rem;line-height:1}}.heading-2{font-size:1.875rem;line-height:2.25rem;font-weight:700;line-height:1.25;letter-spacing:-.025em}@media (min-width: 768px){.heading-2{font-size:2.25rem;line-height:2.5rem}}.heading-3{font-size:1.5rem;line-height:2rem;font-weight:600;line-height:1.375}@media (min-width: 768px){.heading-3{font-size:1.875rem;line-height:2.25rem}}.heading-4{font-size:1.25rem;line-height:1.75rem;font-weight:600;line-height:1.375}@media (min-width: 768px){.heading-4{font-size:1.5rem;line-height:2rem}}.heading-5{font-size:1.125rem;line-height:1.75rem;font-weight:600;line-height:1.5}@media (min-width: 768px){.heading-5{font-size:1.25rem;line-height:1.75rem}}.heading-6{font-size:1rem;line-height:1.5rem;font-weight:600;line-height:1.5}@media (min-width: 768px){.heading-6{font-size:1.125rem;line-height:1.75rem}}.body-lg{font-size:1.125rem;line-height:1.75rem;line-height:1.625}.body-base{font-size:1rem;line-height:1.5rem;line-height:1.5}.body-sm{font-size:.875rem;line-height:1.25rem;line-height:1.5}.body-xs{font-size:.75rem;line-height:1rem;line-height:1.5}.text-muted{--tw-text-opacity: 1;color:rgb(75 85 99 / var(--tw-text-opacity, 1))}.text-muted:is(.dark *){--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.text-subtle{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.text-subtle:is(.dark *){--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.text-primary{--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1))}.text-primary:is(.dark *){--tw-text-opacity: 1;color:rgb(56 189 248 / var(--tw-text-opacity, 1))}.text-secondary{--tw-text-opacity: 1;color:rgb(101 163 13 / var(--tw-text-opacity, 1))}.text-secondary:is(.dark *){--tw-text-opacity: 1;color:rgb(163 230 53 / var(--tw-text-opacity, 1))}.text-success{--tw-text-opacity: 1;color:rgb(22 163 74 / var(--tw-text-opacity, 1))}.text-success:is(.dark *){--tw-text-opacity: 1;color:rgb(74 222 128 / var(--tw-text-opacity, 1))}.text-warning{--tw-text-opacity: 1;color:rgb(202 138 4 / var(--tw-text-opacity, 1))}.text-warning:is(.dark *){--tw-text-opacity: 1;color:rgb(250 204 21 / var(--tw-text-opacity, 1))}.text-error{--tw-text-opacity: 1;color:rgb(220 38 38 / var(--tw-text-opacity, 1))}.text-error:is(.dark *){--tw-text-opacity: 1;color:rgb(248 113 113 / var(--tw-text-opacity, 1))}.text-strong{font-weight:600;--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.text-strong:is(.dark *){--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}:root{--color-primary: 14 165 233;--color-secondary: 132 204 22;--color-success: 34 197 94;--color-warning: 245 158 11;--color-error: 239 68 68;--focus-ring: 59 130 246}.dark{--color-primary: 14 165 233;--color-secondary: 132 204 22;--color-success: 34 197 94;--color-warning: 245 158 11;--color-error: 239 68 68}body{font-feature-settings:"rlig" 1,"calt" 1;--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1));transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.2s}body:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(17 24 39 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}*{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.2s}a{text-underline-offset:4px}a:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}a:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.\!container{width:100%!important}.container{width:100%}@media (min-width: 320px){.\!container{max-width:320px!important}.container{max-width:320px}}@media (min-width: 640px){.\!container{max-width:640px!important}.container{max-width:640px}}@media (min-width: 768px){.\!container{max-width:768px!important}.container{max-width:768px}}@media (min-width: 1024px){.\!container{max-width:1024px!important}.container{max-width:1024px}}@media (min-width: 1280px){.\!container{max-width:1280px!important}.container{max-width:1280px}}@media (min-width: 1536px){.\!container{max-width:1536px!important}.container{max-width:1536px}}.display-xl{font-size:3rem;font-weight:800;line-height:1;letter-spacing:-.025em}@media (min-width: 768px){.display-xl{font-size:3.75rem;line-height:1}}.display-lg{font-size:2.25rem;line-height:2.5rem;font-weight:800;line-height:1;letter-spacing:-.025em}@media (min-width: 768px){.display-lg{font-size:3rem;line-height:1}}.display-md{font-size:1.875rem;line-height:2.25rem;font-weight:700;line-height:1.25;letter-spacing:-.025em}@media (min-width: 768px){.display-md{font-size:2.25rem;line-height:2.5rem}}.display-sm{font-size:1.5rem;line-height:2rem;font-weight:700;line-height:1.25}@media (min-width: 768px){.display-sm{font-size:1.875rem;line-height:2.25rem}}.caption{font-size:.75rem;line-height:1rem;font-weight:500;text-transform:uppercase;letter-spacing:.025em;--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.caption:is(.dark *){--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.overline{font-size:.875rem;line-height:1.25rem;font-weight:600;text-transform:uppercase;letter-spacing:.05em;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.overline:is(.dark *){--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.link-primary{--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1));text-decoration-line:underline;text-underline-offset:2px;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.link-primary:hover{--tw-text-opacity: 1;color:rgb(3 105 161 / var(--tw-text-opacity, 1))}.link-primary:is(.dark *){--tw-text-opacity: 1;color:rgb(56 189 248 / var(--tw-text-opacity, 1))}.link-primary:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(125 211 252 / var(--tw-text-opacity, 1))}.link-bare{color:currentColor;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.link-bare:hover{--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1))}.truncate-1{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.truncate-2{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:2}.truncate-3{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:3}.btn{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn:disabled{pointer-events:none;opacity:.5}.btn:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-primary{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-primary:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-primary:disabled{pointer-events:none;opacity:.5}.btn-primary:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-primary{--tw-bg-opacity: 1;background-color:rgb(2 132 199 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.btn-primary:hover{--tw-bg-opacity: 1;background-color:rgb(3 105 161 / var(--tw-bg-opacity, 1))}.btn-secondary{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-secondary:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-secondary:disabled{pointer-events:none;opacity:.5}.btn-secondary:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-secondary{--tw-bg-opacity: 1;background-color:rgb(101 163 13 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.btn-secondary:hover{--tw-bg-opacity: 1;background-color:rgb(77 124 15 / var(--tw-bg-opacity, 1))}.btn-outline{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-outline:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-outline:disabled{pointer-events:none;opacity:.5}.btn-outline:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-outline{border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.btn-outline:hover{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.btn-outline:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.btn-outline:hover:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(55 65 81 / var(--tw-bg-opacity, 1))}.btn-ghost{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-ghost:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-ghost:disabled{pointer-events:none;opacity:.5}.btn-ghost:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-ghost{background-color:transparent;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.btn-ghost:hover{--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.btn-ghost:is(.dark *){--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.btn-ghost:hover:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1))}.btn-link{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-link:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-link:disabled{pointer-events:none;opacity:.5}.btn-link:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-link{background-color:transparent;--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1));text-decoration-line:underline}.btn-link:hover{--tw-text-opacity: 1;color:rgb(3 105 161 / var(--tw-text-opacity, 1))}.btn-link:is(.dark *){--tw-text-opacity: 1;color:rgb(14 165 233 / var(--tw-text-opacity, 1))}.btn-link:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1))}.btn-destructive{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-destructive:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-destructive:disabled{pointer-events:none;opacity:.5}.btn-destructive:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-destructive{--tw-bg-opacity: 1;background-color:rgb(239 68 68 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.btn-destructive:hover{--tw-bg-opacity: 1;background-color:rgb(185 28 28 / var(--tw-bg-opacity, 1))}.btn-success{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-success:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-success:disabled{pointer-events:none;opacity:.5}.btn-success:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-success{--tw-bg-opacity: 1;background-color:rgb(22 163 74 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.btn-success:hover{--tw-bg-opacity: 1;background-color:rgb(21 128 61 / var(--tw-bg-opacity, 1))}.btn-disabled{pointer-events:none;opacity:.5}.btn-loading{position:relative}.btn-icon{border-radius:9999px;padding:.5rem}.btn-xs{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-xs:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-xs:disabled{pointer-events:none;opacity:.5}.btn-xs:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-xs{border-radius:.125rem;padding:.25rem .5rem;font-size:.75rem;line-height:1rem}.btn-sm{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-sm:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-sm:disabled{pointer-events:none;opacity:.5}.btn-sm:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-sm{border-radius:.375rem;padding:.5rem .75rem;font-size:.875rem;line-height:1.25rem}.btn-md{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-md:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-md:disabled{pointer-events:none;opacity:.5}.btn-md:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-md{border-radius:.375rem;padding:.5rem 1rem;font-size:.875rem;line-height:1.25rem}.btn-lg{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-lg:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-lg:disabled{pointer-events:none;opacity:.5}.btn-lg:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-lg{border-radius:.5rem;padding:.75rem 1.5rem;font-size:1rem;line-height:1.5rem}.btn-xl{display:inline-flex;align-items:center;justify-content:center;border-radius:.375rem;font-size:.875rem;line-height:1.25rem;font-weight:500;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.btn-xl:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-color: rgb(var(--focus-ring));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.btn-xl:disabled{pointer-events:none;opacity:.5}.btn-xl:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.btn-xl{border-radius:.5rem;padding:1rem 2rem;font-size:1.125rem;line-height:1.75rem}.form-input{display:flex;height:2.5rem;width:100%;border-radius:.375rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));padding:.5rem .75rem;font-size:.875rem;line-height:1.25rem}.form-input::-moz-placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input::placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(14 165 233 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px}.form-input:disabled{cursor:not-allowed;opacity:.5}.form-input:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.form-input:is(.dark *)::-moz-placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input:is(.dark *)::placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-sm{display:flex;height:2.5rem;width:100%;border-radius:.375rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));padding:.5rem .75rem;font-size:.875rem;line-height:1.25rem}.form-input-sm::-moz-placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-sm::placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-sm:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(14 165 233 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px}.form-input-sm:disabled{cursor:not-allowed;opacity:.5}.form-input-sm:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.form-input-sm:is(.dark *)::-moz-placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-sm:is(.dark *)::placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-sm{height:2rem;padding:.25rem .5rem;font-size:.75rem;line-height:1rem}.form-input-lg{display:flex;height:2.5rem;width:100%;border-radius:.375rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));padding:.5rem .75rem;font-size:.875rem;line-height:1.25rem}.form-input-lg::-moz-placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-lg::placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-lg:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(14 165 233 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px}.form-input-lg:disabled{cursor:not-allowed;opacity:.5}.form-input-lg:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.form-input-lg:is(.dark *)::-moz-placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-lg:is(.dark *)::placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-lg{height:3rem;padding:.75rem 1rem;font-size:1rem;line-height:1.5rem}.form-input-error{display:flex;height:2.5rem;width:100%;border-radius:.375rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));padding:.5rem .75rem;font-size:.875rem;line-height:1.25rem}.form-input-error::-moz-placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-error::placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-error:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(14 165 233 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px}.form-input-error:disabled{cursor:not-allowed;opacity:.5}.form-input-error:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.form-input-error:is(.dark *)::-moz-placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-error:is(.dark *)::placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-error{--tw-border-opacity: 1;border-color:rgb(239 68 68 / var(--tw-border-opacity, 1))}.form-input-error:focus-visible{--tw-ring-opacity: 1;--tw-ring-color: rgb(239 68 68 / var(--tw-ring-opacity, 1))}.form-input-success{display:flex;height:2.5rem;width:100%;border-radius:.375rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));padding:.5rem .75rem;font-size:.875rem;line-height:1.25rem}.form-input-success::-moz-placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-success::placeholder{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.form-input-success:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(14 165 233 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px}.form-input-success:disabled{cursor:not-allowed;opacity:.5}.form-input-success:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.form-input-success:is(.dark *)::-moz-placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-success:is(.dark *)::placeholder{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.form-input-success{--tw-border-opacity: 1;border-color:rgb(34 197 94 / var(--tw-border-opacity, 1))}.form-input-success:focus-visible{--tw-ring-opacity: 1;--tw-ring-color: rgb(34 197 94 / var(--tw-ring-opacity, 1))}.form-label{font-size:.875rem;line-height:1.25rem;font-weight:500;line-height:1;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.form-label:is(.dark *){--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.card{border-radius:.5rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(3 7 18 / var(--tw-text-opacity, 1));--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.card:is(.dark *){--tw-border-opacity: 1;border-color:rgb(31 41 55 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(3 7 18 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(249 250 251 / var(--tw-text-opacity, 1))}.card-bordered{border-radius:.5rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(3 7 18 / var(--tw-text-opacity, 1));--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.card-bordered:is(.dark *){--tw-border-opacity: 1;border-color:rgb(31 41 55 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(3 7 18 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(249 250 251 / var(--tw-text-opacity, 1))}.card-bordered{border-width:2px}.card-elevated{border-radius:.5rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(3 7 18 / var(--tw-text-opacity, 1));--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.card-elevated:is(.dark *){--tw-border-opacity: 1;border-color:rgb(31 41 55 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(3 7 18 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(249 250 251 / var(--tw-text-opacity, 1))}.card-elevated{--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / .1), 0 2px 4px -2px rgb(0 0 0 / .1);--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow);transition-property:box-shadow;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.card-elevated:hover{--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / .1), 0 4px 6px -4px rgb(0 0 0 / .1);--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.card-interactive{border-radius:.5rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(3 7 18 / var(--tw-text-opacity, 1));--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.card-interactive:is(.dark *){--tw-border-opacity: 1;border-color:rgb(31 41 55 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(3 7 18 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(249 250 251 / var(--tw-text-opacity, 1))}.card-interactive{cursor:pointer;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.card-interactive:hover{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.card-interactive:hover:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(17 24 39 / var(--tw-bg-opacity, 1))}.alert{border-radius:.5rem;border-width:1px;padding:1rem}.alert-info{border-radius:.5rem;border-width:1px;padding:1rem;--tw-border-opacity: 1;border-color:rgb(186 230 253 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(240 249 255 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(7 89 133 / var(--tw-text-opacity, 1))}.alert-info:is(.dark *){--tw-border-opacity: 1;border-color:rgb(7 89 133 / var(--tw-border-opacity, 1));background-color:#0c4a6e33;--tw-text-opacity: 1;color:rgb(186 230 253 / var(--tw-text-opacity, 1))}.alert-success{border-radius:.5rem;border-width:1px;padding:1rem;--tw-border-opacity: 1;border-color:rgb(187 247 208 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(240 253 244 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(22 101 52 / var(--tw-text-opacity, 1))}.alert-success:is(.dark *){--tw-border-opacity: 1;border-color:rgb(22 101 52 / var(--tw-border-opacity, 1));background-color:#14532d33;--tw-text-opacity: 1;color:rgb(187 247 208 / var(--tw-text-opacity, 1))}.alert-warning{border-radius:.5rem;border-width:1px;padding:1rem;--tw-border-opacity: 1;border-color:rgb(254 240 138 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(254 252 232 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(133 77 14 / var(--tw-text-opacity, 1))}.alert-warning:is(.dark *){--tw-border-opacity: 1;border-color:rgb(133 77 14 / var(--tw-border-opacity, 1));background-color:#713f1233;--tw-text-opacity: 1;color:rgb(254 240 138 / var(--tw-text-opacity, 1))}.alert-error{border-radius:.5rem;border-width:1px;padding:1rem;--tw-border-opacity: 1;border-color:rgb(254 202 202 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(254 242 242 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(153 27 27 / var(--tw-text-opacity, 1))}.alert-error:is(.dark *){--tw-border-opacity: 1;border-color:rgb(153 27 27 / var(--tw-border-opacity, 1));background-color:#7f1d1d33;--tw-text-opacity: 1;color:rgb(254 202 202 / var(--tw-text-opacity, 1))}.badge{display:inline-flex;align-items:center;border-radius:9999px;padding:.125rem .625rem;font-size:.75rem;line-height:1rem;font-weight:500}.badge-primary{display:inline-flex;align-items:center;border-radius:9999px;padding:.125rem .625rem;font-size:.75rem;line-height:1rem;font-weight:500;--tw-bg-opacity: 1;background-color:rgb(224 242 254 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(7 89 133 / var(--tw-text-opacity, 1))}.badge-primary:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(12 74 110 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(125 211 252 / var(--tw-text-opacity, 1))}.badge-secondary{display:inline-flex;align-items:center;border-radius:9999px;padding:.125rem .625rem;font-size:.75rem;line-height:1rem;font-weight:500;--tw-bg-opacity: 1;background-color:rgb(236 252 203 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(63 98 18 / var(--tw-text-opacity, 1))}.badge-secondary:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(54 83 20 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(190 242 100 / var(--tw-text-opacity, 1))}.badge-success{display:inline-flex;align-items:center;border-radius:9999px;padding:.125rem .625rem;font-size:.75rem;line-height:1rem;font-weight:500;--tw-bg-opacity: 1;background-color:rgb(220 252 231 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(22 101 52 / var(--tw-text-opacity, 1))}.badge-success:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(20 83 45 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(134 239 172 / var(--tw-text-opacity, 1))}.badge-warning{display:inline-flex;align-items:center;border-radius:9999px;padding:.125rem .625rem;font-size:.75rem;line-height:1rem;font-weight:500;--tw-bg-opacity: 1;background-color:rgb(254 249 195 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(133 77 14 / var(--tw-text-opacity, 1))}.badge-warning:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(113 63 18 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(253 224 71 / var(--tw-text-opacity, 1))}.badge-error{display:inline-flex;align-items:center;border-radius:9999px;padding:.125rem .625rem;font-size:.75rem;line-height:1rem;font-weight:500;--tw-bg-opacity: 1;background-color:rgb(254 226 226 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(153 27 27 / var(--tw-text-opacity, 1))}.badge-error:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(127 29 29 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(252 165 165 / var(--tw-text-opacity, 1))}.badge-gray{display:inline-flex;align-items:center;border-radius:9999px;padding:.125rem .625rem;font-size:.75rem;line-height:1rem;font-weight:500;--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(31 41 55 / var(--tw-text-opacity, 1))}.badge-gray:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1));--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.avatar{display:inline-flex;align-items:center;justify-content:center;overflow:hidden;border-radius:9999px;--tw-bg-opacity: 1;background-color:rgb(107 114 128 / var(--tw-bg-opacity, 1))}.avatar-xs{display:inline-flex;align-items:center;justify-content:center;overflow:hidden;border-radius:9999px;--tw-bg-opacity: 1;background-color:rgb(107 114 128 / var(--tw-bg-opacity, 1));height:1.5rem;width:1.5rem;font-size:.75rem;line-height:1rem}.avatar-sm{display:inline-flex;align-items:center;justify-content:center;overflow:hidden;border-radius:9999px;--tw-bg-opacity: 1;background-color:rgb(107 114 128 / var(--tw-bg-opacity, 1));height:2rem;width:2rem;font-size:.875rem;line-height:1.25rem}.avatar-md{display:inline-flex;align-items:center;justify-content:center;overflow:hidden;border-radius:9999px;--tw-bg-opacity: 1;background-color:rgb(107 114 128 / var(--tw-bg-opacity, 1));height:2.5rem;width:2.5rem;font-size:1rem;line-height:1.5rem}.avatar-lg{display:inline-flex;align-items:center;justify-content:center;overflow:hidden;border-radius:9999px;--tw-bg-opacity: 1;background-color:rgb(107 114 128 / var(--tw-bg-opacity, 1));height:3rem;width:3rem;font-size:1.125rem;line-height:1.75rem}.avatar-xl{display:inline-flex;align-items:center;justify-content:center;overflow:hidden;border-radius:9999px;--tw-bg-opacity: 1;background-color:rgb(107 114 128 / var(--tw-bg-opacity, 1));height:4rem;width:4rem;font-size:1.25rem;line-height:1.75rem}.spinner{display:inline-block}.spinner{animation:spin 1s linear infinite;border-radius:9999px;border-width:2px;border-style:solid;border-color:currentColor;border-right-color:transparent}.spinner-sm{display:inline-block}.spinner-sm{animation:spin 1s linear infinite;border-radius:9999px;border-width:2px;border-style:solid;border-color:currentColor;border-right-color:transparent;height:1rem;width:1rem}.spinner-md{display:inline-block}.spinner-md{animation:spin 1s linear infinite;border-radius:9999px;border-width:2px;border-style:solid;border-color:currentColor;border-right-color:transparent;height:1.5rem;width:1.5rem}.spinner-lg{display:inline-block}.spinner-lg{animation:spin 1s linear infinite;border-radius:9999px;border-width:2px;border-style:solid;border-color:currentColor;border-right-color:transparent;height:2rem;width:2rem}.focus-ring:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(14 165 233 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.focus-ring:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.focus-ring-error:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(239 68 68 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.focus-ring-error:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.focus-ring-success:focus-visible{outline:2px solid transparent;outline-offset:2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-opacity: 1;--tw-ring-color: rgb(34 197 94 / var(--tw-ring-opacity, 1));--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff}.focus-ring-success:focus-visible:is(.dark *){--tw-ring-offset-color: #111827}.link{--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1));text-decoration-line:underline;text-underline-offset:2px;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.link:hover{--tw-text-opacity: 1;color:rgb(3 105 161 / var(--tw-text-opacity, 1))}.link:is(.dark *){--tw-text-opacity: 1;color:rgb(14 165 233 / var(--tw-text-opacity, 1))}.link:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1))}.link-muted{--tw-text-opacity: 1;color:rgb(75 85 99 / var(--tw-text-opacity, 1));text-decoration-line:underline;text-underline-offset:2px;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.link-muted:hover{--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.link-muted:is(.dark *){--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.link-muted:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.link-bare{color:currentColor;text-decoration-line:none;text-underline-offset:2px;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.link-bare:hover{--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1));text-decoration-line:underline}.link-bare:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(56 189 248 / var(--tw-text-opacity, 1))}.pointer-events-none{pointer-events:none}.visible{visibility:visible}.invisible{visibility:hidden}.collapse{visibility:collapse}.static{position:static}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.sticky{position:sticky}.inset-0{top:0;right:0;bottom:0;left:0}.inset-y-0{top:0;bottom:0}.left-0{left:0}.right-0{right:0}.top-0{top:0}.top-6{top:1.5rem}.z-10{z-index:10}.z-40{z-index:40}.z-50{z-index:50}.col-span-1{grid-column:span 1 / span 1}.m-0{margin:0}.mx-4{margin-left:1rem;margin-right:1rem}.mx-auto{margin-left:auto;margin-right:auto}.-ml-1{margin-left:-.25rem}.mb-16{margin-bottom:4rem}.mb-2{margin-bottom:.5rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.mb-6{margin-bottom:1.5rem}.mb-8{margin-bottom:2rem}.ml-1{margin-left:.25rem}.ml-2{margin-left:.5rem}.ml-3{margin-left:.75rem}.ml-4{margin-left:1rem}.mr-2{margin-right:.5rem}.mr-3{margin-right:.75rem}.mr-4{margin-right:1rem}.mt-1{margin-top:.25rem}.mt-2{margin-top:.5rem}.mt-3{margin-top:.75rem}.mt-4{margin-top:1rem}.mt-8{margin-top:2rem}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.table{display:table}.grid{display:grid}.hidden{display:none}.h-10{height:2.5rem}.h-12{height:3rem}.h-14{height:3.5rem}.h-16{height:4rem}.h-2{height:.5rem}.h-20{height:5rem}.h-24{height:6rem}.h-28{height:7rem}.h-3{height:.75rem}.h-32{height:8rem}.h-4{height:1rem}.h-40{height:10rem}.h-48{height:12rem}.h-5{height:1.25rem}.h-6{height:1.5rem}.h-72{height:18rem}.h-8{height:2rem}.h-96{height:24rem}.h-\[260px\]{height:260px}.h-full{height:100%}.h-screen{height:100vh}.min-h-screen{min-height:100vh}.w-10{width:2.5rem}.w-12{width:3rem}.w-16{width:4rem}.w-2{width:.5rem}.w-20{width:5rem}.w-24{width:6rem}.w-3{width:.75rem}.w-32{width:8rem}.w-4{width:1rem}.w-48{width:12rem}.w-5{width:1.25rem}.w-6{width:1.5rem}.w-64{width:16rem}.w-8{width:2rem}.w-full{width:100%}.max-w-2xl{max-width:42rem}.max-w-3xl{max-width:48rem}.max-w-4xl{max-width:56rem}.max-w-5xl{max-width:64rem}.max-w-6xl{max-width:72rem}.max-w-7xl{max-width:80rem}.max-w-8xl{max-width:88rem}.max-w-full{max-width:100%}.max-w-lg{max-width:32rem}.max-w-md{max-width:28rem}.max-w-none{max-width:none}.max-w-sm{max-width:24rem}.max-w-xl{max-width:36rem}.flex-1{flex:1 1 0%}.flex-shrink{flex-shrink:1}.flex-shrink-0,.shrink-0{flex-shrink:0}.border-collapse{border-collapse:collapse}.-translate-x-full{--tw-translate-x: -100%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-x-0{--tw-translate-x: 0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-rotate-90{--tw-rotate: -90deg;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.rotate-0{--tw-rotate: 0deg;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.rotate-90{--tw-rotate: 90deg;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-0{--tw-scale-x: 0;--tw-scale-y: 0;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-100{--tw-scale-x: 1;--tw-scale-y: 1;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes pulse{50%{opacity:.5}}.animate-pulse{animation:pulse 2s cubic-bezier(.4,0,.6,1) infinite}@keyframes spin{to{transform:rotate(360deg)}}.animate-spin{animation:spin 1s linear infinite}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-12{grid-template-columns:repeat(12,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.grid-cols-5{grid-template-columns:repeat(5,minmax(0,1fr))}.grid-cols-6{grid-template-columns:repeat(6,minmax(0,1fr))}.flex-row{flex-direction:row}.flex-row-reverse{flex-direction:row-reverse}.flex-col{flex-direction:column}.flex-col-reverse{flex-direction:column-reverse}.flex-wrap{flex-wrap:wrap}.flex-wrap-reverse{flex-wrap:wrap-reverse}.flex-nowrap{flex-wrap:nowrap}.items-start{align-items:flex-start}.items-end{align-items:flex-end}.items-center{align-items:center}.items-baseline{align-items:baseline}.items-stretch{align-items:stretch}.justify-start{justify-content:flex-start}.justify-end{justify-content:flex-end}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.justify-around{justify-content:space-around}.justify-evenly{justify-content:space-evenly}.gap-0{gap:0px}.gap-1{gap:.25rem}.gap-10{gap:2.5rem}.gap-12{gap:3rem}.gap-2{gap:.5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.gap-5{gap:1.25rem}.gap-6{gap:1.5rem}.gap-8{gap:2rem}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(.5rem * var(--tw-space-x-reverse));margin-left:calc(.5rem * calc(1 - var(--tw-space-x-reverse)))}.space-x-4>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(1rem * var(--tw-space-x-reverse));margin-left:calc(1rem * calc(1 - var(--tw-space-x-reverse)))}.space-x-6>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(1.5rem * var(--tw-space-x-reverse));margin-left:calc(1.5rem * calc(1 - var(--tw-space-x-reverse)))}.space-y-1>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.25rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.25rem * var(--tw-space-y-reverse))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.5rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem * var(--tw-space-y-reverse))}.space-y-3>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.75rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.75rem * var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(1rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem * var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(1.5rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem * var(--tw-space-y-reverse))}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.whitespace-pre-wrap{white-space:pre-wrap}.break-words{overflow-wrap:break-word}.rounded{border-radius:.25rem}.rounded-2xl{border-radius:1rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-md{border-radius:.375rem}.rounded-none{border-radius:0}.rounded-xl{border-radius:.75rem}.border{border-width:1px}.border-2{border-width:2px}.border-b{border-bottom-width:1px}.border-r{border-right-width:1px}.border-t{border-top-width:1px}.border-none{border-style:none}.border-gray-200{--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1))}.border-gray-300{--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1))}.border-gray-800{--tw-border-opacity: 1;border-color:rgb(31 41 55 / var(--tw-border-opacity, 1))}.border-green-200{--tw-border-opacity: 1;border-color:rgb(187 247 208 / var(--tw-border-opacity, 1))}.border-red-200{--tw-border-opacity: 1;border-color:rgb(254 202 202 / var(--tw-border-opacity, 1))}.border-red-300{--tw-border-opacity: 1;border-color:rgb(252 165 165 / var(--tw-border-opacity, 1))}.border-transparent{border-color:transparent}.border-white{--tw-border-opacity: 1;border-color:rgb(255 255 255 / var(--tw-border-opacity, 1))}.bg-black{--tw-bg-opacity: 1;background-color:rgb(0 0 0 / var(--tw-bg-opacity, 1))}.bg-black\/50{background-color:#00000080}.bg-blue-100{--tw-bg-opacity: 1;background-color:rgb(219 234 254 / var(--tw-bg-opacity, 1))}.bg-blue-50{--tw-bg-opacity: 1;background-color:rgb(239 246 255 / var(--tw-bg-opacity, 1))}.bg-blue-500{--tw-bg-opacity: 1;background-color:rgb(59 130 246 / var(--tw-bg-opacity, 1))}.bg-gray-100{--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.bg-gray-200{--tw-bg-opacity: 1;background-color:rgb(229 231 235 / var(--tw-bg-opacity, 1))}.bg-gray-300{--tw-bg-opacity: 1;background-color:rgb(209 213 219 / var(--tw-bg-opacity, 1))}.bg-gray-50{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.bg-gray-900{--tw-bg-opacity: 1;background-color:rgb(17 24 39 / var(--tw-bg-opacity, 1))}.bg-green-100{--tw-bg-opacity: 1;background-color:rgb(220 252 231 / var(--tw-bg-opacity, 1))}.bg-green-50{--tw-bg-opacity: 1;background-color:rgb(240 253 244 / var(--tw-bg-opacity, 1))}.bg-green-500{--tw-bg-opacity: 1;background-color:rgb(34 197 94 / var(--tw-bg-opacity, 1))}.bg-green-600{--tw-bg-opacity: 1;background-color:rgb(22 163 74 / var(--tw-bg-opacity, 1))}.bg-indigo-100{--tw-bg-opacity: 1;background-color:rgb(224 231 255 / var(--tw-bg-opacity, 1))}.bg-orange-100{--tw-bg-opacity: 1;background-color:rgb(255 237 213 / var(--tw-bg-opacity, 1))}.bg-orange-500{--tw-bg-opacity: 1;background-color:rgb(249 115 22 / var(--tw-bg-opacity, 1))}.bg-purple-100{--tw-bg-opacity: 1;background-color:rgb(243 232 255 / var(--tw-bg-opacity, 1))}.bg-purple-500{--tw-bg-opacity: 1;background-color:rgb(168 85 247 / var(--tw-bg-opacity, 1))}.bg-red-100{--tw-bg-opacity: 1;background-color:rgb(254 226 226 / var(--tw-bg-opacity, 1))}.bg-red-50{--tw-bg-opacity: 1;background-color:rgb(254 242 242 / var(--tw-bg-opacity, 1))}.bg-red-500{--tw-bg-opacity: 1;background-color:rgb(239 68 68 / var(--tw-bg-opacity, 1))}.bg-white{--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1))}.bg-white\/80{background-color:#fffc}.bg-yellow-100{--tw-bg-opacity: 1;background-color:rgb(254 249 195 / var(--tw-bg-opacity, 1))}.bg-yellow-500{--tw-bg-opacity: 1;background-color:rgb(234 179 8 / var(--tw-bg-opacity, 1))}.bg-opacity-50{--tw-bg-opacity: .5}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-none{background-image:none}.from-green-50{--tw-gradient-from: #f0fdf4 var(--tw-gradient-from-position);--tw-gradient-to: rgb(240 253 244 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.from-green-500{--tw-gradient-from: #22c55e var(--tw-gradient-from-position);--tw-gradient-to: rgb(34 197 94 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.to-green-100{--tw-gradient-to: #dcfce7 var(--tw-gradient-to-position)}.to-green-700{--tw-gradient-to: #15803d var(--tw-gradient-to-position)}.fill-gray-300{fill:#d1d5db}.p-0{padding:0}.p-2{padding:.5rem}.p-3{padding:.75rem}.p-4{padding:1rem}.p-5{padding:1.25rem}.p-6{padding:1.5rem}.p-8{padding:2rem}.px-0{padding-left:0;padding-right:0}.px-10{padding-left:2.5rem;padding-right:2.5rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-2\.5{padding-left:.625rem;padding-right:.625rem}.px-3{padding-left:.75rem;padding-right:.75rem}.px-4{padding-left:1rem;padding-right:1rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.px-8{padding-left:2rem;padding-right:2rem}.py-0\.5{padding-top:.125rem;padding-bottom:.125rem}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-12{padding-top:3rem;padding-bottom:3rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-20{padding-top:5rem;padding-bottom:5rem}.py-24{padding-top:6rem;padding-bottom:6rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-8{padding-top:2rem;padding-bottom:2rem}.pb-2{padding-bottom:.5rem}.pb-4{padding-bottom:1rem}.pr-10{padding-right:2.5rem}.pr-3{padding-right:.75rem}.pt-1{padding-top:.25rem}.pt-2{padding-top:.5rem}.pt-20{padding-top:5rem}.pt-8{padding-top:2rem}.text-left{text-align:left}.text-center{text-align:center}.text-right{text-align:right}.text-justify{text-align:justify}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-4xl{font-size:2.25rem;line-height:2.5rem}.text-6xl{font-size:3.75rem;line-height:1}.text-base{font-size:1rem;line-height:1.5rem}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-bold{font-weight:700}.font-extrabold{font-weight:800}.font-medium{font-weight:500}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.lowercase{text-transform:lowercase}.capitalize{text-transform:capitalize}.leading-relaxed{line-height:1.625}.tracking-tight{letter-spacing:-.025em}.tracking-wide{letter-spacing:.025em}.text-blue-500{--tw-text-opacity: 1;color:rgb(59 130 246 / var(--tw-text-opacity, 1))}.text-blue-600{--tw-text-opacity: 1;color:rgb(37 99 235 / var(--tw-text-opacity, 1))}.text-error{--tw-text-opacity: 1;color:rgb(239 68 68 / var(--tw-text-opacity, 1))}.text-gray-300{--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.text-gray-400{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.text-gray-500{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.text-gray-600{--tw-text-opacity: 1;color:rgb(75 85 99 / var(--tw-text-opacity, 1))}.text-gray-700{--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.text-gray-800{--tw-text-opacity: 1;color:rgb(31 41 55 / var(--tw-text-opacity, 1))}.text-gray-900{--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.text-green-100{--tw-text-opacity: 1;color:rgb(220 252 231 / var(--tw-text-opacity, 1))}.text-green-200{--tw-text-opacity: 1;color:rgb(187 247 208 / var(--tw-text-opacity, 1))}.text-green-400{--tw-text-opacity: 1;color:rgb(74 222 128 / var(--tw-text-opacity, 1))}.text-green-600{--tw-text-opacity: 1;color:rgb(22 163 74 / var(--tw-text-opacity, 1))}.text-green-700{--tw-text-opacity: 1;color:rgb(21 128 61 / var(--tw-text-opacity, 1))}.text-green-800{--tw-text-opacity: 1;color:rgb(22 101 52 / var(--tw-text-opacity, 1))}.text-indigo-600{--tw-text-opacity: 1;color:rgb(79 70 229 / var(--tw-text-opacity, 1))}.text-orange-600{--tw-text-opacity: 1;color:rgb(234 88 12 / var(--tw-text-opacity, 1))}.text-primary-600{--tw-text-opacity: 1;color:rgb(2 132 199 / var(--tw-text-opacity, 1))}.text-purple-600{--tw-text-opacity: 1;color:rgb(147 51 234 / var(--tw-text-opacity, 1))}.text-red-400{--tw-text-opacity: 1;color:rgb(248 113 113 / var(--tw-text-opacity, 1))}.text-red-500{--tw-text-opacity: 1;color:rgb(239 68 68 / var(--tw-text-opacity, 1))}.text-red-600{--tw-text-opacity: 1;color:rgb(220 38 38 / var(--tw-text-opacity, 1))}.text-red-700{--tw-text-opacity: 1;color:rgb(185 28 28 / var(--tw-text-opacity, 1))}.text-red-800{--tw-text-opacity: 1;color:rgb(153 27 27 / var(--tw-text-opacity, 1))}.text-secondary-600{--tw-text-opacity: 1;color:rgb(101 163 13 / var(--tw-text-opacity, 1))}.text-success{--tw-text-opacity: 1;color:rgb(34 197 94 / var(--tw-text-opacity, 1))}.text-warning{--tw-text-opacity: 1;color:rgb(245 158 11 / var(--tw-text-opacity, 1))}.text-white{--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.text-yellow-600{--tw-text-opacity: 1;color:rgb(202 138 4 / var(--tw-text-opacity, 1))}.underline{text-decoration-line:underline}.overline{text-decoration-line:overline}.no-underline{text-decoration-line:none}.placeholder-gray-500::-moz-placeholder{--tw-placeholder-opacity: 1;color:rgb(107 114 128 / var(--tw-placeholder-opacity, 1))}.placeholder-gray-500::placeholder{--tw-placeholder-opacity: 1;color:rgb(107 114 128 / var(--tw-placeholder-opacity, 1))}.opacity-0{opacity:0}.opacity-100{opacity:1}.opacity-25{opacity:.25}.opacity-30{opacity:.3}.opacity-75{opacity:.75}.shadow{--tw-shadow: 0 1px 3px 0 rgb(0 0 0 / .1), 0 1px 2px -1px rgb(0 0 0 / .1);--tw-shadow-colored: 0 1px 3px 0 var(--tw-shadow-color), 0 1px 2px -1px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-2xl{--tw-shadow: 0 25px 50px -12px rgb(0 0 0 / .25);--tw-shadow-colored: 0 25px 50px -12px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-lg{--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / .1), 0 4px 6px -4px rgb(0 0 0 / .1);--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-md{--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / .1), 0 2px 4px -2px rgb(0 0 0 / .1);--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-none{--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-sm{--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-xl{--tw-shadow: 0 20px 25px -5px rgb(0 0 0 / .1), 0 8px 10px -6px rgb(0 0 0 / .1);--tw-shadow-colored: 0 20px 25px -5px var(--tw-shadow-color), 0 8px 10px -6px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.outline{outline-style:solid}.ring{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)}.blur{--tw-blur: blur(8px);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur-sm{--tw-backdrop-blur: blur(4px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.transition{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-shadow{transition-property:box-shadow;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-transform{transition-property:transform;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.duration-1000{transition-duration:1s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.ease-in-out{transition-timing-function:cubic-bezier(.4,0,.2,1)}.ease-out{transition-timing-function:cubic-bezier(0,0,.2,1)}.skip-link{position:absolute;left:0;top:0;z-index:50;--tw-translate-y: -100%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));--tw-bg-opacity: 1;background-color:rgb(2 132 199 / var(--tw-bg-opacity, 1));padding:.5rem 1rem;--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1));transition-property:transform;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.skip-link:focus{--tw-translate-y: 0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.hover\:border-blue-300:hover{--tw-border-opacity: 1;border-color:rgb(147 197 253 / var(--tw-border-opacity, 1))}.hover\:border-gray-300:hover{--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1))}.hover\:border-green-300:hover{--tw-border-opacity: 1;border-color:rgb(134 239 172 / var(--tw-border-opacity, 1))}.hover\:bg-blue-50:hover{--tw-bg-opacity: 1;background-color:rgb(239 246 255 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-100:hover{--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-200:hover{--tw-bg-opacity: 1;background-color:rgb(229 231 235 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-50:hover{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.hover\:bg-green-200:hover{--tw-bg-opacity: 1;background-color:rgb(187 247 208 / var(--tw-bg-opacity, 1))}.hover\:bg-green-50:hover{--tw-bg-opacity: 1;background-color:rgb(240 253 244 / var(--tw-bg-opacity, 1))}.hover\:bg-green-700:hover{--tw-bg-opacity: 1;background-color:rgb(21 128 61 / var(--tw-bg-opacity, 1))}.hover\:bg-red-50:hover{--tw-bg-opacity: 1;background-color:rgb(254 242 242 / var(--tw-bg-opacity, 1))}.hover\:bg-white:hover{--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1))}.hover\:text-blue-600:hover{--tw-text-opacity: 1;color:rgb(37 99 235 / var(--tw-text-opacity, 1))}.hover\:text-blue-800:hover{--tw-text-opacity: 1;color:rgb(30 64 175 / var(--tw-text-opacity, 1))}.hover\:text-gray-700:hover{--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.hover\:text-gray-900:hover{--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.hover\:text-green-500:hover{--tw-text-opacity: 1;color:rgb(34 197 94 / var(--tw-text-opacity, 1))}.hover\:text-green-600:hover{--tw-text-opacity: 1;color:rgb(22 163 74 / var(--tw-text-opacity, 1))}.hover\:text-green-800:hover{--tw-text-opacity: 1;color:rgb(22 101 52 / var(--tw-text-opacity, 1))}.hover\:text-red-600:hover{--tw-text-opacity: 1;color:rgb(220 38 38 / var(--tw-text-opacity, 1))}.hover\:text-red-800:hover{--tw-text-opacity: 1;color:rgb(153 27 27 / var(--tw-text-opacity, 1))}.hover\:text-white:hover{--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.hover\:underline:hover{text-decoration-line:underline}.hover\:shadow-lg:hover{--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / .1), 0 4px 6px -4px rgb(0 0 0 / .1);--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.focus\:border-transparent:focus{border-color:transparent}.focus\:outline-none:focus{outline:2px solid transparent;outline-offset:2px}.focus\:outline-2:focus{outline-width:2px}.focus\:outline-offset-2:focus{outline-offset:2px}.focus\:outline-blue-500:focus{outline-color:#3b82f6}.focus\:ring-2:focus{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)}.focus\:ring-green-500:focus{--tw-ring-opacity: 1;--tw-ring-color: rgb(34 197 94 / var(--tw-ring-opacity, 1))}.focus\:ring-offset-2:focus{--tw-ring-offset-width: 2px}.focus-visible\:outline-none:focus-visible{outline:2px solid transparent;outline-offset:2px}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:opacity-50:disabled{opacity:.5}.dark\:border-gray-600:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1))}.dark\:border-gray-700:is(.dark *){--tw-border-opacity: 1;border-color:rgb(55 65 81 / var(--tw-border-opacity, 1))}.dark\:border-red-600:is(.dark *){--tw-border-opacity: 1;border-color:rgb(220 38 38 / var(--tw-border-opacity, 1))}.dark\:border-red-800:is(.dark *){--tw-border-opacity: 1;border-color:rgb(153 27 27 / var(--tw-border-opacity, 1))}.dark\:bg-blue-900:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(30 58 138 / var(--tw-bg-opacity, 1))}.dark\:bg-blue-900\/20:is(.dark *){background-color:#1e3a8a33}.dark\:bg-blue-900\/30:is(.dark *){background-color:#1e3a8a4d}.dark\:bg-gray-100:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.dark\:bg-gray-700:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(55 65 81 / var(--tw-bg-opacity, 1))}.dark\:bg-gray-800:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1))}.dark\:bg-gray-900:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(17 24 39 / var(--tw-bg-opacity, 1))}.dark\:bg-gray-900\/80:is(.dark *){background-color:#111827cc}.dark\:bg-green-500:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(34 197 94 / var(--tw-bg-opacity, 1))}.dark\:bg-green-900:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(20 83 45 / var(--tw-bg-opacity, 1))}.dark\:bg-green-900\/20:is(.dark *){background-color:#14532d33}.dark\:bg-green-900\/30:is(.dark *){background-color:#14532d4d}.dark\:bg-indigo-900\/30:is(.dark *){background-color:#312e814d}.dark\:bg-orange-900\/30:is(.dark *){background-color:#7c2d124d}.dark\:bg-purple-900\/30:is(.dark *){background-color:#581c874d}.dark\:bg-red-900\/30:is(.dark *){background-color:#7f1d1d4d}.dark\:bg-yellow-900\/30:is(.dark *){background-color:#713f124d}.dark\:from-gray-900:is(.dark *){--tw-gradient-from: #111827 var(--tw-gradient-from-position);--tw-gradient-to: rgb(17 24 39 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.dark\:from-green-600:is(.dark *){--tw-gradient-from: #16a34a var(--tw-gradient-from-position);--tw-gradient-to: rgb(22 163 74 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.dark\:to-gray-800:is(.dark *){--tw-gradient-to: #1f2937 var(--tw-gradient-to-position)}.dark\:to-green-800:is(.dark *){--tw-gradient-to: #166534 var(--tw-gradient-to-position)}.dark\:fill-gray-600:is(.dark *){fill:#4b5563}.dark\:text-blue-300:is(.dark *){--tw-text-opacity: 1;color:rgb(147 197 253 / var(--tw-text-opacity, 1))}.dark\:text-blue-400:is(.dark *){--tw-text-opacity: 1;color:rgb(96 165 250 / var(--tw-text-opacity, 1))}.dark\:text-gray-100:is(.dark *){--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.dark\:text-gray-200:is(.dark *){--tw-text-opacity: 1;color:rgb(229 231 235 / var(--tw-text-opacity, 1))}.dark\:text-gray-300:is(.dark *){--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.dark\:text-gray-400:is(.dark *){--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.dark\:text-gray-600:is(.dark *){--tw-text-opacity: 1;color:rgb(75 85 99 / var(--tw-text-opacity, 1))}.dark\:text-green-400:is(.dark *){--tw-text-opacity: 1;color:rgb(74 222 128 / var(--tw-text-opacity, 1))}.dark\:text-green-700:is(.dark *){--tw-text-opacity: 1;color:rgb(21 128 61 / var(--tw-text-opacity, 1))}.dark\:text-indigo-400:is(.dark *){--tw-text-opacity: 1;color:rgb(129 140 248 / var(--tw-text-opacity, 1))}.dark\:text-orange-400:is(.dark *){--tw-text-opacity: 1;color:rgb(251 146 60 / var(--tw-text-opacity, 1))}.dark\:text-purple-400:is(.dark *){--tw-text-opacity: 1;color:rgb(192 132 252 / var(--tw-text-opacity, 1))}.dark\:text-red-400:is(.dark *){--tw-text-opacity: 1;color:rgb(248 113 113 / var(--tw-text-opacity, 1))}.dark\:text-white:is(.dark *){--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.dark\:text-yellow-400:is(.dark *){--tw-text-opacity: 1;color:rgb(250 204 21 / var(--tw-text-opacity, 1))}.dark\:placeholder-gray-400:is(.dark *)::-moz-placeholder{--tw-placeholder-opacity: 1;color:rgb(156 163 175 / var(--tw-placeholder-opacity, 1))}.dark\:placeholder-gray-400:is(.dark *)::placeholder{--tw-placeholder-opacity: 1;color:rgb(156 163 175 / var(--tw-placeholder-opacity, 1))}.dark\:shadow-gray-900\/50:is(.dark *){--tw-shadow-color: rgb(17 24 39 / .5);--tw-shadow: var(--tw-shadow-colored)}.dark\:hover\:border-blue-600:hover:is(.dark *){--tw-border-opacity: 1;border-color:rgb(37 99 235 / var(--tw-border-opacity, 1))}.dark\:hover\:border-gray-600:hover:is(.dark *){--tw-border-opacity: 1;border-color:rgb(75 85 99 / var(--tw-border-opacity, 1))}.dark\:hover\:border-green-600:hover:is(.dark *){--tw-border-opacity: 1;border-color:rgb(22 163 74 / var(--tw-border-opacity, 1))}.dark\:hover\:bg-blue-900\/30:hover:is(.dark *){background-color:#1e3a8a4d}.dark\:hover\:bg-gray-100:hover:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.dark\:hover\:bg-gray-200:hover:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(229 231 235 / var(--tw-bg-opacity, 1))}.dark\:hover\:bg-gray-700:hover:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(55 65 81 / var(--tw-bg-opacity, 1))}.dark\:hover\:bg-green-600:hover:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(22 163 74 / var(--tw-bg-opacity, 1))}.dark\:hover\:bg-green-900\/30:hover:is(.dark *){background-color:#14532d4d}.dark\:hover\:bg-red-900\/30:hover:is(.dark *){background-color:#7f1d1d4d}.dark\:hover\:text-blue-300:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(147 197 253 / var(--tw-text-opacity, 1))}.dark\:hover\:text-blue-400:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(96 165 250 / var(--tw-text-opacity, 1))}.dark\:hover\:text-gray-200:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(229 231 235 / var(--tw-text-opacity, 1))}.dark\:hover\:text-gray-300:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.dark\:hover\:text-green-300:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(134 239 172 / var(--tw-text-opacity, 1))}.dark\:hover\:text-green-400:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(74 222 128 / var(--tw-text-opacity, 1))}.dark\:hover\:text-green-700:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(21 128 61 / var(--tw-text-opacity, 1))}.dark\:hover\:text-red-400:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(248 113 113 / var(--tw-text-opacity, 1))}.dark\:hover\:text-white:hover:is(.dark *){--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}@media (min-width: 640px){.sm\:h-\[300px\]{height:300px}.sm\:flex-row{flex-direction:row}.sm\:gap-5{gap:1.25rem}.sm\:px-6{padding-left:1.5rem;padding-right:1.5rem}}@media (min-width: 768px){.md\:col-span-2{grid-column:span 2 / span 2}.md\:mb-8{margin-bottom:2rem}.md\:flex{display:flex}.md\:hidden{display:none}.md\:h-\[360px\]{height:360px}.md\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.md\:gap-6{gap:1.5rem}.md\:p-6{padding:1.5rem}.md\:text-2xl{font-size:1.5rem;line-height:2rem}.md\:text-3xl{font-size:1.875rem;line-height:2.25rem}.md\:text-4xl{font-size:2.25rem;line-height:2.5rem}.md\:text-6xl{font-size:3.75rem;line-height:1}}@media (min-width: 1024px){.lg\:fixed{position:fixed}.lg\:top-16{top:4rem}.lg\:z-0{z-index:0}.lg\:col-span-1{grid-column:span 1 / span 1}.lg\:col-span-2{grid-column:span 2 / span 2}.lg\:ml-64{margin-left:16rem}.lg\:hidden{display:none}.lg\:h-\[calc\(100vh-4rem\)\]{height:calc(100vh - 4rem)}.lg\:translate-x-0{--tw-translate-x: 0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.lg\:p-8{padding:2rem}.lg\:px-8{padding-left:2rem;padding-right:2rem}}@media (min-width: 1280px){.xl\:col-span-1{grid-column:span 1 / span 1}.xl\:col-span-2{grid-column:span 2 / span 2}.xl\:col-span-3{grid-column:span 3 / span 3}.xl\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}} diff --git a/frontend/dist/assets/index-tEqMizXb.js b/frontend/dist/assets/index-tEqMizXb.js new file mode 100644 index 0000000000..c6f271c95e --- /dev/null +++ b/frontend/dist/assets/index-tEqMizXb.js @@ -0,0 +1,189 @@ +const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["assets/LandingPage-EXm-0yOi.js","assets/IconButton-DmJ5sToe.js","assets/Pagination-EPc_5dQ6.js","assets/TreeDetailModal-PvfFs-NG.js","assets/useKeyboardNavigation-DT9lybaQ.js","assets/ChartComponents-BKyCVLLP.js","assets/Navbar-BjYzKaxh.js","assets/DarkModeToggle-ZfYhBz0e.js","assets/vendor-BtP0CW_r.js","assets/LoginPage-BeBDoSBE.js","assets/PasswordInput-8zSydSJy.js","assets/RegisterPage-DEfT8Iyu.js","assets/OverviewDashboardPage-ChIgynnA.js","assets/EnhancedStatCard-CVgqadrW.js","assets/GlobalFilters-B5C4gkva.js","assets/GlobalFilters-DdjxYr1F.css","assets/PieChart-DRbmPJpB.js","assets/useSidebarState-CJJaSDFr.js","assets/FinancialDashboardPage-UxW0hI6p.js","assets/useDashboardStats-3Mt6zhuV.js","assets/EcologicalDashboardPage-B0Gl5Ex5.js","assets/MapPage-B-W_JiyY.js","assets/MapPage-25t4MxmA.css","assets/ExportPage-DZ0UV3sF.js","assets/ErrorPages-DbEIbd0-.js"])))=>i.map(i=>d[i]); +var dn=Object.defineProperty;var fn=(e,t,r)=>t in e?dn(e,t,{enumerable:!0,configurable:!0,writable:!0,value:r}):e[t]=r;var ke=(e,t,r)=>fn(e,typeof t!="symbol"?t+"":t,r);import{r as hr,g as Et,a as pr}from"./vendor-BtP0CW_r.js";function hn(e,t){for(var r=0;rn[o]})}}}return Object.freeze(Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}))}(function(){const t=document.createElement("link").relList;if(t&&t.supports&&t.supports("modulepreload"))return;for(const o of document.querySelectorAll('link[rel="modulepreload"]'))n(o);new MutationObserver(o=>{for(const a of o)if(a.type==="childList")for(const s of a.addedNodes)s.tagName==="LINK"&&s.rel==="modulepreload"&&n(s)}).observe(document,{childList:!0,subtree:!0});function r(o){const a={};return o.integrity&&(a.integrity=o.integrity),o.referrerPolicy&&(a.referrerPolicy=o.referrerPolicy),o.crossOrigin==="use-credentials"?a.credentials="include":o.crossOrigin==="anonymous"?a.credentials="omit":a.credentials="same-origin",a}function n(o){if(o.ep)return;o.ep=!0;const a=r(o);fetch(o.href,a)}})();var ot={exports:{}},xe={};/** + * @license React + * react-jsx-runtime.production.min.js + * + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */var zt;function pn(){if(zt)return xe;zt=1;var e=hr(),t=Symbol.for("react.element"),r=Symbol.for("react.fragment"),n=Object.prototype.hasOwnProperty,o=e.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,a={key:!0,ref:!0,__self:!0,__source:!0};function s(c,i,u){var d,l={},p=null,m=null;u!==void 0&&(p=""+u),i.key!==void 0&&(p=""+i.key),i.ref!==void 0&&(m=i.ref);for(d in i)n.call(i,d)&&!a.hasOwnProperty(d)&&(l[d]=i[d]);if(c&&c.defaultProps)for(d in i=c.defaultProps,i)l[d]===void 0&&(l[d]=i[d]);return{$$typeof:t,type:c,key:p,ref:m,props:l,_owner:o.current}}return xe.Fragment=r,xe.jsx=s,xe.jsxs=s,xe}var Bt;function mn(){return Bt||(Bt=1,ot.exports=pn()),ot.exports}var h=mn(),f=hr();const F=Et(f),gn=hn({__proto__:null,default:F},[f]);var Le={},Ut;function vn(){if(Ut)return Le;Ut=1;var e=pr();return Le.createRoot=e.createRoot,Le.hydrateRoot=e.hydrateRoot,Le}var yn=vn();const xn=Et(yn),bn="modulepreload",wn=function(e){return"/"+e},Ft={},W=function(t,r,n){let o=Promise.resolve();if(r&&r.length>0){let s=function(u){return Promise.all(u.map(d=>Promise.resolve(d).then(l=>({status:"fulfilled",value:l}),l=>({status:"rejected",reason:l}))))};document.getElementsByTagName("link");const c=document.querySelector("meta[property=csp-nonce]"),i=(c==null?void 0:c.nonce)||(c==null?void 0:c.getAttribute("nonce"));o=s(r.map(u=>{if(u=wn(u),u in Ft)return;Ft[u]=!0;const d=u.endsWith(".css"),l=d?'[rel="stylesheet"]':"";if(document.querySelector(`link[href="${u}"]${l}`))return;const p=document.createElement("link");if(p.rel=d?"stylesheet":bn,d||(p.as="script"),p.crossOrigin="",p.href=u,i&&p.setAttribute("nonce",i),document.head.appendChild(p),d)return new Promise((m,v)=>{p.addEventListener("load",m),p.addEventListener("error",()=>v(new Error(`Unable to preload CSS for ${u}`)))})}))}function a(s){const c=new Event("vite:preloadError",{cancelable:!0});if(c.payload=s,window.dispatchEvent(c),!c.defaultPrevented)throw s}return o.then(s=>{for(const c of s||[])c.status==="rejected"&&a(c.reason);return t().catch(a)})};var Sn=pr();const zs=Et(Sn);/** + * @remix-run/router v1.23.0 + * + * Copyright (c) Remix Software Inc. + * + * This source code is licensed under the MIT license found in the + * LICENSE.md file in the root directory of this source tree. + * + * @license MIT + */function Ee(){return Ee=Object.assign?Object.assign.bind():function(e){for(var t=1;t"u")throw new Error(t)}function mr(e,t){if(!e){typeof console<"u"&&console.warn(t);try{throw new Error(t)}catch{}}}function Rn(){return Math.random().toString(36).substr(2,8)}function qt(e,t){return{usr:e.state,key:e.key,idx:t}}function dt(e,t,r,n){return r===void 0&&(r=null),Ee({pathname:typeof e=="string"?e:e.pathname,search:"",hash:""},typeof t=="string"?de(t):t,{state:r,key:t&&t.key||n||Rn()})}function We(e){let{pathname:t="/",search:r="",hash:n=""}=e;return r&&r!=="?"&&(t+=r.charAt(0)==="?"?r:"?"+r),n&&n!=="#"&&(t+=n.charAt(0)==="#"?n:"#"+n),t}function de(e){let t={};if(e){let r=e.indexOf("#");r>=0&&(t.hash=e.substr(r),e=e.substr(0,r));let n=e.indexOf("?");n>=0&&(t.search=e.substr(n),e=e.substr(0,n)),e&&(t.pathname=e)}return t}function Cn(e,t,r,n){n===void 0&&(n={});let{window:o=document.defaultView,v5Compat:a=!1}=n,s=o.history,c=Y.Pop,i=null,u=d();u==null&&(u=0,s.replaceState(Ee({},s.state,{idx:u}),""));function d(){return(s.state||{idx:null}).idx}function l(){c=Y.Pop;let g=d(),w=g==null?null:g-u;u=g,i&&i({action:c,location:y.location,delta:w})}function p(g,w){c=Y.Push;let S=dt(y.location,g,w);u=d()+1;let x=qt(S,u),R=y.createHref(S);try{s.pushState(x,"",R)}catch(j){if(j instanceof DOMException&&j.name==="DataCloneError")throw j;o.location.assign(R)}a&&i&&i({action:c,location:y.location,delta:1})}function m(g,w){c=Y.Replace;let S=dt(y.location,g,w);u=d();let x=qt(S,u),R=y.createHref(S);s.replaceState(x,"",R),a&&i&&i({action:c,location:y.location,delta:0})}function v(g){let w=o.location.origin!=="null"?o.location.origin:o.location.href,S=typeof g=="string"?g:We(g);return S=S.replace(/ $/,"%20"),I(w,"No window.location.(origin|href) available to create URL for href: "+S),new URL(S,w)}let y={get action(){return c},get location(){return e(o,s)},listen(g){if(i)throw new Error("A history only accepts one active listener");return o.addEventListener(Wt,l),i=g,()=>{o.removeEventListener(Wt,l),i=null}},createHref(g){return t(o,g)},createURL:v,encodeLocation(g){let w=v(g);return{pathname:w.pathname,search:w.search,hash:w.hash}},push:p,replace:m,go(g){return s.go(g)}};return y}var Vt;(function(e){e.data="data",e.deferred="deferred",e.redirect="redirect",e.error="error"})(Vt||(Vt={}));function Pn(e,t,r){return r===void 0&&(r="/"),jn(e,t,r)}function jn(e,t,r,n){let o=typeof t=="string"?de(t):t,a=Rt(o.pathname||"/",r);if(a==null)return null;let s=gr(e);_n(s);let c=null;for(let i=0;c==null&&i{let i={relativePath:c===void 0?a.path||"":c,caseSensitive:a.caseSensitive===!0,childrenIndex:s,route:a};i.relativePath.startsWith("/")&&(I(i.relativePath.startsWith(n),'Absolute route path "'+i.relativePath+'" nested under path '+('"'+n+'" is not valid. An absolute child route path ')+"must start with the combined path of all its parent routes."),i.relativePath=i.relativePath.slice(n.length));let u=H([n,i.relativePath]),d=r.concat(i);a.children&&a.children.length>0&&(I(a.index!==!0,"Index routes must not have child routes. Please remove "+('all child routes from route path "'+u+'".')),gr(a.children,t,d,u)),!(a.path==null&&!a.index)&&t.push({path:u,score:Nn(u,a.index),routesMeta:d})};return e.forEach((a,s)=>{var c;if(a.path===""||!((c=a.path)!=null&&c.includes("?")))o(a,s);else for(let i of vr(a.path))o(a,s,i)}),t}function vr(e){let t=e.split("/");if(t.length===0)return[];let[r,...n]=t,o=r.endsWith("?"),a=r.replace(/\?$/,"");if(n.length===0)return o?[a,""]:[a];let s=vr(n.join("/")),c=[];return c.push(...s.map(i=>i===""?a:[a,i].join("/"))),o&&c.push(...s),c.map(i=>e.startsWith("/")&&i===""?"/":i)}function _n(e){e.sort((t,r)=>t.score!==r.score?r.score-t.score:$n(t.routesMeta.map(n=>n.childrenIndex),r.routesMeta.map(n=>n.childrenIndex)))}const In=/^:[\w-]+$/,On=3,Tn=2,An=1,kn=10,Ln=-2,Gt=e=>e==="*";function Nn(e,t){let r=e.split("/"),n=r.length;return r.some(Gt)&&(n+=Ln),t&&(n+=Tn),r.filter(o=>!Gt(o)).reduce((o,a)=>o+(In.test(a)?On:a===""?An:kn),n)}function $n(e,t){return e.length===t.length&&e.slice(0,-1).every((n,o)=>n===t[o])?e[e.length-1]-t[t.length-1]:0}function Dn(e,t,r){let{routesMeta:n}=e,o={},a="/",s=[];for(let c=0;c{let{paramName:p,isOptional:m}=d;if(p==="*"){let y=c[l]||"";s=a.slice(0,a.length-y.length).replace(/(.)\/+$/,"$1")}const v=c[l];return m&&!v?u[p]=void 0:u[p]=(v||"").replace(/%2F/g,"/"),u},{}),pathname:a,pathnameBase:s,pattern:e}}function zn(e,t,r){t===void 0&&(t=!1),r===void 0&&(r=!0),mr(e==="*"||!e.endsWith("*")||e.endsWith("/*"),'Route path "'+e+'" will be treated as if it were '+('"'+e.replace(/\*$/,"/*")+'" because the `*` character must ')+"always follow a `/` in the pattern. To get rid of this warning, "+('please change the route path to "'+e.replace(/\*$/,"/*")+'".'));let n=[],o="^"+e.replace(/\/*\*?$/,"").replace(/^\/*/,"/").replace(/[\\.*+^${}|()[\]]/g,"\\$&").replace(/\/:([\w-]+)(\?)?/g,(s,c,i)=>(n.push({paramName:c,isOptional:i!=null}),i?"/?([^\\/]+)?":"/([^\\/]+)"));return e.endsWith("*")?(n.push({paramName:"*"}),o+=e==="*"||e==="/*"?"(.*)$":"(?:\\/(.+)|\\/*)$"):r?o+="\\/*$":e!==""&&e!=="/"&&(o+="(?:(?=\\/|$))"),[new RegExp(o,t?void 0:"i"),n]}function Bn(e){try{return e.split("/").map(t=>decodeURIComponent(t).replace(/\//g,"%2F")).join("/")}catch(t){return mr(!1,'The URL path "'+e+'" could not be decoded because it is is a malformed URL segment. This is probably due to a bad percent '+("encoding ("+t+").")),e}}function Rt(e,t){if(t==="/")return e;if(!e.toLowerCase().startsWith(t.toLowerCase()))return null;let r=t.endsWith("/")?t.length-1:t.length,n=e.charAt(r);return n&&n!=="/"?null:e.slice(r)||"/"}function Un(e,t){t===void 0&&(t="/");let{pathname:r,search:n="",hash:o=""}=typeof e=="string"?de(e):e;return{pathname:r?r.startsWith("/")?r:Fn(r,t):t,search:Vn(n),hash:Gn(o)}}function Fn(e,t){let r=t.replace(/\/+$/,"").split("/");return e.split("/").forEach(o=>{o===".."?r.length>1&&r.pop():o!=="."&&r.push(o)}),r.length>1?r.join("/"):"/"}function at(e,t,r,n){return"Cannot include a '"+e+"' character in a manually specified "+("`to."+t+"` field ["+JSON.stringify(n)+"]. Please separate it out to the ")+("`to."+r+"` field. Alternatively you may provide the full path as ")+'a string in and the router will parse it for you.'}function Wn(e){return e.filter((t,r)=>r===0||t.route.path&&t.route.path.length>0)}function Ct(e,t){let r=Wn(e);return t?r.map((n,o)=>o===r.length-1?n.pathname:n.pathnameBase):r.map(n=>n.pathnameBase)}function Pt(e,t,r,n){n===void 0&&(n=!1);let o;typeof e=="string"?o=de(e):(o=Ee({},e),I(!o.pathname||!o.pathname.includes("?"),at("?","pathname","search",o)),I(!o.pathname||!o.pathname.includes("#"),at("#","pathname","hash",o)),I(!o.search||!o.search.includes("#"),at("#","search","hash",o)));let a=e===""||o.pathname==="",s=a?"/":o.pathname,c;if(s==null)c=r;else{let l=t.length-1;if(!n&&s.startsWith("..")){let p=s.split("/");for(;p[0]==="..";)p.shift(),l-=1;o.pathname=p.join("/")}c=l>=0?t[l]:"/"}let i=Un(o,c),u=s&&s!=="/"&&s.endsWith("/"),d=(a||s===".")&&r.endsWith("/");return!i.pathname.endsWith("/")&&(u||d)&&(i.pathname+="/"),i}const H=e=>e.join("/").replace(/\/\/+/g,"/"),qn=e=>e.replace(/\/+$/,"").replace(/^\/*/,"/"),Vn=e=>!e||e==="?"?"":e.startsWith("?")?e:"?"+e,Gn=e=>!e||e==="#"?"":e.startsWith("#")?e:"#"+e;function Jn(e){return e!=null&&typeof e.status=="number"&&typeof e.statusText=="string"&&typeof e.internal=="boolean"&&"data"in e}const yr=["post","put","patch","delete"];new Set(yr);const Yn=["get",...yr];new Set(Yn);/** + * React Router v6.30.1 + * + * Copyright (c) Remix Software Inc. + * + * This source code is licensed under the MIT license found in the + * LICENSE.md file in the root directory of this source tree. + * + * @license MIT + */function Re(){return Re=Object.assign?Object.assign.bind():function(e){for(var t=1;t{c.current=!0}),f.useCallback(function(u,d){if(d===void 0&&(d={}),!c.current)return;if(typeof u=="number"){n.go(u);return}let l=Pt(u,JSON.parse(s),a,d.relative==="path");e==null&&t!=="/"&&(l.pathname=l.pathname==="/"?t:H([t,l.pathname])),(d.replace?n.replace:n.push)(l,d.state,d)},[t,n,s,a,e])}function wr(e,t){let{relative:r}=t===void 0?{}:t,{future:n}=f.useContext(K),{matches:o}=f.useContext(X),{pathname:a}=he(),s=JSON.stringify(Ct(o,n.v7_relativeSplatPath));return f.useMemo(()=>Pt(e,JSON.parse(s),a,r==="path"),[e,s,a,r])}function Zn(e,t){return Qn(e,t)}function Qn(e,t,r,n){fe()||I(!1);let{navigator:o}=f.useContext(K),{matches:a}=f.useContext(X),s=a[a.length-1],c=s?s.params:{};s&&s.pathname;let i=s?s.pathnameBase:"/";s&&s.route;let u=he(),d;if(t){var l;let g=typeof t=="string"?de(t):t;i==="/"||(l=g.pathname)!=null&&l.startsWith(i)||I(!1),d=g}else d=u;let p=d.pathname||"/",m=p;if(i!=="/"){let g=i.replace(/^\//,"").split("/");m="/"+p.replace(/^\//,"").split("/").slice(g.length).join("/")}let v=Pn(e,{pathname:m}),y=oo(v&&v.map(g=>Object.assign({},g,{params:Object.assign({},c,g.params),pathname:H([i,o.encodeLocation?o.encodeLocation(g.pathname).pathname:g.pathname]),pathnameBase:g.pathnameBase==="/"?i:H([i,o.encodeLocation?o.encodeLocation(g.pathnameBase).pathname:g.pathnameBase])})),a,r,n);return t&&y?f.createElement(Je.Provider,{value:{location:Re({pathname:"/",search:"",hash:"",state:null,key:"default"},d),navigationType:Y.Pop}},y):y}function eo(){let e=co(),t=Jn(e)?e.status+" "+e.statusText:e instanceof Error?e.message:JSON.stringify(e),r=e instanceof Error?e.stack:null,o={padding:"0.5rem",backgroundColor:"rgba(200,200,200, 0.5)"};return f.createElement(f.Fragment,null,f.createElement("h2",null,"Unexpected Application Error!"),f.createElement("h3",{style:{fontStyle:"italic"}},t),r?f.createElement("pre",{style:o},r):null,null)}const to=f.createElement(eo,null);class ro extends f.Component{constructor(t){super(t),this.state={location:t.location,revalidation:t.revalidation,error:t.error}}static getDerivedStateFromError(t){return{error:t}}static getDerivedStateFromProps(t,r){return r.location!==t.location||r.revalidation!=="idle"&&t.revalidation==="idle"?{error:t.error,location:t.location,revalidation:t.revalidation}:{error:t.error!==void 0?t.error:r.error,location:r.location,revalidation:t.revalidation||r.revalidation}}componentDidCatch(t,r){console.error("React Router caught the following error during render",t,r)}render(){return this.state.error!==void 0?f.createElement(X.Provider,{value:this.props.routeContext},f.createElement(xr.Provider,{value:this.state.error,children:this.props.component})):this.props.children}}function no(e){let{routeContext:t,match:r,children:n}=e,o=f.useContext(jt);return o&&o.static&&o.staticContext&&(r.route.errorElement||r.route.ErrorBoundary)&&(o.staticContext._deepestRenderedBoundaryId=r.route.id),f.createElement(X.Provider,{value:t},n)}function oo(e,t,r,n){var o;if(t===void 0&&(t=[]),r===void 0&&(r=null),n===void 0&&(n=null),e==null){var a;if(!r)return null;if(r.errors)e=r.matches;else if((a=n)!=null&&a.v7_partialHydration&&t.length===0&&!r.initialized&&r.matches.length>0)e=r.matches;else return null}let s=e,c=(o=r)==null?void 0:o.errors;if(c!=null){let d=s.findIndex(l=>l.route.id&&(c==null?void 0:c[l.route.id])!==void 0);d>=0||I(!1),s=s.slice(0,Math.min(s.length,d+1))}let i=!1,u=-1;if(r&&n&&n.v7_partialHydration)for(let d=0;d=0?s=s.slice(0,u+1):s=[s[0]];break}}}return s.reduceRight((d,l,p)=>{let m,v=!1,y=null,g=null;r&&(m=c&&l.route.id?c[l.route.id]:void 0,y=l.route.errorElement||to,i&&(u<0&&p===0?(uo("route-fallback"),v=!0,g=null):u===p&&(v=!0,g=l.route.hydrateFallbackElement||null)));let w=t.concat(s.slice(0,p+1)),S=()=>{let x;return m?x=y:v?x=g:l.route.Component?x=f.createElement(l.route.Component,null):l.route.element?x=l.route.element:x=d,f.createElement(no,{match:l,routeContext:{outlet:d,matches:w,isDataRoute:r!=null},children:x})};return r&&(l.route.ErrorBoundary||l.route.errorElement||p===0)?f.createElement(ro,{location:r.location,revalidation:r.revalidation,component:y,error:m,children:S(),routeContext:{outlet:null,matches:w,isDataRoute:!0}}):S()},null)}var Sr=function(e){return e.UseBlocker="useBlocker",e.UseRevalidator="useRevalidator",e.UseNavigateStable="useNavigate",e}(Sr||{}),Er=function(e){return e.UseBlocker="useBlocker",e.UseLoaderData="useLoaderData",e.UseActionData="useActionData",e.UseRouteError="useRouteError",e.UseNavigation="useNavigation",e.UseRouteLoaderData="useRouteLoaderData",e.UseMatches="useMatches",e.UseRevalidator="useRevalidator",e.UseNavigateStable="useNavigate",e.UseRouteId="useRouteId",e}(Er||{});function ao(e){let t=f.useContext(jt);return t||I(!1),t}function so(e){let t=f.useContext(Hn);return t||I(!1),t}function io(e){let t=f.useContext(X);return t||I(!1),t}function Rr(e){let t=io(),r=t.matches[t.matches.length-1];return r.route.id||I(!1),r.route.id}function co(){var e;let t=f.useContext(xr),r=so(),n=Rr();return t!==void 0?t:(e=r.errors)==null?void 0:e[n]}function lo(){let{router:e}=ao(Sr.UseNavigateStable),t=Rr(Er.UseNavigateStable),r=f.useRef(!1);return br(()=>{r.current=!0}),f.useCallback(function(o,a){a===void 0&&(a={}),r.current&&(typeof o=="number"?e.navigate(o):e.navigate(o,Re({fromRouteId:t},a)))},[e,t])}const Jt={};function uo(e,t,r){Jt[e]||(Jt[e]=!0)}function fo(e,t){e==null||e.v7_startTransition,e==null||e.v7_relativeSplatPath}function ho(e){let{to:t,replace:r,state:n,relative:o}=e;fe()||I(!1);let{future:a,static:s}=f.useContext(K),{matches:c}=f.useContext(X),{pathname:i}=he(),u=Ye(),d=Pt(t,Ct(c,a.v7_relativeSplatPath),i,o==="path"),l=JSON.stringify(d);return f.useEffect(()=>u(JSON.parse(l),{replace:r,state:n,relative:o}),[u,l,o,r,n]),null}function D(e){I(!1)}function po(e){let{basename:t="/",children:r=null,location:n,navigationType:o=Y.Pop,navigator:a,static:s=!1,future:c}=e;fe()&&I(!1);let i=t.replace(/^\/*/,"/"),u=f.useMemo(()=>({basename:i,navigator:a,static:s,future:Re({v7_relativeSplatPath:!1},c)}),[i,c,a,s]);typeof n=="string"&&(n=de(n));let{pathname:d="/",search:l="",hash:p="",state:m=null,key:v="default"}=n,y=f.useMemo(()=>{let g=Rt(d,i);return g==null?null:{location:{pathname:g,search:l,hash:p,state:m,key:v},navigationType:o}},[i,d,l,p,m,v,o]);return y==null?null:f.createElement(K.Provider,{value:u},f.createElement(Je.Provider,{children:r,value:y}))}function mo(e){let{children:t,location:r}=e;return Zn(ft(t),r)}new Promise(()=>{});function ft(e,t){t===void 0&&(t=[]);let r=[];return f.Children.forEach(e,(n,o)=>{if(!f.isValidElement(n))return;let a=[...t,o];if(n.type===f.Fragment){r.push.apply(r,ft(n.props.children,a));return}n.type!==D&&I(!1),!n.props.index||!n.props.children||I(!1);let s={id:n.props.id||a.join("-"),caseSensitive:n.props.caseSensitive,element:n.props.element,Component:n.props.Component,index:n.props.index,path:n.props.path,loader:n.props.loader,action:n.props.action,errorElement:n.props.errorElement,ErrorBoundary:n.props.ErrorBoundary,hasErrorBoundary:n.props.ErrorBoundary!=null||n.props.errorElement!=null,shouldRevalidate:n.props.shouldRevalidate,handle:n.props.handle,lazy:n.props.lazy};n.props.children&&(s.children=ft(n.props.children,a)),r.push(s)}),r}/** + * React Router DOM v6.30.1 + * + * Copyright (c) Remix Software Inc. + * + * This source code is licensed under the MIT license found in the + * LICENSE.md file in the root directory of this source tree. + * + * @license MIT + */function ht(){return ht=Object.assign?Object.assign.bind():function(e){for(var t=1;t=0)&&(r[o]=e[o]);return r}function vo(e){return!!(e.metaKey||e.altKey||e.ctrlKey||e.shiftKey)}function yo(e,t){return e.button===0&&(!t||t==="_self")&&!vo(e)}function pt(e){return e===void 0&&(e=""),new URLSearchParams(typeof e=="string"||Array.isArray(e)||e instanceof URLSearchParams?e:Object.keys(e).reduce((t,r)=>{let n=e[r];return t.concat(Array.isArray(n)?n.map(o=>[r,o]):[[r,n]])},[]))}function xo(e,t){let r=pt(e);return t&&t.forEach((n,o)=>{r.has(o)||t.getAll(o).forEach(a=>{r.append(o,a)})}),r}const bo=["onClick","relative","reloadDocument","replace","state","target","to","preventScrollReset","viewTransition"],wo="6";try{window.__reactRouterVersion=wo}catch{}const So="startTransition",Yt=gn[So];function Eo(e){let{basename:t,children:r,future:n,window:o}=e,a=f.useRef();a.current==null&&(a.current=En({window:o,v5Compat:!0}));let s=a.current,[c,i]=f.useState({action:s.action,location:s.location}),{v7_startTransition:u}=n||{},d=f.useCallback(l=>{u&&Yt?Yt(()=>i(l)):i(l)},[i,u]);return f.useLayoutEffect(()=>s.listen(d),[s,d]),f.useEffect(()=>fo(n),[n]),f.createElement(po,{basename:t,children:r,location:c.location,navigationType:c.action,navigator:s,future:n})}const Ro=typeof window<"u"&&typeof window.document<"u"&&typeof window.document.createElement<"u",Co=/^(?:[a-z][a-z0-9+.-]*:|\/\/)/i,Bs=f.forwardRef(function(t,r){let{onClick:n,relative:o,reloadDocument:a,replace:s,state:c,target:i,to:u,preventScrollReset:d,viewTransition:l}=t,p=go(t,bo),{basename:m}=f.useContext(K),v,y=!1;if(typeof u=="string"&&Co.test(u)&&(v=u,Ro))try{let x=new URL(window.location.href),R=u.startsWith("//")?new URL(x.protocol+u):new URL(u),j=Rt(R.pathname,m);R.origin===x.origin&&j!=null?u=j+R.search+R.hash:y=!0}catch{}let g=Kn(u,{relative:o}),w=Po(u,{replace:s,state:c,target:i,preventScrollReset:d,relative:o,viewTransition:l});function S(x){n&&n(x),x.defaultPrevented||w(x)}return f.createElement("a",ht({},p,{href:v||g,onClick:y||a?n:S,ref:r,target:i}))});var Ht;(function(e){e.UseScrollRestoration="useScrollRestoration",e.UseSubmit="useSubmit",e.UseSubmitFetcher="useSubmitFetcher",e.UseFetcher="useFetcher",e.useViewTransitionState="useViewTransitionState"})(Ht||(Ht={}));var Kt;(function(e){e.UseFetcher="useFetcher",e.UseFetchers="useFetchers",e.UseScrollRestoration="useScrollRestoration"})(Kt||(Kt={}));function Po(e,t){let{target:r,replace:n,state:o,preventScrollReset:a,relative:s,viewTransition:c}=t===void 0?{}:t,i=Ye(),u=he(),d=wr(e,{relative:s});return f.useCallback(l=>{if(yo(l,r)){l.preventDefault();let p=n!==void 0?n:We(u)===We(d);i(e,{replace:p,state:o,preventScrollReset:a,relative:s,viewTransition:c})}},[u,i,d,n,o,r,e,a,s,c])}function Us(e){let t=f.useRef(pt(e)),r=f.useRef(!1),n=he(),o=f.useMemo(()=>xo(n.search,r.current?null:t.current),[n.search]),a=Ye(),s=f.useCallback((c,i)=>{const u=pt(typeof c=="function"?c(o):c);r.current=!0,a("?"+u,i)},[a,o]);return[o,s]}const jo="https://project-final-frontend-4bia.onrender.com/api",_o=["failed to fetch","network request failed","timeout","connection refused","service unavailable","internal server error"],J={maxAttempts:4,maxDelay:2e4,coldStartTimeouts:[3e4,45e3,6e4,9e4],retryDelays:[5e3,1e4,15e3,2e4]},Io=e=>{if(!e||!e.message)return!1;const t=e.message.toLowerCase();return _o.some(r=>t.includes(r))},Oo=e=>J.retryDelays[e-1]||J.maxDelay,Cr=()=>{try{return localStorage.getItem("authToken")}catch(e){return console.warn("Failed to retrieve auth token from localStorage:",e),null}},Xt=(e={})=>{const t={"Content-Type":"application/json",...e},r=Cr();return r&&(t.Authorization=`Bearer ${r}`),t},To=(e={})=>{const t={...e},r=Cr();return r&&(t.Authorization=`Bearer ${r}`),t};class Ao{constructor(){this.baseURL=jo}async request(t,r={}){const n=`${this.baseURL}${t}`,{retryAttempts:o=0,onRetry:a}=r,s=oi.abort(),s),d=await fetch(n,{...c,signal:i.signal});if(clearTimeout(u),!d.ok){const l=await d.json().catch(()=>({})),p=new Error(l.message||`HTTP error! status: ${d.status}`);throw p.status=d.status,p}return await d.json()}catch(i){if(i.name==="AbortError"){const d=new Error("Request timeout");throw d.name="AbortError",d.status=408,d}if(Io(i)&&osetTimeout(p,l)),this.request(t,{...r,retryAttempts:d,onRetry:a})}throw console.error("API request failed:",i),i}}async get(t,r={}){const n=new URLSearchParams(r).toString(),o=n?`${t}?${n}`:t;return this.request(o,{method:"GET"})}async post(t,r={},n={}){return this.request(t,{method:"POST",body:JSON.stringify(r),...n})}async put(t,r={}){return this.request(t,{method:"PUT",body:JSON.stringify(r)})}async patch(t,r={}){return this.request(t,{method:"PATCH",body:JSON.stringify(r)})}async delete(t){return this.request(t,{method:"DELETE"})}async uploadFile(t,r){return this.request(t,{method:"POST",body:r,headers:To()})}async downloadFile(t,r={},n){const o=new URLSearchParams(r).toString(),a=`${this.baseURL}${t}${o?`?${o}`:""}`;try{const s=await fetch(a,{method:"GET",headers:Xt()});if(!s.ok){const d=await s.json().catch(()=>({})),l=new Error(d.message||`HTTP error! status: ${s.status}`);throw l.status=s.status,l}const c=await s.blob(),i=window.URL.createObjectURL(c),u=document.createElement("a");return u.href=i,u.download=n||"download",document.body.appendChild(u),u.click(),document.body.removeChild(u),window.URL.revokeObjectURL(i),{success:!0,filename:n}}catch(s){throw console.error("File download failed:",s),s}}}const Q=new Ao,st={register:(e,t={})=>Q.post("/auth/register",e,t),login:(e,t={})=>Q.post("/auth/login",e,t),logout:()=>Q.post("/auth/logout"),logoutAll:()=>Q.post("/auth/logout-all"),refreshToken:()=>Q.post("/auth/refresh"),getProfile:()=>Q.get("/auth/profile"),updateProfile:e=>Q.put("/auth/profile",e)},Pr=f.createContext(),ko=()=>{const e=f.useContext(Pr);if(!e)throw new Error("useAuth must be used within an AuthProvider");return e},Lo=({children:e})=>{const[t,r]=f.useState(null),[n,o]=f.useState(!0),a=Ye();f.useEffect(()=>{const l=localStorage.getItem("authToken"),p=localStorage.getItem("userData");if(l&&p)try{r(JSON.parse(p))}catch(m){console.error("Error parsing user data:",m),localStorage.removeItem("authToken"),localStorage.removeItem("userData")}o(!1)},[]);const d={user:t,loading:n,login:async(l,p,m)=>{try{o(!0);const v=await st.login({email:l,password:p},{onRetry:m}),{token:y,user:g}=v.data;return localStorage.setItem("authToken",y),localStorage.setItem("userData",JSON.stringify(g)),r(g),a("/dashboard"),{success:!0}}catch(v){return{success:!1,error:v.message}}finally{o(!1)}},register:async(l,p,m)=>{try{if(o(!0),p!==m)throw new Error("Passwords do not match");const v=await st.register({email:l,password:p}),{token:y,user:g}=v.data;return localStorage.setItem("authToken",y),localStorage.setItem("userData",JSON.stringify(g)),r(g),a("/dashboard"),{success:!0}}catch(v){return{success:!1,error:v.message}}finally{o(!1)}},logout:async()=>{try{await st.logout()}catch(l){console.error("Logout API call failed:",l)}finally{localStorage.removeItem("authToken"),localStorage.removeItem("userData"),r(null),a("/")}},isAdmin:()=>(t==null?void 0:t.role)==="admin"};return h.jsx(Pr.Provider,{value:d,children:e})},No=()=>{const{user:e,loading:t}=ko();return{user:e,loading:t,isAuthenticated:!!e,isLoading:t}};var A=function(){return A=Object.assign||function(t){for(var r,n=1,o=arguments.length;n0?T(pe,--L):0,ie--,O===10&&(ie=1,Ke--),O}function $(){return O=L2||gt(O)>3?"":" "}function Vo(e,t){for(;--t&&$()&&!(O<48||O>102||O>57&&O<65||O>70&&O<97););return Ze(e,Me()+(t<6&&te()==32&&$()==32))}function vt(e){for(;$();)switch(O){case e:return L;case 34:case 39:e!==34&&e!==39&&vt(O);break;case 40:e===41&&vt(e);break;case 92:$();break}return L}function Go(e,t){for(;$()&&e+O!==57;)if(e+O===84&&te()===47)break;return"/*"+Ze(t,L-1)+"*"+It(e===47?e:$())}function Jo(e){for(;!gt(te());)$();return Ze(e,L)}function Yo(e){return Wo(ze("",null,null,null,[""],e=Fo(e),0,[0],e))}function ze(e,t,r,n,o,a,s,c,i){for(var u=0,d=0,l=s,p=0,m=0,v=0,y=1,g=1,w=1,S=0,x="",R=o,j=a,P=n,E=x;g;)switch(v=S,S=$()){case 40:if(v!=108&&T(E,l-1)==58){De(E+=b(it(S),"&","&\f"),"&\f",Ir(u?c[u-1]:0))!=-1&&(w=-1);break}case 34:case 39:case 91:E+=it(S);break;case 9:case 10:case 13:case 32:E+=qo(v);break;case 92:E+=Vo(Me()-1,7);continue;case 47:switch(te()){case 42:case 47:we(Ho(Go($(),Me()),t,r,i),i);break;default:E+="/"}break;case 123*y:c[u++]=M(E)*w;case 125*y:case 59:case 0:switch(S){case 0:case 125:g=0;case 59+d:w==-1&&(E=b(E,/\f/g,"")),m>0&&M(E)-l&&we(m>32?er(E+";",n,r,l-1,i):er(b(E," ","")+";",n,r,l-2,i),i);break;case 59:E+=";";default:if(we(P=Qt(E,t,r,u,d,o,c,x,R=[],j=[],l,a),a),S===123)if(d===0)ze(E,t,P,P,R,a,l,c,j);else switch(p===99&&T(E,3)===110?100:p){case 100:case 108:case 109:case 115:ze(e,P,P,n&&we(Qt(e,P,P,0,0,o,c,x,o,R=[],l,j),j),o,j,l,c,n?R:j);break;default:ze(E,P,P,P,[""],j,0,c,j)}}u=d=m=0,y=w=1,x=E="",l=s;break;case 58:l=1+M(E),m=v;default:if(y<1){if(S==123)--y;else if(S==125&&y++==0&&Uo()==125)continue}switch(E+=It(S),S*y){case 38:w=d>0?1:(E+="\f",-1);break;case 44:c[u++]=(M(E)-1)*w,w=1;break;case 64:te()===45&&(E+=it($())),p=te(),d=l=M(x=E+=Jo(Me())),S++;break;case 45:v===45&&M(E)==2&&(y=0)}}return a}function Qt(e,t,r,n,o,a,s,c,i,u,d,l){for(var p=o-1,m=o===0?a:[""],v=Tr(m),y=0,g=0,w=0;y0?m[S]+" "+x:b(x,/&\f/g,m[S])))&&(i[w++]=R);return Xe(e,t,r,o===0?He:c,i,u,d,l)}function Ho(e,t,r,n){return Xe(e,t,r,jr,It(Bo()),se(e,2,-2),0,n)}function er(e,t,r,n,o){return Xe(e,t,r,_t,se(e,0,n),se(e,n+1,-1),n,o)}function kr(e,t,r){switch(Mo(e,t)){case 5103:return C+"print-"+e+e;case 5737:case 4201:case 3177:case 3433:case 1641:case 4457:case 2921:case 5572:case 6356:case 5844:case 3191:case 6645:case 3005:case 6391:case 5879:case 5623:case 6135:case 4599:case 4855:case 4215:case 6389:case 5109:case 5365:case 5621:case 3829:return C+e+e;case 4789:return Se+e+e;case 5349:case 4246:case 4810:case 6968:case 2756:return C+e+Se+e+_+e+e;case 5936:switch(T(e,t+11)){case 114:return C+e+_+b(e,/[svh]\w+-[tblr]{2}/,"tb")+e;case 108:return C+e+_+b(e,/[svh]\w+-[tblr]{2}/,"tb-rl")+e;case 45:return C+e+_+b(e,/[svh]\w+-[tblr]{2}/,"lr")+e}case 6828:case 4268:case 2903:return C+e+_+e+e;case 6165:return C+e+_+"flex-"+e+e;case 5187:return C+e+b(e,/(\w+).+(:[^]+)/,C+"box-$1$2"+_+"flex-$1$2")+e;case 5443:return C+e+_+"flex-item-"+b(e,/flex-|-self/g,"")+(B(e,/flex-|baseline/)?"":_+"grid-row-"+b(e,/flex-|-self/g,""))+e;case 4675:return C+e+_+"flex-line-pack"+b(e,/align-content|flex-|-self/g,"")+e;case 5548:return C+e+_+b(e,"shrink","negative")+e;case 5292:return C+e+_+b(e,"basis","preferred-size")+e;case 6060:return C+"box-"+b(e,"-grow","")+C+e+_+b(e,"grow","positive")+e;case 4554:return C+b(e,/([^-])(transform)/g,"$1"+C+"$2")+e;case 6187:return b(b(b(e,/(zoom-|grab)/,C+"$1"),/(image-set)/,C+"$1"),e,"")+e;case 5495:case 3959:return b(e,/(image-set\([^]*)/,C+"$1$`$1");case 4968:return b(b(e,/(.+:)(flex-)?(.*)/,C+"box-pack:$3"+_+"flex-pack:$3"),/s.+-b[^;]+/,"justify")+C+e+e;case 4200:if(!B(e,/flex-|baseline/))return _+"grid-column-align"+se(e,t)+e;break;case 2592:case 3360:return _+b(e,"template-","")+e;case 4384:case 3616:return r&&r.some(function(n,o){return t=o,B(n.props,/grid-\w+-end/)})?~De(e+(r=r[t].value),"span",0)?e:_+b(e,"-start","")+e+_+"grid-row-span:"+(~De(r,"span",0)?B(r,/\d+/):+B(r,/\d+/)-+B(e,/\d+/))+";":_+b(e,"-start","")+e;case 4896:case 4128:return r&&r.some(function(n){return B(n.props,/grid-\w+-start/)})?e:_+b(b(e,"-end","-span"),"span ","")+e;case 4095:case 3583:case 4068:case 2532:return b(e,/(.+)-inline(.+)/,C+"$1$2")+e;case 8116:case 7059:case 5753:case 5535:case 5445:case 5701:case 4933:case 4677:case 5533:case 5789:case 5021:case 4765:if(M(e)-1-t>6)switch(T(e,t+1)){case 109:if(T(e,t+4)!==45)break;case 102:return b(e,/(.+:)(.+)-([^]+)/,"$1"+C+"$2-$3$1"+Se+(T(e,t+3)==108?"$3":"$2-$3"))+e;case 115:return~De(e,"stretch",0)?kr(b(e,"stretch","fill-available"),t,r)+e:e}break;case 5152:case 5920:return b(e,/(.+?):(\d+)(\s*\/\s*(span)?\s*(\d+))?(.*)/,function(n,o,a,s,c,i,u){return _+o+":"+a+u+(s?_+o+"-span:"+(c?i:+i-+a)+u:"")+e});case 4949:if(T(e,t+6)===121)return b(e,":",":"+C)+e;break;case 6444:switch(T(e,T(e,14)===45?18:11)){case 120:return b(e,/(.+:)([^;\s!]+)(;|(\s+)?!.+)?/,"$1"+C+(T(e,14)===45?"inline-":"")+"box$3$1"+C+"$2$3$1"+_+"$2box$3")+e;case 100:return b(e,":",":"+_)+e}break;case 5719:case 2647:case 2135:case 3927:case 2391:return b(e,"scroll-","scroll-snap-")+e}return e}function qe(e,t){for(var r="",n=0;n-1&&!e.return)switch(e.type){case _t:e.return=kr(e.value,e.length,r);return;case _r:return qe([G(e,{value:b(e.value,"@","@"+C)})],n);case He:if(e.length)return zo(r=e.props,function(o){switch(B(o,n=/(::plac\w+|:read-\w+)/)){case":read-only":case":read-write":oe(G(e,{props:[b(o,/:(read-\w+)/,":"+Se+"$1")]})),oe(G(e,{props:[o]})),mt(e,{props:Zt(r,n)});break;case"::placeholder":oe(G(e,{props:[b(o,/:(plac\w+)/,":"+C+"input-$1")]})),oe(G(e,{props:[b(o,/:(plac\w+)/,":"+Se+"$1")]})),oe(G(e,{props:[b(o,/:(plac\w+)/,_+"input-$1")]})),oe(G(e,{props:[o]})),mt(e,{props:Zt(r,n)});break}return""})}}var ea={animationIterationCount:1,aspectRatio:1,borderImageOutset:1,borderImageSlice:1,borderImageWidth:1,boxFlex:1,boxFlexGroup:1,boxOrdinalGroup:1,columnCount:1,columns:1,flex:1,flexGrow:1,flexPositive:1,flexShrink:1,flexNegative:1,flexOrder:1,gridRow:1,gridRowEnd:1,gridRowSpan:1,gridRowStart:1,gridColumn:1,gridColumnEnd:1,gridColumnSpan:1,gridColumnStart:1,msGridRow:1,msGridRowSpan:1,msGridColumn:1,msGridColumnSpan:1,fontWeight:1,lineHeight:1,opacity:1,order:1,orphans:1,tabSize:1,widows:1,zIndex:1,zoom:1,WebkitLineClamp:1,fillOpacity:1,floodOpacity:1,stopOpacity:1,strokeDasharray:1,strokeDashoffset:1,strokeMiterlimit:1,strokeOpacity:1,strokeWidth:1},k={},ce=typeof process<"u"&&k!==void 0&&(k.REACT_APP_SC_ATTR||k.SC_ATTR)||"data-styled",Lr="active",Nr="data-styled-version",Qe="6.1.19",Ot=`/*!sc*/ +`,Ve=typeof window<"u"&&typeof document<"u",ta=!!(typeof SC_DISABLE_SPEEDY=="boolean"?SC_DISABLE_SPEEDY:typeof process<"u"&&k!==void 0&&k.REACT_APP_SC_DISABLE_SPEEDY!==void 0&&k.REACT_APP_SC_DISABLE_SPEEDY!==""?k.REACT_APP_SC_DISABLE_SPEEDY!=="false"&&k.REACT_APP_SC_DISABLE_SPEEDY:typeof process<"u"&&k!==void 0&&k.SC_DISABLE_SPEEDY!==void 0&&k.SC_DISABLE_SPEEDY!==""&&k.SC_DISABLE_SPEEDY!=="false"&&k.SC_DISABLE_SPEEDY),et=Object.freeze([]),le=Object.freeze({});function ra(e,t,r){return r===void 0&&(r=le),e.theme!==r.theme&&e.theme||t||r.theme}var $r=new Set(["a","abbr","address","area","article","aside","audio","b","base","bdi","bdo","big","blockquote","body","br","button","canvas","caption","cite","code","col","colgroup","data","datalist","dd","del","details","dfn","dialog","div","dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","hr","html","i","iframe","img","input","ins","kbd","keygen","label","legend","li","link","main","map","mark","menu","menuitem","meta","meter","nav","noscript","object","ol","optgroup","option","output","p","param","picture","pre","progress","q","rp","rt","ruby","s","samp","script","section","select","small","source","span","strong","style","sub","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","track","u","ul","use","var","video","wbr","circle","clipPath","defs","ellipse","foreignObject","g","image","line","linearGradient","marker","mask","path","pattern","polygon","polyline","radialGradient","rect","stop","svg","text","tspan"]),na=/[!"#$%&'()*+,./:;<=>?@[\\\]^`{|}~-]+/g,oa=/(^-|-$)/g;function tr(e){return e.replace(na,"-").replace(oa,"")}var aa=/(a)(d)/gi,Ne=52,rr=function(e){return String.fromCharCode(e+(e>25?39:97))};function yt(e){var t,r="";for(t=Math.abs(e);t>Ne;t=t/Ne|0)r=rr(t%Ne)+r;return(rr(t%Ne)+r).replace(aa,"$1-$2")}var ct,Dr=5381,ae=function(e,t){for(var r=t.length;r;)e=33*e^t.charCodeAt(--r);return e},Mr=function(e){return ae(Dr,e)};function zr(e){return yt(Mr(e)>>>0)}function sa(e){return e.displayName||e.name||"Component"}function lt(e){return typeof e=="string"&&!0}var Br=typeof Symbol=="function"&&Symbol.for,Ur=Br?Symbol.for("react.memo"):60115,ia=Br?Symbol.for("react.forward_ref"):60112,ca={childContextTypes:!0,contextType:!0,contextTypes:!0,defaultProps:!0,displayName:!0,getDefaultProps:!0,getDerivedStateFromError:!0,getDerivedStateFromProps:!0,mixins:!0,propTypes:!0,type:!0},la={name:!0,length:!0,prototype:!0,caller:!0,callee:!0,arguments:!0,arity:!0},Fr={$$typeof:!0,compare:!0,defaultProps:!0,displayName:!0,propTypes:!0,type:!0},ua=((ct={})[ia]={$$typeof:!0,render:!0,defaultProps:!0,displayName:!0,propTypes:!0},ct[Ur]=Fr,ct);function nr(e){return("type"in(t=e)&&t.type.$$typeof)===Ur?Fr:"$$typeof"in e?ua[e.$$typeof]:ca;var t}var da=Object.defineProperty,fa=Object.getOwnPropertyNames,or=Object.getOwnPropertySymbols,ha=Object.getOwnPropertyDescriptor,pa=Object.getPrototypeOf,ar=Object.prototype;function Wr(e,t,r){if(typeof t!="string"){if(ar){var n=pa(t);n&&n!==ar&&Wr(e,n,r)}var o=fa(t);or&&(o=o.concat(or(t)));for(var a=nr(e),s=nr(t),c=0;c0?" Args: ".concat(t.join(", ")):""))}var ma=function(){function e(t){this.groupSizes=new Uint32Array(512),this.length=512,this.tag=t}return e.prototype.indexOfGroup=function(t){for(var r=0,n=0;n=this.groupSizes.length){for(var n=this.groupSizes,o=n.length,a=o;t>=a;)if((a<<=1)<0)throw je(16,"".concat(t));this.groupSizes=new Uint32Array(a),this.groupSizes.set(n),this.length=a;for(var s=o;s=this.length||this.groupSizes[t]===0)return r;for(var n=this.groupSizes[t],o=this.indexOfGroup(t),a=o+n,s=o;s=0){var n=document.createTextNode(r);return this.element.insertBefore(n,this.nodes[t]||null),this.length++,!0}return!1},e.prototype.deleteRule=function(t){this.element.removeChild(this.nodes[t]),this.length--},e.prototype.getRule=function(t){return t0&&(g+="".concat(w,","))}),i+="".concat(v).concat(y,'{content:"').concat(g,'"}').concat(Ot)},d=0;d0?".".concat(t):p},d=i.slice();d.push(function(p){p.type===He&&p.value.includes("&")&&(p.props[0]=p.props[0].replace(Pa,r).replace(n,u))}),s.prefix&&d.push(Qo),d.push(Ko);var l=function(p,m,v,y){m===void 0&&(m=""),v===void 0&&(v=""),y===void 0&&(y="&"),t=y,r=m,n=new RegExp("\\".concat(r,"\\b"),"g");var g=p.replace(ja,""),w=Yo(v||m?"".concat(v," ").concat(m," { ").concat(g," }"):g);s.namespace&&(w=Gr(w,s.namespace));var S=[];return qe(w,Xo(d.concat(Zo(function(x){return S.push(x)})))),S};return l.hash=i.length?i.reduce(function(p,m){return m.name||je(15),ae(p,m.name)},Dr).toString():"",l}var Ia=new Vr,wt=_a(),Jr=F.createContext({shouldForwardProp:void 0,styleSheet:Ia,stylis:wt});Jr.Consumer;F.createContext(void 0);function cr(){return f.useContext(Jr)}var Yr=function(){function e(t,r){var n=this;this.inject=function(o,a){a===void 0&&(a=wt);var s=n.name+a.hash;o.hasNameForId(n.id,s)||o.insertRules(n.id,s,a(n.rules,s,"@keyframes"))},this.name=t,this.id="sc-keyframes-".concat(t),this.rules=r,At(this,function(){throw je(12,String(n.name))})}return e.prototype.getName=function(t){return t===void 0&&(t=wt),this.name+t.hash},e}(),Oa=function(e){return e>="A"&&e<="Z"};function lr(e){for(var t="",r=0;r>>0);if(!r.hasNameForId(this.componentId,s)){var c=n(a,".".concat(s),void 0,this.componentId);r.insertRules(this.componentId,s,c)}o=ee(o,s),this.staticRulesId=s}else{for(var i=ae(this.baseHash,n.hash),u="",d=0;d>>0);r.hasNameForId(this.componentId,m)||r.insertRules(this.componentId,m,n(u,".".concat(m),void 0,this.componentId)),o=ee(o,m)}}return o},e}(),Xr=F.createContext(void 0);Xr.Consumer;var ut={};function La(e,t,r){var n=Tt(e),o=e,a=!lt(e),s=t.attrs,c=s===void 0?et:s,i=t.componentId,u=i===void 0?function(R,j){var P=typeof R!="string"?"sc":tr(R);ut[P]=(ut[P]||0)+1;var E="".concat(P,"-").concat(zr(Qe+P+ut[P]));return j?"".concat(j,"-").concat(E):E}(t.displayName,t.parentComponentId):i,d=t.displayName,l=d===void 0?function(R){return lt(R)?"styled.".concat(R):"Styled(".concat(sa(R),")")}(e):d,p=t.displayName&&t.componentId?"".concat(tr(t.displayName),"-").concat(t.componentId):t.componentId||u,m=n&&o.attrs?o.attrs.concat(c).filter(Boolean):c,v=t.shouldForwardProp;if(n&&o.shouldForwardProp){var y=o.shouldForwardProp;if(t.shouldForwardProp){var g=t.shouldForwardProp;v=function(R,j){return y(R,j)&&g(R,j)}}else v=y}var w=new ka(r,p,n?o.componentStyle:void 0);function S(R,j){return function(P,E,ne){var _e=P.attrs,nn=P.componentStyle,on=P.defaultProps,an=P.foldedComponentIds,sn=P.styledComponentId,cn=P.target,ln=F.useContext(Xr),un=cr(),tt=P.shouldForwardProp||un.shouldForwardProp,Dt=ra(E,ln,on)||le,z=function(Oe,ve,Te){for(var ye,Z=A(A({},ve),{className:void 0,theme:Te}),nt=0;nte||"40px"}; + height: ${({size:e})=>e||"40px"}; + border: 3px solid #f3f3f3; + border-top: 3px solid #007bff; + border-radius: 50%; + animation: ${Na} 1s linear infinite; + + ${({color:e})=>e&&` + border-top-color: ${e}; + `} +`,Da=({size:e="40px",color:t="#007bff",className:r="",...n})=>h.jsx($a,{size:e,color:t,className:r,...n}),Ma=N.div` + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + min-height: 200px; + padding: 2rem; + + ${({fullscreen:e})=>e&&` + position: fixed; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: rgba(255, 255, 255, 0.9); + z-index: 9999; + min-height: 100vh; + `} +`,za=N.div` + margin-bottom: 1rem; +`,Ba=N.p` + color: #666; + font-size: 0.9rem; + margin: 0; + text-align: center; +`,Qr=({text:e="Loading...",size:t="40px",fullscreen:r=!1,className:n="",spinnerColor:o="#007bff"})=>h.jsxs(Ma,{fullscreen:r,className:n,role:"status","aria-live":"polite","aria-label":e,children:[h.jsx(za,{children:h.jsx(Da,{size:t,color:o})}),h.jsx(Ba,{children:e})]}),be=({children:e})=>{const{isAuthenticated:t,isLoading:r}=No();return r?h.jsx(Qr,{text:"Authenticating...",fullscreen:!0,size:"48px"}):t?e:h.jsx(ho,{to:"/login",replace:!0})},en=f.createContext(),Ua=(e,t)=>{switch(t.type){case"ADD_TOAST":return[...e,t.toast];case"REMOVE_TOAST":return e.filter(r=>r.id!==t.id);case"CLEAR_ALL":return[];default:return e}},Fa=({children:e})=>{const[t,r]=f.useReducer(Ua,[]),n=f.useCallback(l=>{const p=Date.now()+Math.random(),m={id:p,type:"info",duration:5e3,...l};return r({type:"ADD_TOAST",toast:m}),m.duration>0&&setTimeout(()=>{r({type:"REMOVE_TOAST",id:p})},m.duration),p},[]),o=f.useCallback(l=>{r({type:"REMOVE_TOAST",id:l})},[]),a=f.useCallback(()=>{r({type:"CLEAR_ALL"})},[]),s=f.useCallback((l,p={})=>n({message:l,type:"success",...p}),[n]),c=f.useCallback((l,p={})=>n({message:l,type:"error",duration:7e3,...p}),[n]),i=f.useCallback((l,p={})=>n({message:l,type:"warning",duration:6e3,...p}),[n]),u=f.useCallback((l,p={})=>n({message:l,type:"info",...p}),[n]),d={toasts:t,addToast:n,removeToast:o,clearAll:a,showSuccess:s,showError:c,showWarning:i,showInfo:u};return h.jsx(en.Provider,{value:d,children:e})},Wa=()=>{const e=f.useContext(en);if(!e)throw new Error("useToast must be used within a ToastProvider");return e},tn=f.createContext(),Fs=()=>{const e=f.useContext(tn);if(!e)throw new Error("useDarkMode must be used within a DarkModeProvider");return e},qa=({children:e})=>{const[t,r]=f.useState(()=>{if(typeof window<"u"){const i=localStorage.getItem("darkMode");return i!==null?JSON.parse(i):window.matchMedia("(prefers-color-scheme: dark)").matches}return!1}),[n,o]=f.useState(!1),a=()=>{r(i=>!i)},s=i=>{r(i)};f.useEffect(()=>{if(!n&&typeof window<"u"){const i=localStorage.getItem("darkMode");if(i!==null){const u=JSON.parse(i);r(u)}else{const u=window.matchMedia("(prefers-color-scheme: dark)").matches;r(u)}o(!0)}},[n]),f.useEffect(()=>{n&&typeof window<"u"&&(localStorage.setItem("darkMode",JSON.stringify(t)),t?(document.documentElement.classList.add("dark"),document.body.style.colorScheme="dark"):(document.documentElement.classList.remove("dark"),document.body.style.colorScheme="light"))},[t,n]),f.useEffect(()=>{const i=window.matchMedia("(prefers-color-scheme: dark)"),u=d=>{localStorage.getItem("darkMode")===null&&r(d.matches)};return i.addEventListener("change",u),()=>i.removeEventListener("change",u)},[]);const c={isDarkMode:t,toggleDarkMode:a,setDarkMode:s,isInitialized:n};return h.jsx(tn.Provider,{value:c,children:e})},U={SUCCESS:"success",ERROR:"error",WARNING:"warning",INFO:"info"},fr={[U.SUCCESS]:"#28a745",[U.ERROR]:"#dc3545",[U.WARNING]:"#ffc107",[U.INFO]:"#17a2b8"},Va={[U.SUCCESS]:"Success",[U.ERROR]:"Error",[U.WARNING]:"Warning",[U.INFO]:"Info"},Nt=e=>fr[e]||fr[U.INFO],Ga=e=>Va[e]||"Notification",Ja=N.div` + width: 20px; + height: 20px; + flex-shrink: 0; + margin-top: 0.125rem; + + svg { + width: 100%; + height: 100%; + fill: ${({$type:e})=>Nt(e)}; + } +`,Ya=()=>h.jsx("svg",{viewBox:"0 0 20 20",children:h.jsx("path",{d:"M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z"})}),Ha=()=>h.jsx("svg",{viewBox:"0 0 20 20",children:h.jsx("path",{d:"M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z"})}),Ka=()=>h.jsx("svg",{viewBox:"0 0 20 20",children:h.jsx("path",{d:"M8.257 3.099c.765-1.36 2.722-1.36 3.486 0l5.58 9.92c.75 1.334-.213 2.98-1.742 2.98H4.42c-1.53 0-2.493-1.646-1.743-2.98l5.58-9.92zM11 13a1 1 0 11-2 0 1 1 0 012 0zm-1-8a1 1 0 00-1 1v3a1 1 0 002 0V6a1 1 0 00-1-1z"})}),Xa=()=>h.jsx("svg",{viewBox:"0 0 20 20",children:h.jsx("path",{d:"M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-7-4a1 1 0 11-2 0 1 1 0 012 0zM9 9a1 1 0 000 2v3a1 1 0 001 1h1a1 1 0 100-2v-3a1 1 0 00-1-1H9z"})}),Za=e=>{switch(e){case"success":return h.jsx(Ya,{});case"error":return h.jsx(Ha,{});case"warning":return h.jsx(Ka,{});case"info":default:return h.jsx(Xa,{})}},Qa=({type:e})=>h.jsx(Ja,{$type:e,children:Za(e)}),es=e=>{const[t,r]=f.useState(100);return f.useEffect(()=>{if(e>0){const n=setInterval(()=>{r(o=>{const a=o-100/(e/100);return a<=0?0:a})},100);return()=>clearInterval(n)}},[e]),t},ts=Lt` + from { + transform: translateX(100%); + opacity: 0; + } + to { + transform: translateX(0); + opacity: 1; + } +`,rs=Lt` + from { + transform: translateX(0); + opacity: 1; + } + to { + transform: translateX(100%); + opacity: 0; + } +`,ns=N.div` + position: fixed; + top: 1rem; + right: 1rem; + z-index: 9999; + display: flex; + flex-direction: column; + gap: 0.5rem; + max-width: 420px; + + @media (max-width: 768px) { + left: 1rem; + right: 1rem; + max-width: none; + } +`,os=N.div` + ${({$isExiting:e})=>kt` + animation: ${e?rs:ts} 0.3s ease-out; + `} +`,as=N.div` + background: white; + border-radius: 8px; + padding: 1rem; + box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15); + border-left: 4px solid ${({$type:e})=>Nt(e)}; + display: flex; + align-items: flex-start; + gap: 0.75rem; + position: relative; + + &:focus-within { + outline: 2px solid #007bff; + outline-offset: 2px; + } +`,ss=N.div` + flex: 1; + + h4 { + margin: 0 0 0.25rem 0; + font-size: 0.9rem; + font-weight: 600; + color: #333; + } + + p { + margin: 0; + font-size: 0.875rem; + line-height: 1.4; + color: #666; + } +`,is=N.button` + background: none; + border: none; + cursor: pointer; + padding: 0.25rem; + color: #999; + flex-shrink: 0; + border-radius: 4px; + + &:hover { + color: #666; + background-color: rgba(0, 0, 0, 0.05); + } + + &:focus { + outline: 2px solid #007bff; + outline-offset: 2px; + } + + svg { + width: 16px; + height: 16px; + fill: currentColor; + } +`,cs=N.div` + position: absolute; + bottom: 0; + left: 0; + height: 3px; + background-color: ${({$type:e})=>Nt(e)}; + border-radius: 0 0 8px 8px; + width: ${({$progress:e})=>e}%; + transition: width 0.1s linear; +`,ls=({toast:e,onRemove:t})=>{const[r,n]=f.useState(!1),o=es(e.duration),a=()=>{n(!0),setTimeout(()=>t(e.id),300)};return h.jsx(os,{$isExiting:r,children:h.jsxs(as,{$type:e.type,role:"alert","aria-live":e.type==="error"?"assertive":"polite","aria-atomic":"true",children:[h.jsx(Qa,{type:e.type}),h.jsxs(ss,{children:[h.jsx("h4",{children:e.title||Ga(e.type)}),h.jsx("p",{children:e.message})]}),h.jsx(is,{onClick:a,"aria-label":"Close notification",type:"button",children:h.jsx("svg",{viewBox:"0 0 20 20",children:h.jsx("path",{d:"M6.28 5.22a.75.75 0 00-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 101.06 1.06L10 11.06l3.72 3.72a.75.75 0 101.06-1.06L11.06 10l3.72-3.72a.75.75 0 00-1.06-1.06L10 8.94 6.28 5.22z"})})}),e.duration>0&&h.jsx(cs,{$type:e.type,$progress:o})]})})},us=()=>{const{toasts:e,removeToast:t}=Wa();return h.jsx(ns,{"aria-live":"polite","aria-label":"Notifications",children:e.map(r=>h.jsx(ls,{toast:r,onRemove:t},r.id))})};function rn(e){var t,r,n="";if(typeof e=="string"||typeof e=="number")n+=e;else if(typeof e=="object")if(Array.isArray(e)){var o=e.length;for(t=0;th.jsx("div",{className:me("bg-white rounded-lg p-8 max-w-2xl w-full shadow-lg text-center",e),...r,children:t}),hs=({className:e,children:t,...r})=>h.jsx("h1",{className:me("text-red-600 text-3xl mb-4 font-semibold",e),...r,children:t}),ps=({className:e,children:t,...r})=>h.jsx("p",{className:me("text-gray-600 text-lg leading-relaxed mb-8",e),...r,children:t}),ms=({className:e,error:t,errorInfo:r,...n})=>h.jsxs("details",{className:me("mb-8 text-left",e),...n,children:[h.jsx("summary",{className:"cursor-pointer text-blue-600 font-medium mb-2 hover:text-blue-800",children:"Show technical details"}),h.jsxs("pre",{className:"bg-gray-50 p-4 rounded overflow-x-auto text-sm text-gray-700 whitespace-pre-wrap break-words",children:[t==null?void 0:t.toString(),r==null?void 0:r.componentStack]})]}),gs={primary:"btn-primary",secondary:"btn-secondary",outline:"btn-outline",ghost:"btn-ghost",link:"btn-link",destructive:"btn-destructive",success:"btn-success"},vs={sm:"h-8 px-3 text-xs",md:"h-10 px-4 py-2",lg:"h-12 px-8",xl:"h-14 px-10 text-lg",icon:"h-10 w-10"},Fe=F.forwardRef(({className:e,variant:t="primary",size:r="md",disabled:n=!1,loading:o=!1,type:a="button",children:s,...c},i)=>h.jsxs("button",{className:me("btn",gs[t],vs[r],n&&"btn-disabled",o&&"btn-loading",e),ref:i,disabled:n||o,type:a,...c,children:[o&&h.jsx("span",{className:"btn-spinner","aria-hidden":"true",role:"img",children:h.jsxs("svg",{className:"animate-spin h-4 w-4",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24",children:[h.jsx("circle",{className:"opacity-25",cx:"12",cy:"12",r:"10",stroke:"currentColor",strokeWidth:"4"}),h.jsx("path",{className:"opacity-75",fill:"currentColor",d:"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"})]})}),s]}));Fe.displayName="Button";const ys=({onRetry:e,onReload:t,onReportError:r})=>h.jsxs("div",{className:"flex gap-4 justify-center flex-wrap",children:[h.jsx(Fe,{onClick:e,variant:"primary",children:"Try Again"}),h.jsx(Fe,{onClick:t,variant:"secondary",children:"Refresh Page"}),h.jsx(Fe,{onClick:r,variant:"secondary",children:"Report Error"})]}),$t=e=>{if(!e)return null;const t=new Date(e);return isNaN(t.getTime())?null:t},Ws=e=>{const t=$t(e);return t?t.toLocaleDateString():"Invalid date"},qs=e=>{const t=$t(e);return t?t.toISOString().split("T")[0]:""},xs=e=>{const t=$t(e);return t?t.toISOString():null},bs=({children:e,className:t,...r})=>h.jsx("div",{className:me("min-h-screen flex items-center justify-center p-8 bg-gray-50",t),role:"alert","aria-live":"assertive",...r,children:e});class ws extends F.Component{constructor(r){super(r);ke(this,"handleRetry",()=>{this.setState({hasError:!1,error:null,errorInfo:null})});ke(this,"handleReload",()=>{window.location.reload()});ke(this,"handleReportError",()=>{this.errorReporting.reportError(this.state.error,this.state.errorInfo)});this.state={hasError:!1,error:null,errorInfo:null},this.errorReporting={reportError:(n,o)=>{const a={error:n==null?void 0:n.toString(),stack:n==null?void 0:n.stack,componentStack:o==null?void 0:o.componentStack,userAgent:navigator.userAgent,timestamp:xs(new Date),url:window.location.href};navigator.clipboard.writeText(JSON.stringify(a,null,2)).then(()=>{alert("Error details copied to clipboard. Please send this to support.")}).catch(()=>{alert("Failed to copy error details. Please try again.")})},logError:(n,o)=>{console.error("Error caught by boundary:",n,o)}}}static getDerivedStateFromError(r){return{hasError:!0}}componentDidCatch(r,n){this.setState({error:r,errorInfo:n}),this.errorReporting.logError(r,n)}render(){return this.state.hasError?h.jsx(bs,{children:h.jsxs(fs,{children:[h.jsx(hs,{children:"Oops! Something went wrong"}),h.jsx(ps,{children:"We're sorry, but something unexpected happened. You can try refreshing the page or contact support if the problem persists."}),this.state.error&&h.jsx(ms,{error:this.state.error,errorInfo:this.state.errorInfo}),h.jsx(ys,{onRetry:this.handleRetry,onReload:this.handleReload,onReportError:this.handleReportError})]})}):this.props.children}}const Ss=({children:e})=>e,Es=f.lazy(()=>W(()=>import("./LandingPage-EXm-0yOi.js"),__vite__mapDeps([0,1,2,3,4,5,6,7,8])).then(e=>({default:e.LandingPage}))),Rs=f.lazy(()=>W(()=>import("./LoginPage-BeBDoSBE.js"),__vite__mapDeps([9,6,4,7,1,10,8])).then(e=>({default:e.LoginPage}))),Cs=f.lazy(()=>W(()=>import("./RegisterPage-DEfT8Iyu.js"),__vite__mapDeps([11,10,8])).then(e=>({default:e.RegisterPage}))),Ps=f.lazy(()=>W(()=>import("./OverviewDashboardPage-ChIgynnA.js"),__vite__mapDeps([12,5,13,8,14,15,16,17,7,1])).then(e=>({default:e.OverviewDashboardPage}))),js=f.lazy(()=>W(()=>import("./FinancialDashboardPage-UxW0hI6p.js"),__vite__mapDeps([18,5,13,8,16,19,14,15,17,7,1])).then(e=>({default:e.FinancialDashboardPage}))),_s=f.lazy(()=>W(()=>import("./EcologicalDashboardPage-B0Gl5Ex5.js"),__vite__mapDeps([20,5,13,8,19,14,15,17,7,1])).then(e=>({default:e.EcologicalDashboardPage}))),Is=f.lazy(()=>W(()=>import("./MapPage-B-W_JiyY.js"),__vite__mapDeps([21,8,17,7,1,3,4,22])).then(e=>({default:e.MapPage}))),Os=f.lazy(()=>W(()=>import("./ExportPage-DZ0UV3sF.js"),__vite__mapDeps([23,17,7,1,14,15,8])).then(e=>({default:e.ExportPage}))),Ts=f.lazy(()=>W(()=>import("./ErrorPages-DbEIbd0-.js"),__vite__mapDeps([24,1,2,3,4,5,8])).then(e=>({default:e.NotFoundPage}))),As=()=>h.jsx(ws,{children:h.jsx(Eo,{future:{v7_startTransition:!0,v7_relativeSplatPath:!0},children:h.jsx(qa,{children:h.jsx(Fa,{children:h.jsx(Lo,{children:h.jsxs(Ss,{children:[h.jsxs(f.Suspense,{fallback:h.jsx(Qr,{fullscreen:!0,text:"Loading page..."}),children:[h.jsx("a",{href:"#main-content",className:"skip-link",children:"Skip to content"}),h.jsxs(mo,{children:[h.jsx(D,{path:"/",element:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(Es,{})})}),h.jsx(D,{path:"/login",element:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(Rs,{})})}),h.jsx(D,{path:"/register",element:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(Cs,{})})}),h.jsx(D,{path:"/dashboard",element:h.jsx(be,{children:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(Ps,{})})})}),h.jsx(D,{path:"/dashboard/financial",element:h.jsx(be,{children:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(js,{})})})}),h.jsx(D,{path:"/dashboard/ecological",element:h.jsx(be,{children:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(_s,{})})})}),h.jsx(D,{path:"/map",element:h.jsx(be,{children:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(Is,{})})})}),h.jsx(D,{path:"/export",element:h.jsx(be,{children:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(Os,{})})})}),h.jsx(D,{path:"*",element:h.jsx("main",{id:"main-content",tabIndex:"-1",children:h.jsx(Ts,{})})})]})]}),h.jsx(us,{})]})})})})})}),ks=(e="root")=>{const t=document.getElementById(e);if(!t)throw new Error(`Root element with id '${e}' not found in the document`);return t},Ls=e=>xn.createRoot(e),Ns=e=>{e.render(h.jsx(F.StrictMode,{children:h.jsx(As,{})}))},$s=()=>{try{const e=ks(),t=Ls(e);Ns(t)}catch(e){throw console.error("Failed to initialize application:",e),e}};$s();export{Ao as A,Fe as B,Bs as L,F as R,Ye as a,Qr as b,ds as c,N as d,Sn as e,Wa as f,gn as g,zs as h,Us as i,h as j,qs as k,he as l,Fs as m,me as n,Ws as o,f as r,ko as u}; diff --git a/frontend/dist/assets/useDashboardStats-3Mt6zhuV.js b/frontend/dist/assets/useDashboardStats-3Mt6zhuV.js new file mode 100644 index 0000000000..d0a612a9e5 --- /dev/null +++ b/frontend/dist/assets/useDashboardStats-3Mt6zhuV.js @@ -0,0 +1 @@ +import{r as i,c as q}from"./index-tEqMizXb.js";import{h as V,af as G,j as U,k as Q,s as de,aC as F,aD as Z,aE as X,m as pe,aF as he,n as me,o as ve,y as fe,p as ee,q as ye,aG as xe,aH as ge,t as Ae,w as be,x as Ie,aI as Pe,G as Ee,A as ae,O as L,aJ as te,D as R,F as Ce,aK as H,I as J,J as Y,ao as _,K as De,M as Se,aL as N,aM as ke,a1 as B,$ as Oe,aN as M,aO as we,z as Ne,a2 as Le,a3 as _e,e as Ke}from"./EnhancedStatCard-CVgqadrW.js";import{t as Re}from"./GlobalFilters-B5C4gkva.js";var re=(e,a,t,r)=>U(e,"xAxis",a,r),ne=(e,a,t,r)=>Q(e,"xAxis",a,r),ie=(e,a,t,r)=>U(e,"yAxis",t,r),se=(e,a,t,r)=>Q(e,"yAxis",t,r),je=V([G,re,ie,ne,se],(e,a,t,r,s)=>Z(e,"xAxis")?X(a,r,!1):X(t,s,!1)),Me=(e,a,t,r,s)=>{var n,c=G(e),l=Z(c,"xAxis"),o;if(l?o=F(e,"yAxis",t,r):o=F(e,"xAxis",a,r),o!=null){var{dataKey:u,stackId:h}=s;if(h!=null){var f=(n=o[h])===null||n===void 0?void 0:n.stackedData;return f==null?void 0:f.find(d=>d.key===u)}}},Te=(e,a,t,r,s)=>s,We=V([pe,Te],(e,a)=>{if(e.some(t=>t.type==="area"&&a.dataKey===t.dataKey&&he(a.stackId)===t.stackId&&a.data===t.data))return a}),Be=V([G,re,ie,ne,se,Me,de,je,We],(e,a,t,r,s,n,c,l,o)=>{var{chartData:u,dataStartIndex:h,dataEndIndex:f}=c;if(!(o==null||e!=="horizontal"&&e!=="vertical"||a==null||t==null||r==null||s==null||r.length===0||s.length===0||l==null)){var{data:d}=o,v;if(d&&d.length>0?v=d:v=u==null?void 0:u.slice(h,f+1),v!=null){var y=void 0;return ia({layout:e,xAxis:a,yAxis:t,xAxisTicks:r,yAxisTicks:s,dataStartIndex:h,areaSettings:o,stackedData:n,displayedData:v,chartBaseValue:y,bandSize:l})}}}),ze=["layout","type","stroke","connectNulls","isRange"],Ve=["activeDot","animationBegin","animationDuration","animationEasing","connectNulls","dot","fill","fillOpacity","hide","isAnimationActive","legendType","stroke","xAxisId","yAxisId"];function le(e,a){if(e==null)return{};var t,r,s=Ge(e,a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);for(r=0;r{var{dataKey:a,name:t,stroke:r,fill:s,legendType:n,hide:c}=e;return[{inactive:c,dataKey:a,type:n,color:T(r,s),value:ee(t,a),payload:e}]};function Je(e){var{dataKey:a,data:t,stroke:r,strokeWidth:s,fill:n,name:c,hide:l,unit:o}=e;return{dataDefinedOnItem:t,positions:void 0,settings:{stroke:r,strokeWidth:s,fill:n,dataKey:a,nameKey:void 0,name:ee(c,a),hide:l,type:e.tooltipType,color:T(r,n),unit:o}}}var Ye=(e,a)=>{var t;if(i.isValidElement(e))t=i.cloneElement(e,a);else if(typeof e=="function")t=e(a);else{var r=q("recharts-area-dot",typeof e!="boolean"?e.className:"");t=i.createElement(we,C({},a,{className:r}))}return t};function $e(e,a){return e==null?!1:a?!0:e.length===1}function qe(e){var{clipPathId:a,points:t,props:r}=e,{needClip:s,dot:n,dataKey:c}=r;if(!$e(t,n))return null;var l=te(n),o=L(r,!1),u=L(n,!0),h=t.map((d,v)=>{var y=w(w(w({key:"dot-".concat(v),r:3},o),u),{},{index:v,cx:d.x,cy:d.y,dataKey:c,value:d.value,payload:d.payload,points:t});return Ye(n,y)}),f={clipPath:s?"url(#clipPath-".concat(l?"":"dots-").concat(a,")"):void 0};return i.createElement(R,C({className:"recharts-area-dots"},f),h)}function z(e){var{points:a,baseLine:t,needClip:r,clipPathId:s,props:n,showLabels:c}=e,{layout:l,type:o,stroke:u,connectNulls:h,isRange:f}=n,d=le(n,ze);return i.createElement(i.Fragment,null,(a==null?void 0:a.length)>1&&i.createElement(R,{clipPath:r?"url(#clipPath-".concat(s,")"):void 0},i.createElement(B,C({},L(d,!0),{points:a,connectNulls:h,type:o,baseLine:t,layout:l,stroke:"none",className:"recharts-area-area"})),u!=="none"&&i.createElement(B,C({},L(n,!1),{className:"recharts-area-curve",layout:l,type:o,connectNulls:h,fill:"none",points:a})),u!=="none"&&f&&i.createElement(B,C({},L(n,!1),{className:"recharts-area-curve",layout:l,type:o,connectNulls:h,fill:"none",points:t}))),i.createElement(qe,{points:a,props:n,clipPathId:s}),c&&Oe.renderCallByParent(n,a))}function Ue(e){var{alpha:a,baseLine:t,points:r,strokeWidth:s}=e,n=r[0].y,c=r[r.length-1].y;if(!M(n)||!M(c))return null;var l=a*Math.abs(n-c),o=Math.max(...r.map(u=>u.x||0));return _(t)?o=Math.max(t,o):t&&Array.isArray(t)&&t.length&&(o=Math.max(...t.map(u=>u.x||0),o)),_(o)?i.createElement("rect",{x:0,y:nu.y||0));return _(t)?o=Math.max(t,o):t&&Array.isArray(t)&&t.length&&(o=Math.max(...t.map(u=>u.y||0),o)),_(o)?i.createElement("rect",{x:n{typeof v=="function"&&v(),g(!1)},[v]),P=i.useCallback(()=>{typeof d=="function"&&d(),g(!0)},[d]),A=s.current,p=n.current;return i.createElement(Se,{begin:u,duration:h,isActive:o,easing:f,from:{t:0},to:{t:1},onAnimationEnd:E,onAnimationStart:P,key:y},b=>{var{t:m}=b;if(A){var D=A.length/c.length,j=m===1?c:c.map((I,K)=>{var k=Math.floor(K*D);if(A[k]){var O=A[k];return w(w({},I),{},{x:N(O.x,I.x,m),y:N(O.y,I.y,m)})}return I}),S;return _(l)?S=N(p,l,m):ae(l)||ke(l)?S=N(p,0,m):S=l.map((I,K)=>{var k=Math.floor(K*D);if(Array.isArray(p)&&p[k]){var O=p[k];return w(w({},I),{},{x:N(O.x,I.x,m),y:N(O.y,I.y,m)})}return I}),m>0&&(s.current=j,n.current=S),i.createElement(z,{points:j,baseLine:S,needClip:a,clipPathId:t,props:r,showLabels:!x})}return m>0&&(s.current=c,n.current=l),i.createElement(R,null,i.createElement("defs",null,i.createElement("clipPath",{id:"animationClipPath-".concat(t)},i.createElement(Ze,{alpha:m,points:c,baseLine:l,layout:r.layout,strokeWidth:r.strokeWidth}))),i.createElement(R,{clipPath:"url(#animationClipPath-".concat(t,")")},i.createElement(z,{points:c,baseLine:l,needClip:a,clipPathId:t,props:r,showLabels:!0})))})}function aa(e){var{needClip:a,clipPathId:t,props:r}=e,{points:s,baseLine:n,isAnimationActive:c}=r,l=i.useRef(null),o=i.useRef(),u=l.current,h=o.current;return c&&s&&s.length&&(u!==s||h!==n)?i.createElement(ea,{needClip:a,clipPathId:t,props:r,previousPointsRef:l,previousBaselineRef:o}):i.createElement(z,{points:s,baseLine:n,needClip:a,clipPathId:t,props:r,showLabels:!0})}class ta extends i.PureComponent{constructor(){super(...arguments),W(this,"id",Ne("recharts-area-"))}render(){var a,{hide:t,dot:r,points:s,className:n,top:c,left:l,needClip:o,xAxisId:u,yAxisId:h,width:f,height:d,id:v,baseLine:y}=this.props;if(t)return null;var x=q("recharts-area",n),g=ae(v)?this.id:v,{r:E=3,strokeWidth:P=2}=(a=L(r,!1))!==null&&a!==void 0?a:{r:3,strokeWidth:2},A=te(r),p=E*2+P;return i.createElement(i.Fragment,null,i.createElement(R,{className:x},o&&i.createElement("defs",null,i.createElement(Ce,{clipPathId:g,xAxisId:u,yAxisId:h}),!A&&i.createElement("clipPath",{id:"clipPath-dots-".concat(g)},i.createElement("rect",{x:l-p/2,y:c-p/2,width:f+p,height:d+p}))),i.createElement(aa,{needClip:o,clipPathId:g,props:this.props})),i.createElement(H,{points:s,mainColor:T(this.props.stroke,this.props.fill),itemDataKey:this.props.dataKey,activeDot:this.props.activeDot}),this.props.isRange&&Array.isArray(y)&&i.createElement(H,{points:y,mainColor:T(this.props.stroke,this.props.fill),itemDataKey:this.props.dataKey,activeDot:this.props.activeDot}))}}var oe={activeDot:!0,animationBegin:0,animationDuration:1500,animationEasing:"ease",connectNulls:!1,dot:!1,fill:"#3182bd",fillOpacity:.6,hide:!1,isAnimationActive:!Ee.isSsr,legendType:"line",stroke:"#3182bd",xAxisId:0,yAxisId:0};function ra(e){var a,t=ye(e,oe),{activeDot:r,animationBegin:s,animationDuration:n,animationEasing:c,connectNulls:l,dot:o,fill:u,fillOpacity:h,hide:f,isAnimationActive:d,legendType:v,stroke:y,xAxisId:x,yAxisId:g}=t,E=le(t,Ve),P=xe(),A=ge(),{needClip:p}=Ae(x,g),b=be(),m=i.useMemo(()=>({baseValue:e.baseValue,stackId:e.stackId,connectNulls:l,data:e.data,dataKey:e.dataKey}),[e.baseValue,e.stackId,l,e.data,e.dataKey]),{points:D,isRange:j,baseLine:S}=(a=Ie(ue=>Be(ue,x,g,b,m)))!==null&&a!==void 0?a:{},{height:I,width:K,x:k,y:O}=Pe();return P!=="horizontal"&&P!=="vertical"||A!=="AreaChart"&&A!=="ComposedChart"?null:i.createElement(ta,C({},E,{activeDot:r,animationBegin:s,animationDuration:n,animationEasing:c,baseLine:S,connectNulls:l,dot:o,fill:u,fillOpacity:h,height:I,hide:f,layout:P,isAnimationActive:d,isRange:j,legendType:v,needClip:p,points:D,stroke:y,width:K,left:k,top:O,xAxisId:x,yAxisId:g}))}var na=(e,a,t,r,s)=>{var n=t??a;if(_(n))return n;var c=e==="horizontal"?s:r,l=c.scale.domain();if(c.type==="number"){var o=Math.max(l[0],l[1]),u=Math.min(l[0],l[1]);return n==="dataMin"?u:n==="dataMax"||o<0?o:Math.max(Math.min(l[0],l[1]),0)}return n==="dataMin"?l[0]:n==="dataMax"?l[1]:l[0]};function ia(e){var{areaSettings:{connectNulls:a,baseValue:t,dataKey:r},stackedData:s,layout:n,chartBaseValue:c,xAxis:l,yAxis:o,displayedData:u,dataStartIndex:h,xAxisTicks:f,yAxisTicks:d,bandSize:v}=e,y=s&&s.length,x=na(n,c,t,l,o),g=n==="horizontal",E=!1,P=u.map((p,b)=>{var m;y?m=s[h+b]:(m=J(p,r),Array.isArray(m)?E=!0:m=[x,m]);var D=m[1]==null||y&&!a&&J(p,r)==null;return g?{x:Y({axis:l,ticks:f,bandSize:v,entry:p,index:b}),y:D?null:o.scale(m[1]),value:m,payload:p}:{x:D?null:l.scale(m[1]),y:Y({axis:o,ticks:d,bandSize:v,entry:p,index:b}),value:m,payload:p}}),A;return y||E?A=P.map(p=>{var b=Array.isArray(p.value)?p.value[0]:null;return g?{x:p.x,y:b!=null&&p.y!=null?o.scale(b):null}:{x:b!=null?l.scale(b):null,y:p.y}}):A=g?o.scale(x):l.scale(x),{points:P,baseLine:A,isRange:E}}class ce extends i.PureComponent{render(){return i.createElement(me,{type:"area",data:this.props.data,dataKey:this.props.dataKey,xAxisId:this.props.xAxisId,yAxisId:this.props.yAxisId,zAxisId:0,stackId:this.props.stackId,hide:this.props.hide,barSize:void 0},i.createElement(ve,{legendPayload:He(this.props)}),i.createElement(fe,{fn:Je,args:this.props}),i.createElement(ra,this.props))}}W(ce,"displayName","Area");W(ce,"defaultProps",oe);var sa=["axis"],ma=i.forwardRef((e,a)=>i.createElement(Le,{chartName:"AreaChart",defaultTooltipEventType:"axis",validateTooltipEventTypes:sa,tooltipPayloadSearcher:_e,categoricalChartProps:e,ref:a}));const la=10*60*1e3,oa=e=>{const a=JSON.stringify(e);return`dashboard_stats_${btoa(a)}`},ca=e=>{try{const a=sessionStorage.getItem(e);if(!a)return null;const{data:t,timestamp:r}=JSON.parse(a);return Date.now()-r>la?(sessionStorage.removeItem(e),null):t}catch(a){return console.warn("Dashboard cache read error:",a),null}},ua=(e,a)=>{try{const t={data:a,timestamp:Date.now()};sessionStorage.setItem(e,JSON.stringify(t))}catch(t){console.warn("Dashboard cache write error:",t)}},va=(e={})=>{var h,f;const[a,t]=i.useState(null),[r,s]=i.useState(!0),[n,c]=i.useState(null),l=i.useMemo(()=>Re(e),[(h=e==null?void 0:e.dateRange)==null?void 0:h.startDate,(f=e==null?void 0:e.dateRange)==null?void 0:f.endDate,e==null?void 0:e.selectedForests,e==null?void 0:e.species]),o=i.useCallback(async()=>{try{s(!0),c(null);const d=oa(l),v=ca(d);if(v){t(v),s(!1);return}const x=(await Ke.getEnhancedStats(l)).data;ua(d,x),t(x)}catch(d){console.error("Error fetching dashboard stats:",d),c(d.message||"Failed to fetch dashboard statistics")}finally{s(!1)}},[l]);i.useEffect(()=>{o()},[o]);const u=i.useCallback(()=>{o()},[o]);return{stats:a,loading:r,error:n,refresh:u}};export{ma as A,ce as a,va as u}; diff --git a/frontend/dist/assets/useKeyboardNavigation-DT9lybaQ.js b/frontend/dist/assets/useKeyboardNavigation-DT9lybaQ.js new file mode 100644 index 0000000000..0577a68b65 --- /dev/null +++ b/frontend/dist/assets/useKeyboardNavigation-DT9lybaQ.js @@ -0,0 +1 @@ +import{r as s}from"./index-tEqMizXb.js";const T=(A={})=>{const{onEscape:f,onEnter:d,onSpace:b,onArrowUp:m,onArrowDown:p,onArrowLeft:k,onArrowRight:E,onTab:w,trapFocus:h=!1,autoFocus:g=!1,disabled:D=!1}=A,l=s.useRef(null),a=s.useRef([]),x=s.useCallback(()=>{if(!l.current)return[];const e=["button:not([disabled])","input:not([disabled])","select:not([disabled])","textarea:not([disabled])","a[href]:not([disabled])",'[tabindex]:not([tabindex="-1"]):not([disabled])',"summary:not([disabled])"];return Array.from(l.current.querySelectorAll(e.join(","))).filter(t=>{const r=window.getComputedStyle(t);return r.display!=="none"&&r.visibility!=="hidden"})},[]),n=s.useCallback(()=>{a.current=x()},[x]),u=s.useCallback(e=>{if(D)return;const{key:t,shiftKey:r,ctrlKey:c,altKey:o,metaKey:i}=e;if(!((c||o||i)&&t!=="Tab"))switch(t){case"Escape":f&&(e.preventDefault(),f(e));break;case"Enter":d&&(e.preventDefault(),d(e));break;case" ":b&&(e.preventDefault(),b(e));break;case"ArrowUp":m&&(e.preventDefault(),m(e));break;case"ArrowDown":p&&(e.preventDefault(),p(e));break;case"ArrowLeft":k&&(e.preventDefault(),k(e));break;case"ArrowRight":E&&(e.preventDefault(),E(e));break;case"Tab":h&&a.current.length>0&&(e.preventDefault(),C(r)),w&&w(e);break}},[D,f,d,b,m,p,k,E,w,h]),C=s.useCallback(e=>{var i;n();const t=a.current;if(t.length===0)return;const r=document.activeElement,c=t.indexOf(r);let o;e?o=c<=0?t.length-1:c-1:o=c>=t.length-1?0:c+1,(i=t[o])==null||i.focus()},[n]),y=s.useCallback(()=>{var t;n(),(t=a.current[0])==null||t.focus()},[n]),v=s.useCallback(()=>{var t;n();const e=a.current;(t=e[e.length-1])==null||t.focus()},[n]),I=s.useCallback(()=>{var o;n();const e=a.current,t=document.activeElement,r=e.indexOf(t),c=r>=e.length-1?0:r+1;(o=e[c])==null||o.focus()},[n]),K=s.useCallback(()=>{var o;n();const e=a.current,t=document.activeElement,r=e.indexOf(t),c=r<=0?e.length-1:r-1;(o=e[c])==null||o.focus()},[n]);return s.useEffect(()=>{const e=l.current;if(e){if(e.addEventListener("keydown",u),g){const t=setTimeout(()=>{y()},100);return()=>{clearTimeout(t),e.removeEventListener("keydown",u)}}return()=>{e.removeEventListener("keydown",u)}}},[u,g,y]),s.useEffect(()=>{n()},[n]),{containerRef:l,focusFirst:y,focusLast:v,focusNext:I,focusPrevious:K,updateFocusableElements:n,getFocusableElements:x}};export{T as u}; diff --git a/frontend/dist/assets/useSidebarState-CJJaSDFr.js b/frontend/dist/assets/useSidebarState-CJJaSDFr.js new file mode 100644 index 0000000000..7491efb3f8 --- /dev/null +++ b/frontend/dist/assets/useSidebarState-CJJaSDFr.js @@ -0,0 +1,10 @@ +import{u as d,j as e,l as g,L as o,r as n}from"./index-tEqMizXb.js";import{D as x}from"./DarkModeToggle-ZfYhBz0e.js";const b=({onToggleSidebar:t})=>{const{user:r,logout:s,isAdmin:a}=d(),i=async()=>{await s()};return e.jsx("header",{className:"bg-white dark:bg-gray-800 border-b border-gray-200 dark:border-gray-700 p-4 sticky top-0 z-50",children:e.jsxs("div",{className:"flex items-center justify-between",children:[e.jsxs("div",{className:"flex items-center gap-4",children:[e.jsx("button",{onClick:t,"aria-label":"Toggle menu",className:"lg:hidden flex items-center justify-center w-10 h-10 bg-none border-none cursor-pointer rounded-lg transition-colors hover:bg-gray-100 dark:hover:bg-gray-700 focus:outline-2 focus:outline-blue-500 focus:outline-offset-2",children:e.jsx("svg",{className:"w-6 h-6 text-gray-700 dark:text-gray-300",viewBox:"0 0 24 24",fill:"none",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4 6h16M4 12h16M4 18h16"})})}),e.jsx("h1",{className:"text-2xl font-bold text-green-600 dark:text-green-400 m-0",children:"Nanwa Dashboard"})]}),e.jsxs("div",{className:"flex items-center gap-4 text-sm",children:[e.jsxs("span",{className:"text-gray-700 dark:text-gray-300",children:["Welcome, ",e.jsx("strong",{children:(r==null?void 0:r.firstName)||(r==null?void 0:r.name)}),a()&&e.jsx("span",{className:"ml-2 px-2 py-1 bg-red-100 dark:bg-red-900/30 text-red-600 dark:text-red-400 text-xs rounded-full font-bold",children:"Admin"})]}),e.jsx(x,{size:"sm"}),e.jsxs("button",{onClick:i,className:"flex items-center gap-2 px-4 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 hover:text-red-600 dark:hover:text-red-400 hover:bg-red-50 dark:hover:bg-red-900/30 rounded-lg transition-colors",title:"Logout",children:[e.jsx("svg",{className:"w-4 h-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M17 16l4-4m0 0l-4-4m4 4H7m6 4v1a3 3 0 01-3 3H6a3 3 0 01-3-3V7a3 3 0 013-3h4a3 3 0 013 3v1"})}),"Logout"]})]})]})})},k=({isOpen:t,onClose:r})=>{const{isAdmin:s}=d(),a=g();return e.jsxs(e.Fragment,{children:[e.jsx("div",{className:`lg:hidden fixed inset-0 bg-black/50 z-40 transition-opacity duration-300 ${t?"opacity-100 visible":"opacity-0 invisible pointer-events-none"}`,onClick:r,"aria-hidden":"true"}),e.jsxs("aside",{className:` + bg-white dark:bg-gray-800 border-r border-gray-200 dark:border-gray-700 + w-64 shrink-0 + lg:fixed lg:top-16 lg:translate-x-0 + fixed top-0 h-screen lg:h-[calc(100vh-4rem)] + overflow-y-auto + z-50 lg:z-0 + transform transition-transform duration-300 ease-in-out + ${t?"translate-x-0":"-translate-x-full"} + `,"aria-label":"Sidebar navigation",children:[e.jsxs("div",{className:"lg:hidden sticky top-0 bg-white dark:bg-gray-800 border-b border-gray-200 dark:border-gray-700 p-4 flex justify-between items-center",children:[e.jsx("h2",{className:"text-lg font-semibold text-gray-900 dark:text-white",children:"Navigation"}),e.jsx("button",{onClick:r,className:"p-2 rounded-lg hover:bg-gray-100 dark:hover:bg-gray-700 transition-colors","aria-label":"Close navigation menu",children:e.jsx("svg",{className:"w-5 h-5 text-gray-600 dark:text-gray-400",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M6 18L18 6M6 6l12 12"})})})]}),e.jsx("div",{className:"p-6",children:e.jsxs("nav",{className:"space-y-1","aria-label":"Dashboard navigation",children:[e.jsxs("div",{className:"pb-2",children:[e.jsx("p",{className:"text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wide px-3 py-2",children:"Dashboards"}),e.jsxs("div",{className:"space-y-1",children:[e.jsxs(o,{to:"/dashboard",onClick:()=>r&&r(),className:`flex items-center px-3 py-2 text-sm font-medium rounded-lg transition-colors ${a.pathname==="/dashboard"?"text-green-600 dark:text-green-400 bg-green-50 dark:bg-green-900/30":"text-gray-700 dark:text-gray-300 hover:text-green-600 dark:hover:text-green-400 hover:bg-green-50 dark:hover:bg-green-900/30"}`,"aria-current":a.pathname==="/dashboard"?"page":void 0,children:[e.jsx("svg",{className:"mr-3 h-5 w-5",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"})}),"Overview"]}),e.jsxs(o,{to:"/dashboard/financial",onClick:()=>r&&r(),className:`flex items-center px-3 py-2 text-sm font-medium rounded-lg transition-colors ${a.pathname==="/dashboard/financial"?"text-blue-600 dark:text-blue-400 bg-blue-50 dark:bg-blue-900/30":"text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 hover:bg-blue-50 dark:hover:bg-blue-900/30"}`,"aria-current":a.pathname==="/dashboard/financial"?"page":void 0,children:[e.jsx("svg",{className:"mr-3 h-5 w-5",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M13 7h8m0 0v8m0-8l-8 8-4-4-6 6"})}),"Financial"]}),e.jsxs(o,{to:"/dashboard/ecological",onClick:()=>r&&r(),className:`flex items-center px-3 py-2 text-sm font-medium rounded-lg transition-colors ${a.pathname==="/dashboard/ecological"?"text-green-600 dark:text-green-400 bg-green-50 dark:bg-green-900/30":"text-gray-700 dark:text-gray-300 hover:text-green-600 dark:hover:text-green-400 hover:bg-green-50 dark:hover:bg-green-900/30"}`,"aria-current":a.pathname==="/dashboard/ecological"?"page":void 0,children:[e.jsx("svg",{className:"mr-3 h-5 w-5",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M4.318 6.318a4.5 4.5 0 000 6.364L12 20.364l7.682-7.682a4.5 4.5 0 00-6.364-6.364L12 7.636l-1.318-1.318a4.5 4.5 0 00-6.364 0z"})}),"Ecological"]})]})]}),e.jsxs("div",{className:"border-t border-gray-200 dark:border-gray-700 pt-2",children:[e.jsx("p",{className:"text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wide px-3 py-2",children:"Tools"}),e.jsxs("div",{className:"space-y-1",children:[e.jsxs(o,{to:"/map",onClick:()=>r&&r(),className:`flex items-center px-3 py-2 text-sm font-medium rounded-lg transition-colors ${a.pathname==="/map"?"text-gray-900 dark:text-white bg-gray-100 dark:bg-gray-700":"text-gray-700 dark:text-gray-300 hover:text-gray-900 dark:hover:text-white hover:bg-gray-100 dark:hover:bg-gray-700"}`,"aria-current":a.pathname==="/map"?"page":void 0,children:[e.jsxs("svg",{className:"mr-3 h-5 w-5",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:[e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z"}),e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M15 11a3 3 0 11-6 0 3 3 0 016 0z"})]}),"Map View"]}),e.jsxs(o,{to:"/export",onClick:()=>r&&r(),className:`flex items-center px-3 py-2 text-sm font-medium rounded-lg transition-colors ${a.pathname==="/export"?"text-gray-900 dark:text-white bg-gray-100 dark:bg-gray-700":"text-gray-700 dark:text-gray-300 hover:text-gray-900 dark:hover:text-white hover:bg-gray-100 dark:hover:bg-gray-700"}`,"aria-current":a.pathname==="/export"?"page":void 0,children:[e.jsx("svg",{className:"mr-3 h-5 w-5",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor",children:e.jsx("path",{strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:2,d:"M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"})}),"Data Export"]})]})]})]})})]})]})},p=(t=!1)=>{const[r,s]=n.useState(t),a=n.useCallback(()=>{s(c=>!c)},[]),i=n.useCallback(()=>{s(!0)},[]),l=n.useCallback(()=>{s(!1)},[]);return{sidebarOpen:r,toggleSidebar:a,openSidebar:i,closeSidebar:l}};export{b as D,k as a,p as u}; diff --git a/frontend/dist/assets/vendor-BtP0CW_r.js b/frontend/dist/assets/vendor-BtP0CW_r.js new file mode 100644 index 0000000000..257971db96 --- /dev/null +++ b/frontend/dist/assets/vendor-BtP0CW_r.js @@ -0,0 +1,32 @@ +function Pc(R){return R&&R.__esModule&&Object.prototype.hasOwnProperty.call(R,"default")?R.default:R}var wi={exports:{}},D={};/** + * @license React + * react.production.min.js + * + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */var ga;function kc(){if(ga)return D;ga=1;var R=Symbol.for("react.element"),b=Symbol.for("react.portal"),m=Symbol.for("react.fragment"),dt=Symbol.for("react.strict_mode"),Fe=Symbol.for("react.profiler"),Ke=Symbol.for("react.provider"),Ye=Symbol.for("react.context"),Te=Symbol.for("react.forward_ref"),de=Symbol.for("react.suspense"),je=Symbol.for("react.memo"),ln=Symbol.for("react.lazy"),ie=Symbol.iterator;function ne(f){return f===null||typeof f!="object"?null:(f=ie&&f[ie]||f["@@iterator"],typeof f=="function"?f:null)}var pt={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},Ue=Object.assign,K={};function B(f,h,M){this.props=f,this.context=h,this.refs=K,this.updater=M||pt}B.prototype.isReactComponent={},B.prototype.setState=function(f,h){if(typeof f!="object"&&typeof f!="function"&&f!=null)throw Error("setState(...): takes an object of state variables to update or a function which returns an object of state variables.");this.updater.enqueueSetState(this,f,h,"setState")},B.prototype.forceUpdate=function(f){this.updater.enqueueForceUpdate(this,f,"forceUpdate")};function mt(){}mt.prototype=B.prototype;function lt(f,h,M){this.props=f,this.context=h,this.refs=K,this.updater=M||pt}var Xe=lt.prototype=new mt;Xe.constructor=lt,Ue(Xe,B.prototype),Xe.isPureReactComponent=!0;var pe=Array.isArray,Ge=Object.prototype.hasOwnProperty,ge={current:null},ke={key:!0,ref:!0,__self:!0,__source:!0};function Ve(f,h,M){var O,F={},j=null,H=null;if(h!=null)for(O in h.ref!==void 0&&(H=h.ref),h.key!==void 0&&(j=""+h.key),h)Ge.call(h,O)&&!ke.hasOwnProperty(O)&&(F[O]=h[O]);var V=arguments.length-2;if(V===1)F.children=M;else if(1>>1,h=S[f];if(0>>1;fFe(F,C))jFe(H,F)?(S[f]=H,S[j]=C,f=j):(S[f]=F,S[O]=C,f=O);else if(jFe(H,C))S[f]=H,S[j]=C,f=j;else break e}}return T}function Fe(S,T){var C=S.sortIndex-T.sortIndex;return C!==0?C:S.id-T.id}if(typeof performance=="object"&&typeof performance.now=="function"){var Ke=performance;R.unstable_now=function(){return Ke.now()}}else{var Ye=Date,Te=Ye.now();R.unstable_now=function(){return Ye.now()-Te}}var de=[],je=[],ln=1,ie=null,ne=3,pt=!1,Ue=!1,K=!1,B=typeof setTimeout=="function"?setTimeout:null,mt=typeof clearTimeout=="function"?clearTimeout:null,lt=typeof setImmediate<"u"?setImmediate:null;typeof navigator<"u"&&navigator.scheduling!==void 0&&navigator.scheduling.isInputPending!==void 0&&navigator.scheduling.isInputPending.bind(navigator.scheduling);function Xe(S){for(var T=m(je);T!==null;){if(T.callback===null)dt(je);else if(T.startTime<=S)dt(je),T.sortIndex=T.expirationTime,b(de,T);else break;T=m(je)}}function pe(S){if(K=!1,Xe(S),!Ue)if(m(de)!==null)Ue=!0,Ee(Ge);else{var T=m(je);T!==null&&J(pe,T.startTime-S)}}function Ge(S,T){Ue=!1,K&&(K=!1,mt(Ve),Ve=-1),pt=!0;var C=ne;try{for(Xe(T),ie=m(de);ie!==null&&(!(ie.expirationTime>T)||S&&!$t());){var f=ie.callback;if(typeof f=="function"){ie.callback=null,ne=ie.priorityLevel;var h=f(ie.expirationTime<=T);T=R.unstable_now(),typeof h=="function"?ie.callback=h:ie===m(de)&&dt(de),Xe(T)}else dt(de);ie=m(de)}if(ie!==null)var M=!0;else{var O=m(je);O!==null&&J(pe,O.startTime-T),M=!1}return M}finally{ie=null,ne=C,pt=!1}}var ge=!1,ke=null,Ve=-1,_t=5,vt=-1;function $t(){return!(R.unstable_now()-vt<_t)}function ut(){if(ke!==null){var S=R.unstable_now();vt=S;var T=!0;try{T=ke(!0,S)}finally{T?Le():(ge=!1,ke=null)}}else ge=!1}var Le;if(typeof lt=="function")Le=function(){lt(ut)};else if(typeof MessageChannel<"u"){var Ze=new MessageChannel,it=Ze.port2;Ze.port1.onmessage=ut,Le=function(){it.postMessage(null)}}else Le=function(){B(ut,0)};function Ee(S){ke=S,ge||(ge=!0,Le())}function J(S,T){Ve=B(function(){S(R.unstable_now())},T)}R.unstable_IdlePriority=5,R.unstable_ImmediatePriority=1,R.unstable_LowPriority=4,R.unstable_NormalPriority=3,R.unstable_Profiling=null,R.unstable_UserBlockingPriority=2,R.unstable_cancelCallback=function(S){S.callback=null},R.unstable_continueExecution=function(){Ue||pt||(Ue=!0,Ee(Ge))},R.unstable_forceFrameRate=function(S){0>S||125f?(S.sortIndex=C,b(je,S),m(de)===null&&S===m(je)&&(K?(mt(Ve),Ve=-1):K=!0,J(pe,C-f))):(S.sortIndex=h,b(de,S),Ue||pt||(Ue=!0,Ee(Ge))),S},R.unstable_shouldYield=$t,R.unstable_wrapCallback=function(S){var T=ne;return function(){var C=ne;ne=T;try{return S.apply(this,arguments)}finally{ne=C}}}}(Ei)),Ei}var ka;function _c(){return ka||(ka=1,ki.exports=Cc()),ki.exports}/** + * @license React + * react-dom.production.min.js + * + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */var Ea;function xc(){if(Ea)return ze;Ea=1;var R=Ec(),b=_c();function m(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n"u"||typeof window.document>"u"||typeof window.document.createElement>"u"),de=Object.prototype.hasOwnProperty,je=/^[:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD][:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD\-.0-9\u00B7\u0300-\u036F\u203F-\u2040]*$/,ln={},ie={};function ne(e){return de.call(ie,e)?!0:de.call(ln,e)?!1:je.test(e)?ie[e]=!0:(ln[e]=!0,!1)}function pt(e,t,n,r){if(n!==null&&n.type===0)return!1;switch(typeof t){case"function":case"symbol":return!0;case"boolean":return r?!1:n!==null?!n.acceptsBooleans:(e=e.toLowerCase().slice(0,5),e!=="data-"&&e!=="aria-");default:return!1}}function Ue(e,t,n,r){if(t===null||typeof t>"u"||pt(e,t,n,r))return!0;if(r)return!1;if(n!==null)switch(n.type){case 3:return!t;case 4:return t===!1;case 5:return isNaN(t);case 6:return isNaN(t)||1>t}return!1}function K(e,t,n,r,l,u,i){this.acceptsBooleans=t===2||t===3||t===4,this.attributeName=r,this.attributeNamespace=l,this.mustUseProperty=n,this.propertyName=e,this.type=t,this.sanitizeURL=u,this.removeEmptyString=i}var B={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){B[e]=new K(e,0,!1,e,null,!1,!1)}),[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var t=e[0];B[t]=new K(t,1,!1,e[1],null,!1,!1)}),["contentEditable","draggable","spellCheck","value"].forEach(function(e){B[e]=new K(e,2,!1,e.toLowerCase(),null,!1,!1)}),["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){B[e]=new K(e,2,!1,e,null,!1,!1)}),"allowFullScreen async autoFocus autoPlay controls default defer disabled disablePictureInPicture disableRemotePlayback formNoValidate hidden loop noModule noValidate open playsInline readOnly required reversed scoped seamless itemScope".split(" ").forEach(function(e){B[e]=new K(e,3,!1,e.toLowerCase(),null,!1,!1)}),["checked","multiple","muted","selected"].forEach(function(e){B[e]=new K(e,3,!0,e,null,!1,!1)}),["capture","download"].forEach(function(e){B[e]=new K(e,4,!1,e,null,!1,!1)}),["cols","rows","size","span"].forEach(function(e){B[e]=new K(e,6,!1,e,null,!1,!1)}),["rowSpan","start"].forEach(function(e){B[e]=new K(e,5,!1,e.toLowerCase(),null,!1,!1)});var mt=/[\-:]([a-z])/g;function lt(e){return e[1].toUpperCase()}"accent-height alignment-baseline arabic-form baseline-shift cap-height clip-path clip-rule color-interpolation color-interpolation-filters color-profile color-rendering dominant-baseline enable-background fill-opacity fill-rule flood-color flood-opacity font-family font-size font-size-adjust font-stretch font-style font-variant font-weight glyph-name glyph-orientation-horizontal glyph-orientation-vertical horiz-adv-x horiz-origin-x image-rendering letter-spacing lighting-color marker-end marker-mid marker-start overline-position overline-thickness paint-order panose-1 pointer-events rendering-intent shape-rendering stop-color stop-opacity strikethrough-position strikethrough-thickness stroke-dasharray stroke-dashoffset stroke-linecap stroke-linejoin stroke-miterlimit stroke-opacity stroke-width text-anchor text-decoration text-rendering underline-position underline-thickness unicode-bidi unicode-range units-per-em v-alphabetic v-hanging v-ideographic v-mathematical vector-effect vert-adv-y vert-origin-x vert-origin-y word-spacing writing-mode xmlns:xlink x-height".split(" ").forEach(function(e){var t=e.replace(mt,lt);B[t]=new K(t,1,!1,e,null,!1,!1)}),"xlink:actuate xlink:arcrole xlink:role xlink:show xlink:title xlink:type".split(" ").forEach(function(e){var t=e.replace(mt,lt);B[t]=new K(t,1,!1,e,"http://www.w3.org/1999/xlink",!1,!1)}),["xml:base","xml:lang","xml:space"].forEach(function(e){var t=e.replace(mt,lt);B[t]=new K(t,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)}),["tabIndex","crossOrigin"].forEach(function(e){B[e]=new K(e,1,!1,e.toLowerCase(),null,!1,!1)}),B.xlinkHref=new K("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1),["src","href","action","formAction"].forEach(function(e){B[e]=new K(e,1,!1,e.toLowerCase(),null,!0,!0)});function Xe(e,t,n,r){var l=B.hasOwnProperty(t)?B[t]:null;(l!==null?l.type!==0:r||!(2o||l[i]!==u[o]){var s=` +`+l[i].replace(" at new "," at ");return e.displayName&&s.includes("")&&(s=s.replace("",e.displayName)),s}while(1<=i&&0<=o);break}}}finally{M=!1,Error.prepareStackTrace=n}return(e=e?e.displayName||e.name:"")?h(e):""}function F(e){switch(e.tag){case 5:return h(e.type);case 16:return h("Lazy");case 13:return h("Suspense");case 19:return h("SuspenseList");case 0:case 2:case 15:return e=O(e.type,!1),e;case 11:return e=O(e.type.render,!1),e;case 1:return e=O(e.type,!0),e;default:return""}}function j(e){if(e==null)return null;if(typeof e=="function")return e.displayName||e.name||null;if(typeof e=="string")return e;switch(e){case ke:return"Fragment";case ge:return"Portal";case _t:return"Profiler";case Ve:return"StrictMode";case Le:return"Suspense";case Ze:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case $t:return(e.displayName||"Context")+".Consumer";case vt:return(e._context.displayName||"Context")+".Provider";case ut:var t=e.render;return e=e.displayName,e||(e=t.displayName||t.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case it:return t=e.displayName||null,t!==null?t:j(e.type)||"Memo";case Ee:t=e._payload,e=e._init;try{return j(e(t))}catch{}}return null}function H(e){var t=e.type;switch(e.tag){case 24:return"Cache";case 9:return(t.displayName||"Context")+".Consumer";case 10:return(t._context.displayName||"Context")+".Provider";case 18:return"DehydratedFragment";case 11:return e=t.render,e=e.displayName||e.name||"",t.displayName||(e!==""?"ForwardRef("+e+")":"ForwardRef");case 7:return"Fragment";case 5:return t;case 4:return"Portal";case 3:return"Root";case 6:return"Text";case 16:return j(t);case 8:return t===Ve?"StrictMode":"Mode";case 22:return"Offscreen";case 12:return"Profiler";case 21:return"Scope";case 13:return"Suspense";case 19:return"SuspenseList";case 25:return"TracingMarker";case 1:case 0:case 17:case 2:case 14:case 15:if(typeof t=="function")return t.displayName||t.name||null;if(typeof t=="string")return t}return null}function V(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function Y(e){var t=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(t==="checkbox"||t==="radio")}function Re(e){var t=Y(e)?"checked":"value",n=Object.getOwnPropertyDescriptor(e.constructor.prototype,t),r=""+e[t];if(!e.hasOwnProperty(t)&&typeof n<"u"&&typeof n.get=="function"&&typeof n.set=="function"){var l=n.get,u=n.set;return Object.defineProperty(e,t,{configurable:!0,get:function(){return l.call(this)},set:function(i){r=""+i,u.call(this,i)}}),Object.defineProperty(e,t,{enumerable:n.enumerable}),{getValue:function(){return r},setValue:function(i){r=""+i},stopTracking:function(){e._valueTracker=null,delete e[t]}}}}function hr(e){e._valueTracker||(e._valueTracker=Re(e))}function Ci(e){if(!e)return!1;var t=e._valueTracker;if(!t)return!0;var n=t.getValue(),r="";return e&&(r=Y(e)?e.checked?"true":"false":e.value),e=r,e!==n?(t.setValue(e),!0):!1}function yr(e){if(e=e||(typeof document<"u"?document:void 0),typeof e>"u")return null;try{return e.activeElement||e.body}catch{return e.body}}function Pl(e,t){var n=t.checked;return C({},t,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:n??e._wrapperState.initialChecked})}function _i(e,t){var n=t.defaultValue==null?"":t.defaultValue,r=t.checked!=null?t.checked:t.defaultChecked;n=V(t.value!=null?t.value:n),e._wrapperState={initialChecked:r,initialValue:n,controlled:t.type==="checkbox"||t.type==="radio"?t.checked!=null:t.value!=null}}function xi(e,t){t=t.checked,t!=null&&Xe(e,"checked",t,!1)}function Nl(e,t){xi(e,t);var n=V(t.value),r=t.type;if(n!=null)r==="number"?(n===0&&e.value===""||e.value!=n)&&(e.value=""+n):e.value!==""+n&&(e.value=""+n);else if(r==="submit"||r==="reset"){e.removeAttribute("value");return}t.hasOwnProperty("value")?zl(e,t.type,n):t.hasOwnProperty("defaultValue")&&zl(e,t.type,V(t.defaultValue)),t.checked==null&&t.defaultChecked!=null&&(e.defaultChecked=!!t.defaultChecked)}function Pi(e,t,n){if(t.hasOwnProperty("value")||t.hasOwnProperty("defaultValue")){var r=t.type;if(!(r!=="submit"&&r!=="reset"||t.value!==void 0&&t.value!==null))return;t=""+e._wrapperState.initialValue,n||t===e.value||(e.value=t),e.defaultValue=t}n=e.name,n!==""&&(e.name=""),e.defaultChecked=!!e._wrapperState.initialChecked,n!==""&&(e.name=n)}function zl(e,t,n){(t!=="number"||yr(e.ownerDocument)!==e)&&(n==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+n&&(e.defaultValue=""+n))}var Rn=Array.isArray;function un(e,t,n,r){if(e=e.options,t){t={};for(var l=0;l"+t.valueOf().toString()+"",t=gr.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;t.firstChild;)e.appendChild(t.firstChild)}});function Mn(e,t){if(t){var n=e.firstChild;if(n&&n===e.lastChild&&n.nodeType===3){n.nodeValue=t;return}}e.textContent=t}var Dn={animationIterationCount:!0,aspectRatio:!0,borderImageOutset:!0,borderImageSlice:!0,borderImageWidth:!0,boxFlex:!0,boxFlexGroup:!0,boxOrdinalGroup:!0,columnCount:!0,columns:!0,flex:!0,flexGrow:!0,flexPositive:!0,flexShrink:!0,flexNegative:!0,flexOrder:!0,gridArea:!0,gridRow:!0,gridRowEnd:!0,gridRowSpan:!0,gridRowStart:!0,gridColumn:!0,gridColumnEnd:!0,gridColumnSpan:!0,gridColumnStart:!0,fontWeight:!0,lineClamp:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,tabSize:!0,widows:!0,zIndex:!0,zoom:!0,fillOpacity:!0,floodOpacity:!0,stopOpacity:!0,strokeDasharray:!0,strokeDashoffset:!0,strokeMiterlimit:!0,strokeOpacity:!0,strokeWidth:!0},_a=["Webkit","ms","Moz","O"];Object.keys(Dn).forEach(function(e){_a.forEach(function(t){t=t+e.charAt(0).toUpperCase()+e.substring(1),Dn[t]=Dn[e]})});function Mi(e,t,n){return t==null||typeof t=="boolean"||t===""?"":n||typeof t!="number"||t===0||Dn.hasOwnProperty(e)&&Dn[e]?(""+t).trim():t+"px"}function Di(e,t){e=e.style;for(var n in t)if(t.hasOwnProperty(n)){var r=n.indexOf("--")===0,l=Mi(n,t[n],r);n==="float"&&(n="cssFloat"),r?e.setProperty(n,l):e[n]=l}}var xa=C({menuitem:!0},{area:!0,base:!0,br:!0,col:!0,embed:!0,hr:!0,img:!0,input:!0,keygen:!0,link:!0,meta:!0,param:!0,source:!0,track:!0,wbr:!0});function Rl(e,t){if(t){if(xa[e]&&(t.children!=null||t.dangerouslySetInnerHTML!=null))throw Error(m(137,e));if(t.dangerouslySetInnerHTML!=null){if(t.children!=null)throw Error(m(60));if(typeof t.dangerouslySetInnerHTML!="object"||!("__html"in t.dangerouslySetInnerHTML))throw Error(m(61))}if(t.style!=null&&typeof t.style!="object")throw Error(m(62))}}function Ml(e,t){if(e.indexOf("-")===-1)return typeof t.is=="string";switch(e){case"annotation-xml":case"color-profile":case"font-face":case"font-face-src":case"font-face-uri":case"font-face-format":case"font-face-name":case"missing-glyph":return!1;default:return!0}}var Dl=null;function Ol(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var Il=null,on=null,sn=null;function Oi(e){if(e=tr(e)){if(typeof Il!="function")throw Error(m(280));var t=e.stateNode;t&&(t=Br(t),Il(e.stateNode,e.type,t))}}function Ii(e){on?sn?sn.push(e):sn=[e]:on=e}function Fi(){if(on){var e=on,t=sn;if(sn=on=null,Oi(e),t)for(e=0;e>>=0,e===0?32:31-(Fa(e)/ja|0)|0}var Cr=64,_r=4194304;function jn(e){switch(e&-e){case 1:return 1;case 2:return 2;case 4:return 4;case 8:return 8;case 16:return 16;case 32:return 32;case 64:case 128:case 256:case 512:case 1024:case 2048:case 4096:case 8192:case 16384:case 32768:case 65536:case 131072:case 262144:case 524288:case 1048576:case 2097152:return e&4194240;case 4194304:case 8388608:case 16777216:case 33554432:case 67108864:return e&130023424;case 134217728:return 134217728;case 268435456:return 268435456;case 536870912:return 536870912;case 1073741824:return 1073741824;default:return e}}function xr(e,t){var n=e.pendingLanes;if(n===0)return 0;var r=0,l=e.suspendedLanes,u=e.pingedLanes,i=n&268435455;if(i!==0){var o=i&~l;o!==0?r=jn(o):(u&=i,u!==0&&(r=jn(u)))}else i=n&~l,i!==0?r=jn(i):u!==0&&(r=jn(u));if(r===0)return 0;if(t!==0&&t!==r&&(t&l)===0&&(l=r&-r,u=t&-t,l>=u||l===16&&(u&4194240)!==0))return t;if((r&4)!==0&&(r|=n&16),t=e.entangledLanes,t!==0)for(e=e.entanglements,t&=r;0n;n++)t.push(e);return t}function Un(e,t,n){e.pendingLanes|=t,t!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,t=31-Je(t),e[t]=n}function Ba(e,t){var n=e.pendingLanes&~t;e.pendingLanes=t,e.suspendedLanes=0,e.pingedLanes=0,e.expiredLanes&=t,e.mutableReadLanes&=t,e.entangledLanes&=t,t=e.entanglements;var r=e.eventTimes;for(e=e.expirationTimes;0=Kn),fo=" ",co=!1;function po(e,t){switch(e){case"keyup":return hf.indexOf(t.keyCode)!==-1;case"keydown":return t.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function mo(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var cn=!1;function gf(e,t){switch(e){case"compositionend":return mo(t);case"keypress":return t.which!==32?null:(co=!0,fo);case"textInput":return e=t.data,e===fo&&co?null:e;default:return null}}function wf(e,t){if(cn)return e==="compositionend"||!bl&&po(e,t)?(e=lo(),Lr=Yl=Tt=null,cn=!1,e):null;switch(e){case"paste":return null;case"keypress":if(!(t.ctrlKey||t.altKey||t.metaKey)||t.ctrlKey&&t.altKey){if(t.char&&1=t)return{node:n,offset:t-e};e=r}e:{for(;n;){if(n.nextSibling){n=n.nextSibling;break e}n=n.parentNode}n=void 0}n=ko(n)}}function Co(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?Co(e,t.parentNode):"contains"in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function _o(){for(var e=window,t=yr();t instanceof e.HTMLIFrameElement;){try{var n=typeof t.contentWindow.location.href=="string"}catch{n=!1}if(n)e=t.contentWindow;else break;t=yr(e.document)}return t}function nu(e){var t=e&&e.nodeName&&e.nodeName.toLowerCase();return t&&(t==="input"&&(e.type==="text"||e.type==="search"||e.type==="tel"||e.type==="url"||e.type==="password")||t==="textarea"||e.contentEditable==="true")}function zf(e){var t=_o(),n=e.focusedElem,r=e.selectionRange;if(t!==n&&n&&n.ownerDocument&&Co(n.ownerDocument.documentElement,n)){if(r!==null&&nu(n)){if(t=r.start,e=r.end,e===void 0&&(e=t),"selectionStart"in n)n.selectionStart=t,n.selectionEnd=Math.min(e,n.value.length);else if(e=(t=n.ownerDocument||document)&&t.defaultView||window,e.getSelection){e=e.getSelection();var l=n.textContent.length,u=Math.min(r.start,l);r=r.end===void 0?u:Math.min(r.end,l),!e.extend&&u>r&&(l=r,r=u,u=l),l=Eo(n,u);var i=Eo(n,r);l&&i&&(e.rangeCount!==1||e.anchorNode!==l.node||e.anchorOffset!==l.offset||e.focusNode!==i.node||e.focusOffset!==i.offset)&&(t=t.createRange(),t.setStart(l.node,l.offset),e.removeAllRanges(),u>r?(e.addRange(t),e.extend(i.node,i.offset)):(t.setEnd(i.node,i.offset),e.addRange(t)))}}for(t=[],e=n;e=e.parentNode;)e.nodeType===1&&t.push({element:e,left:e.scrollLeft,top:e.scrollTop});for(typeof n.focus=="function"&&n.focus(),n=0;n=document.documentMode,dn=null,ru=null,Zn=null,lu=!1;function xo(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;lu||dn==null||dn!==yr(r)||(r=dn,"selectionStart"in r&&nu(r)?r={start:r.selectionStart,end:r.selectionEnd}:(r=(r.ownerDocument&&r.ownerDocument.defaultView||window).getSelection(),r={anchorNode:r.anchorNode,anchorOffset:r.anchorOffset,focusNode:r.focusNode,focusOffset:r.focusOffset}),Zn&&Gn(Zn,r)||(Zn=r,r=Ur(ru,"onSelect"),0yn||(e.current=hu[yn],hu[yn]=null,yn--)}function W(e,t){yn++,hu[yn]=e.current,e.current=t}var Dt={},me=Mt(Dt),Ce=Mt(!1),Yt=Dt;function gn(e,t){var n=e.type.contextTypes;if(!n)return Dt;var r=e.stateNode;if(r&&r.__reactInternalMemoizedUnmaskedChildContext===t)return r.__reactInternalMemoizedMaskedChildContext;var l={},u;for(u in n)l[u]=t[u];return r&&(e=e.stateNode,e.__reactInternalMemoizedUnmaskedChildContext=t,e.__reactInternalMemoizedMaskedChildContext=l),l}function _e(e){return e=e.childContextTypes,e!=null}function Hr(){Q(Ce),Q(me)}function Ao(e,t,n){if(me.current!==Dt)throw Error(m(168));W(me,t),W(Ce,n)}function Bo(e,t,n){var r=e.stateNode;if(t=t.childContextTypes,typeof r.getChildContext!="function")return n;r=r.getChildContext();for(var l in r)if(!(l in t))throw Error(m(108,H(e)||"Unknown",l));return C({},n,r)}function Wr(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||Dt,Yt=me.current,W(me,e),W(Ce,Ce.current),!0}function Ho(e,t,n){var r=e.stateNode;if(!r)throw Error(m(169));n?(e=Bo(e,t,Yt),r.__reactInternalMemoizedMergedChildContext=e,Q(Ce),Q(me),W(me,e)):Q(Ce),W(Ce,n)}var yt=null,$r=!1,yu=!1;function Wo(e){yt===null?yt=[e]:yt.push(e)}function Af(e){$r=!0,Wo(e)}function Ot(){if(!yu&&yt!==null){yu=!0;var e=0,t=A;try{var n=yt;for(A=1;e>=i,l-=i,gt=1<<32-Je(t)+l|n<L?(ae=z,z=null):ae=z.sibling;var U=v(c,z,d[L],w);if(U===null){z===null&&(z=ae);break}e&&z&&U.alternate===null&&t(c,z),a=u(U,a,L),N===null?P=U:N.sibling=U,N=U,z=ae}if(L===d.length)return n(c,z),X&&Gt(c,L),P;if(z===null){for(;LL?(ae=z,z=null):ae=z.sibling;var Wt=v(c,z,U.value,w);if(Wt===null){z===null&&(z=ae);break}e&&z&&Wt.alternate===null&&t(c,z),a=u(Wt,a,L),N===null?P=Wt:N.sibling=Wt,N=Wt,z=ae}if(U.done)return n(c,z),X&&Gt(c,L),P;if(z===null){for(;!U.done;L++,U=d.next())U=g(c,U.value,w),U!==null&&(a=u(U,a,L),N===null?P=U:N.sibling=U,N=U);return X&&Gt(c,L),P}for(z=r(c,z);!U.done;L++,U=d.next())U=k(z,c,L,U.value,w),U!==null&&(e&&U.alternate!==null&&z.delete(U.key===null?L:U.key),a=u(U,a,L),N===null?P=U:N.sibling=U,N=U);return e&&z.forEach(function(Sc){return t(c,Sc)}),X&&Gt(c,L),P}function te(c,a,d,w){if(typeof d=="object"&&d!==null&&d.type===ke&&d.key===null&&(d=d.props.children),typeof d=="object"&&d!==null){switch(d.$$typeof){case Ge:e:{for(var P=d.key,N=a;N!==null;){if(N.key===P){if(P=d.type,P===ke){if(N.tag===7){n(c,N.sibling),a=l(N,d.props.children),a.return=c,c=a;break e}}else if(N.elementType===P||typeof P=="object"&&P!==null&&P.$$typeof===Ee&&Go(P)===N.type){n(c,N.sibling),a=l(N,d.props),a.ref=nr(c,N,d),a.return=c,c=a;break e}n(c,N);break}else t(c,N);N=N.sibling}d.type===ke?(a=rn(d.props.children,c.mode,w,d.key),a.return=c,c=a):(w=gl(d.type,d.key,d.props,null,c.mode,w),w.ref=nr(c,a,d),w.return=c,c=w)}return i(c);case ge:e:{for(N=d.key;a!==null;){if(a.key===N)if(a.tag===4&&a.stateNode.containerInfo===d.containerInfo&&a.stateNode.implementation===d.implementation){n(c,a.sibling),a=l(a,d.children||[]),a.return=c,c=a;break e}else{n(c,a);break}else t(c,a);a=a.sibling}a=mi(d,c.mode,w),a.return=c,c=a}return i(c);case Ee:return N=d._init,te(c,a,N(d._payload),w)}if(Rn(d))return _(c,a,d,w);if(T(d))return x(c,a,d,w);Xr(c,d)}return typeof d=="string"&&d!==""||typeof d=="number"?(d=""+d,a!==null&&a.tag===6?(n(c,a.sibling),a=l(a,d),a.return=c,c=a):(n(c,a),a=pi(d,c.mode,w),a.return=c,c=a),i(c)):n(c,a)}return te}var En=Zo(!0),Jo=Zo(!1),Gr=Mt(null),Zr=null,Cn=null,Cu=null;function _u(){Cu=Cn=Zr=null}function xu(e){var t=Gr.current;Q(Gr),e._currentValue=t}function Pu(e,t,n){for(;e!==null;){var r=e.alternate;if((e.childLanes&t)!==t?(e.childLanes|=t,r!==null&&(r.childLanes|=t)):r!==null&&(r.childLanes&t)!==t&&(r.childLanes|=t),e===n)break;e=e.return}}function _n(e,t){Zr=e,Cu=Cn=null,e=e.dependencies,e!==null&&e.firstContext!==null&&((e.lanes&t)!==0&&(xe=!0),e.firstContext=null)}function He(e){var t=e._currentValue;if(Cu!==e)if(e={context:e,memoizedValue:t,next:null},Cn===null){if(Zr===null)throw Error(m(308));Cn=e,Zr.dependencies={lanes:0,firstContext:e}}else Cn=Cn.next=e;return t}var Zt=null;function Nu(e){Zt===null?Zt=[e]:Zt.push(e)}function qo(e,t,n,r){var l=t.interleaved;return l===null?(n.next=n,Nu(t)):(n.next=l.next,l.next=n),t.interleaved=n,St(e,r)}function St(e,t){e.lanes|=t;var n=e.alternate;for(n!==null&&(n.lanes|=t),n=e,e=e.return;e!==null;)e.childLanes|=t,n=e.alternate,n!==null&&(n.childLanes|=t),n=e,e=e.return;return n.tag===3?n.stateNode:null}var It=!1;function zu(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,interleaved:null,lanes:0},effects:null}}function bo(e,t){e=e.updateQueue,t.updateQueue===e&&(t.updateQueue={baseState:e.baseState,firstBaseUpdate:e.firstBaseUpdate,lastBaseUpdate:e.lastBaseUpdate,shared:e.shared,effects:e.effects})}function kt(e,t){return{eventTime:e,lane:t,tag:0,payload:null,callback:null,next:null}}function Ft(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,(I&2)!==0){var l=r.pending;return l===null?t.next=t:(t.next=l.next,l.next=t),r.pending=t,St(e,n)}return l=r.interleaved,l===null?(t.next=t,Nu(r)):(t.next=l.next,l.next=t),r.interleaved=t,St(e,n)}function Jr(e,t,n){if(t=t.updateQueue,t!==null&&(t=t.shared,(n&4194240)!==0)){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,Hl(e,n)}}function es(e,t){var n=e.updateQueue,r=e.alternate;if(r!==null&&(r=r.updateQueue,n===r)){var l=null,u=null;if(n=n.firstBaseUpdate,n!==null){do{var i={eventTime:n.eventTime,lane:n.lane,tag:n.tag,payload:n.payload,callback:n.callback,next:null};u===null?l=u=i:u=u.next=i,n=n.next}while(n!==null);u===null?l=u=t:u=u.next=t}else l=u=t;n={baseState:r.baseState,firstBaseUpdate:l,lastBaseUpdate:u,shared:r.shared,effects:r.effects},e.updateQueue=n;return}e=n.lastBaseUpdate,e===null?n.firstBaseUpdate=t:e.next=t,n.lastBaseUpdate=t}function qr(e,t,n,r){var l=e.updateQueue;It=!1;var u=l.firstBaseUpdate,i=l.lastBaseUpdate,o=l.shared.pending;if(o!==null){l.shared.pending=null;var s=o,p=s.next;s.next=null,i===null?u=p:i.next=p,i=s;var y=e.alternate;y!==null&&(y=y.updateQueue,o=y.lastBaseUpdate,o!==i&&(o===null?y.firstBaseUpdate=p:o.next=p,y.lastBaseUpdate=s))}if(u!==null){var g=l.baseState;i=0,y=p=s=null,o=u;do{var v=o.lane,k=o.eventTime;if((r&v)===v){y!==null&&(y=y.next={eventTime:k,lane:0,tag:o.tag,payload:o.payload,callback:o.callback,next:null});e:{var _=e,x=o;switch(v=t,k=n,x.tag){case 1:if(_=x.payload,typeof _=="function"){g=_.call(k,g,v);break e}g=_;break e;case 3:_.flags=_.flags&-65537|128;case 0:if(_=x.payload,v=typeof _=="function"?_.call(k,g,v):_,v==null)break e;g=C({},g,v);break e;case 2:It=!0}}o.callback!==null&&o.lane!==0&&(e.flags|=64,v=l.effects,v===null?l.effects=[o]:v.push(o))}else k={eventTime:k,lane:v,tag:o.tag,payload:o.payload,callback:o.callback,next:null},y===null?(p=y=k,s=g):y=y.next=k,i|=v;if(o=o.next,o===null){if(o=l.shared.pending,o===null)break;v=o,o=v.next,v.next=null,l.lastBaseUpdate=v,l.shared.pending=null}}while(!0);if(y===null&&(s=g),l.baseState=s,l.firstBaseUpdate=p,l.lastBaseUpdate=y,t=l.shared.interleaved,t!==null){l=t;do i|=l.lane,l=l.next;while(l!==t)}else u===null&&(l.shared.lanes=0);bt|=i,e.lanes=i,e.memoizedState=g}}function ts(e,t,n){if(e=t.effects,t.effects=null,e!==null)for(t=0;tn?n:4,e(!0);var r=Du.transition;Du.transition={};try{e(!1),t()}finally{A=n,Du.transition=r}}function Ss(){return We().memoizedState}function $f(e,t,n){var r=At(e);if(n={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null},ks(e))Es(t,n);else if(n=qo(e,t,n,r),n!==null){var l=Se();rt(n,e,r,l),Cs(n,t,r)}}function Qf(e,t,n){var r=At(e),l={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null};if(ks(e))Es(t,l);else{var u=e.alternate;if(e.lanes===0&&(u===null||u.lanes===0)&&(u=t.lastRenderedReducer,u!==null))try{var i=t.lastRenderedState,o=u(i,n);if(l.hasEagerState=!0,l.eagerState=o,qe(o,i)){var s=t.interleaved;s===null?(l.next=l,Nu(t)):(l.next=s.next,s.next=l),t.interleaved=l;return}}catch{}finally{}n=qo(e,t,l,r),n!==null&&(l=Se(),rt(n,e,r,l),Cs(n,t,r))}}function ks(e){var t=e.alternate;return e===Z||t!==null&&t===Z}function Es(e,t){ir=tl=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function Cs(e,t,n){if((n&4194240)!==0){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,Hl(e,n)}}var ll={readContext:He,useCallback:ve,useContext:ve,useEffect:ve,useImperativeHandle:ve,useInsertionEffect:ve,useLayoutEffect:ve,useMemo:ve,useReducer:ve,useRef:ve,useState:ve,useDebugValue:ve,useDeferredValue:ve,useTransition:ve,useMutableSource:ve,useSyncExternalStore:ve,useId:ve,unstable_isNewReconciler:!1},Kf={readContext:He,useCallback:function(e,t){return ft().memoizedState=[e,t===void 0?null:t],e},useContext:He,useEffect:ds,useImperativeHandle:function(e,t,n){return n=n!=null?n.concat([e]):null,nl(4194308,4,vs.bind(null,t,e),n)},useLayoutEffect:function(e,t){return nl(4194308,4,e,t)},useInsertionEffect:function(e,t){return nl(4,2,e,t)},useMemo:function(e,t){var n=ft();return t=t===void 0?null:t,e=e(),n.memoizedState=[e,t],e},useReducer:function(e,t,n){var r=ft();return t=n!==void 0?n(t):t,r.memoizedState=r.baseState=t,e={pending:null,interleaved:null,lanes:0,dispatch:null,lastRenderedReducer:e,lastRenderedState:t},r.queue=e,e=e.dispatch=$f.bind(null,Z,e),[r.memoizedState,e]},useRef:function(e){var t=ft();return e={current:e},t.memoizedState=e},useState:fs,useDebugValue:Au,useDeferredValue:function(e){return ft().memoizedState=e},useTransition:function(){var e=fs(!1),t=e[0];return e=Wf.bind(null,e[1]),ft().memoizedState=e,[t,e]},useMutableSource:function(){},useSyncExternalStore:function(e,t,n){var r=Z,l=ft();if(X){if(n===void 0)throw Error(m(407));n=n()}else{if(n=t(),se===null)throw Error(m(349));(qt&30)!==0||us(r,t,n)}l.memoizedState=n;var u={value:n,getSnapshot:t};return l.queue=u,ds(os.bind(null,r,u,e),[e]),r.flags|=2048,ar(9,is.bind(null,r,u,n,t),void 0,null),n},useId:function(){var e=ft(),t=se.identifierPrefix;if(X){var n=wt,r=gt;n=(r&~(1<<32-Je(r)-1)).toString(32)+n,t=":"+t+"R"+n,n=or++,0<\/script>",e=e.removeChild(e.firstChild)):typeof r.is=="string"?e=i.createElement(n,{is:r.is}):(e=i.createElement(n),n==="select"&&(i=e,r.multiple?i.multiple=!0:r.size&&(i.size=r.size))):e=i.createElementNS(e,n),e[st]=t,e[er]=r,Ws(e,t,!1,!1),t.stateNode=e;e:{switch(i=Ml(n,r),n){case"dialog":$("cancel",e),$("close",e),l=r;break;case"iframe":case"object":case"embed":$("load",e),l=r;break;case"video":case"audio":for(l=0;lTn&&(t.flags|=128,r=!0,fr(u,!1),t.lanes=4194304)}else{if(!r)if(e=br(i),e!==null){if(t.flags|=128,r=!0,n=e.updateQueue,n!==null&&(t.updateQueue=n,t.flags|=4),fr(u,!0),u.tail===null&&u.tailMode==="hidden"&&!i.alternate&&!X)return he(t),null}else 2*ee()-u.renderingStartTime>Tn&&n!==1073741824&&(t.flags|=128,r=!0,fr(u,!1),t.lanes=4194304);u.isBackwards?(i.sibling=t.child,t.child=i):(n=u.last,n!==null?n.sibling=i:t.child=i,u.last=i)}return u.tail!==null?(t=u.tail,u.rendering=t,u.tail=t.sibling,u.renderingStartTime=ee(),t.sibling=null,n=G.current,W(G,r?n&1|2:n&1),t):(he(t),null);case 22:case 23:return fi(),r=t.memoizedState!==null,e!==null&&e.memoizedState!==null!==r&&(t.flags|=8192),r&&(t.mode&1)!==0?(Ie&1073741824)!==0&&(he(t),t.subtreeFlags&6&&(t.flags|=8192)):he(t),null;case 24:return null;case 25:return null}throw Error(m(156,t.tag))}function ec(e,t){switch(wu(t),t.tag){case 1:return _e(t.type)&&Hr(),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return xn(),Q(Ce),Q(me),Mu(),e=t.flags,(e&65536)!==0&&(e&128)===0?(t.flags=e&-65537|128,t):null;case 5:return Lu(t),null;case 13:if(Q(G),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(m(340));kn()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return Q(G),null;case 4:return xn(),null;case 10:return xu(t.type._context),null;case 22:case 23:return fi(),null;case 24:return null;default:return null}}var sl=!1,ye=!1,tc=typeof WeakSet=="function"?WeakSet:Set,E=null;function Nn(e,t){var n=e.ref;if(n!==null)if(typeof n=="function")try{n(null)}catch(r){q(e,t,r)}else n.current=null}function qu(e,t,n){try{n()}catch(r){q(e,t,r)}}var Ks=!1;function nc(e,t){if(fu=zr,e=_o(),nu(e)){if("selectionStart"in e)var n={start:e.selectionStart,end:e.selectionEnd};else e:{n=(n=e.ownerDocument)&&n.defaultView||window;var r=n.getSelection&&n.getSelection();if(r&&r.rangeCount!==0){n=r.anchorNode;var l=r.anchorOffset,u=r.focusNode;r=r.focusOffset;try{n.nodeType,u.nodeType}catch{n=null;break e}var i=0,o=-1,s=-1,p=0,y=0,g=e,v=null;t:for(;;){for(var k;g!==n||l!==0&&g.nodeType!==3||(o=i+l),g!==u||r!==0&&g.nodeType!==3||(s=i+r),g.nodeType===3&&(i+=g.nodeValue.length),(k=g.firstChild)!==null;)v=g,g=k;for(;;){if(g===e)break t;if(v===n&&++p===l&&(o=i),v===u&&++y===r&&(s=i),(k=g.nextSibling)!==null)break;g=v,v=g.parentNode}g=k}n=o===-1||s===-1?null:{start:o,end:s}}else n=null}n=n||{start:0,end:0}}else n=null;for(cu={focusedElem:e,selectionRange:n},zr=!1,E=t;E!==null;)if(t=E,e=t.child,(t.subtreeFlags&1028)!==0&&e!==null)e.return=t,E=e;else for(;E!==null;){t=E;try{var _=t.alternate;if((t.flags&1024)!==0)switch(t.tag){case 0:case 11:case 15:break;case 1:if(_!==null){var x=_.memoizedProps,te=_.memoizedState,c=t.stateNode,a=c.getSnapshotBeforeUpdate(t.elementType===t.type?x:et(t.type,x),te);c.__reactInternalSnapshotBeforeUpdate=a}break;case 3:var d=t.stateNode.containerInfo;d.nodeType===1?d.textContent="":d.nodeType===9&&d.documentElement&&d.removeChild(d.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(m(163))}}catch(w){q(t,t.return,w)}if(e=t.sibling,e!==null){e.return=t.return,E=e;break}E=t.return}return _=Ks,Ks=!1,_}function cr(e,t,n){var r=t.updateQueue;if(r=r!==null?r.lastEffect:null,r!==null){var l=r=r.next;do{if((l.tag&e)===e){var u=l.destroy;l.destroy=void 0,u!==void 0&&qu(t,n,u)}l=l.next}while(l!==r)}}function al(e,t){if(t=t.updateQueue,t=t!==null?t.lastEffect:null,t!==null){var n=t=t.next;do{if((n.tag&e)===e){var r=n.create;n.destroy=r()}n=n.next}while(n!==t)}}function bu(e){var t=e.ref;if(t!==null){var n=e.stateNode;switch(e.tag){case 5:e=n;break;default:e=n}typeof t=="function"?t(e):t.current=e}}function Ys(e){var t=e.alternate;t!==null&&(e.alternate=null,Ys(t)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(t=e.stateNode,t!==null&&(delete t[st],delete t[er],delete t[vu],delete t[Uf],delete t[Vf])),e.stateNode=null,e.return=null,e.dependencies=null,e.memoizedProps=null,e.memoizedState=null,e.pendingProps=null,e.stateNode=null,e.updateQueue=null}function Xs(e){return e.tag===5||e.tag===3||e.tag===4}function Gs(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||Xs(e.return))return null;e=e.return}for(e.sibling.return=e.return,e=e.sibling;e.tag!==5&&e.tag!==6&&e.tag!==18;){if(e.flags&2||e.child===null||e.tag===4)continue e;e.child.return=e,e=e.child}if(!(e.flags&2))return e.stateNode}}function ei(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.nodeType===8?n.parentNode.insertBefore(e,t):n.insertBefore(e,t):(n.nodeType===8?(t=n.parentNode,t.insertBefore(e,n)):(t=n,t.appendChild(e)),n=n._reactRootContainer,n!=null||t.onclick!==null||(t.onclick=Ar));else if(r!==4&&(e=e.child,e!==null))for(ei(e,t,n),e=e.sibling;e!==null;)ei(e,t,n),e=e.sibling}function ti(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.insertBefore(e,t):n.appendChild(e);else if(r!==4&&(e=e.child,e!==null))for(ti(e,t,n),e=e.sibling;e!==null;)ti(e,t,n),e=e.sibling}var fe=null,tt=!1;function jt(e,t,n){for(n=n.child;n!==null;)Zs(e,t,n),n=n.sibling}function Zs(e,t,n){if(ot&&typeof ot.onCommitFiberUnmount=="function")try{ot.onCommitFiberUnmount(Er,n)}catch{}switch(n.tag){case 5:ye||Nn(n,t);case 6:var r=fe,l=tt;fe=null,jt(e,t,n),fe=r,tt=l,fe!==null&&(tt?(e=fe,n=n.stateNode,e.nodeType===8?e.parentNode.removeChild(n):e.removeChild(n)):fe.removeChild(n.stateNode));break;case 18:fe!==null&&(tt?(e=fe,n=n.stateNode,e.nodeType===8?mu(e.parentNode,n):e.nodeType===1&&mu(e,n),Wn(e)):mu(fe,n.stateNode));break;case 4:r=fe,l=tt,fe=n.stateNode.containerInfo,tt=!0,jt(e,t,n),fe=r,tt=l;break;case 0:case 11:case 14:case 15:if(!ye&&(r=n.updateQueue,r!==null&&(r=r.lastEffect,r!==null))){l=r=r.next;do{var u=l,i=u.destroy;u=u.tag,i!==void 0&&((u&2)!==0||(u&4)!==0)&&qu(n,t,i),l=l.next}while(l!==r)}jt(e,t,n);break;case 1:if(!ye&&(Nn(n,t),r=n.stateNode,typeof r.componentWillUnmount=="function"))try{r.props=n.memoizedProps,r.state=n.memoizedState,r.componentWillUnmount()}catch(o){q(n,t,o)}jt(e,t,n);break;case 21:jt(e,t,n);break;case 22:n.mode&1?(ye=(r=ye)||n.memoizedState!==null,jt(e,t,n),ye=r):jt(e,t,n);break;default:jt(e,t,n)}}function Js(e){var t=e.updateQueue;if(t!==null){e.updateQueue=null;var n=e.stateNode;n===null&&(n=e.stateNode=new tc),t.forEach(function(r){var l=cc.bind(null,e,r);n.has(r)||(n.add(r),r.then(l,l))})}}function nt(e,t){var n=t.deletions;if(n!==null)for(var r=0;rl&&(l=i),r&=~u}if(r=l,r=ee()-r,r=(120>r?120:480>r?480:1080>r?1080:1920>r?1920:3e3>r?3e3:4320>r?4320:1960*lc(r/1960))-r,10e?16:e,Vt===null)var r=!1;else{if(e=Vt,Vt=null,ml=0,(I&6)!==0)throw Error(m(331));var l=I;for(I|=4,E=e.current;E!==null;){var u=E,i=u.child;if((E.flags&16)!==0){var o=u.deletions;if(o!==null){for(var s=0;see()-li?tn(e,0):ri|=n),Ne(e,t)}function fa(e,t){t===0&&((e.mode&1)===0?t=1:(t=_r,_r<<=1,(_r&130023424)===0&&(_r=4194304)));var n=Se();e=St(e,t),e!==null&&(Un(e,t,n),Ne(e,n))}function fc(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),fa(e,n)}function cc(e,t){var n=0;switch(e.tag){case 13:var r=e.stateNode,l=e.memoizedState;l!==null&&(n=l.retryLane);break;case 19:r=e.stateNode;break;default:throw Error(m(314))}r!==null&&r.delete(t),fa(e,n)}var ca;ca=function(e,t,n){if(e!==null)if(e.memoizedProps!==t.pendingProps||Ce.current)xe=!0;else{if((e.lanes&n)===0&&(t.flags&128)===0)return xe=!1,qf(e,t,n);xe=(e.flags&131072)!==0}else xe=!1,X&&(t.flags&1048576)!==0&&$o(t,Kr,t.index);switch(t.lanes=0,t.tag){case 2:var r=t.type;ol(e,t),e=t.pendingProps;var l=gn(t,me.current);_n(t,n),l=Iu(null,t,r,e,l,n);var u=Fu();return t.flags|=1,typeof l=="object"&&l!==null&&typeof l.render=="function"&&l.$$typeof===void 0?(t.tag=1,t.memoizedState=null,t.updateQueue=null,_e(r)?(u=!0,Wr(t)):u=!1,t.memoizedState=l.state!==null&&l.state!==void 0?l.state:null,zu(t),l.updater=ul,t.stateNode=l,l._reactInternals=t,Hu(t,r,e,n),t=Ku(null,t,r,!0,u,n)):(t.tag=0,X&&u&&gu(t),we(null,t,l,n),t=t.child),t;case 16:r=t.elementType;e:{switch(ol(e,t),e=t.pendingProps,l=r._init,r=l(r._payload),t.type=r,l=t.tag=pc(r),e=et(r,e),l){case 0:t=Qu(null,t,r,e,n);break e;case 1:t=js(null,t,r,e,n);break e;case 11:t=Ms(null,t,r,e,n);break e;case 14:t=Ds(null,t,r,et(r.type,e),n);break e}throw Error(m(306,r,""))}return t;case 0:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:et(r,l),Qu(e,t,r,l,n);case 1:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:et(r,l),js(e,t,r,l,n);case 3:e:{if(Us(t),e===null)throw Error(m(387));r=t.pendingProps,u=t.memoizedState,l=u.element,bo(e,t),qr(t,r,null,n);var i=t.memoizedState;if(r=i.element,u.isDehydrated)if(u={element:r,isDehydrated:!1,cache:i.cache,pendingSuspenseBoundaries:i.pendingSuspenseBoundaries,transitions:i.transitions},t.updateQueue.baseState=u,t.memoizedState=u,t.flags&256){l=Pn(Error(m(423)),t),t=Vs(e,t,r,n,l);break e}else if(r!==l){l=Pn(Error(m(424)),t),t=Vs(e,t,r,n,l);break e}else for(Oe=Rt(t.stateNode.containerInfo.firstChild),De=t,X=!0,be=null,n=Jo(t,null,r,n),t.child=n;n;)n.flags=n.flags&-3|4096,n=n.sibling;else{if(kn(),r===l){t=Et(e,t,n);break e}we(e,t,r,n)}t=t.child}return t;case 5:return ns(t),e===null&&ku(t),r=t.type,l=t.pendingProps,u=e!==null?e.memoizedProps:null,i=l.children,du(r,l)?i=null:u!==null&&du(r,u)&&(t.flags|=32),Fs(e,t),we(e,t,i,n),t.child;case 6:return e===null&&ku(t),null;case 13:return As(e,t,n);case 4:return Tu(t,t.stateNode.containerInfo),r=t.pendingProps,e===null?t.child=En(t,null,r,n):we(e,t,r,n),t.child;case 11:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:et(r,l),Ms(e,t,r,l,n);case 7:return we(e,t,t.pendingProps,n),t.child;case 8:return we(e,t,t.pendingProps.children,n),t.child;case 12:return we(e,t,t.pendingProps.children,n),t.child;case 10:e:{if(r=t.type._context,l=t.pendingProps,u=t.memoizedProps,i=l.value,W(Gr,r._currentValue),r._currentValue=i,u!==null)if(qe(u.value,i)){if(u.children===l.children&&!Ce.current){t=Et(e,t,n);break e}}else for(u=t.child,u!==null&&(u.return=t);u!==null;){var o=u.dependencies;if(o!==null){i=u.child;for(var s=o.firstContext;s!==null;){if(s.context===r){if(u.tag===1){s=kt(-1,n&-n),s.tag=2;var p=u.updateQueue;if(p!==null){p=p.shared;var y=p.pending;y===null?s.next=s:(s.next=y.next,y.next=s),p.pending=s}}u.lanes|=n,s=u.alternate,s!==null&&(s.lanes|=n),Pu(u.return,n,t),o.lanes|=n;break}s=s.next}}else if(u.tag===10)i=u.type===t.type?null:u.child;else if(u.tag===18){if(i=u.return,i===null)throw Error(m(341));i.lanes|=n,o=i.alternate,o!==null&&(o.lanes|=n),Pu(i,n,t),i=u.sibling}else i=u.child;if(i!==null)i.return=u;else for(i=u;i!==null;){if(i===t){i=null;break}if(u=i.sibling,u!==null){u.return=i.return,i=u;break}i=i.return}u=i}we(e,t,l.children,n),t=t.child}return t;case 9:return l=t.type,r=t.pendingProps.children,_n(t,n),l=He(l),r=r(l),t.flags|=1,we(e,t,r,n),t.child;case 14:return r=t.type,l=et(r,t.pendingProps),l=et(r.type,l),Ds(e,t,r,l,n);case 15:return Os(e,t,t.type,t.pendingProps,n);case 17:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:et(r,l),ol(e,t),t.tag=1,_e(r)?(e=!0,Wr(t)):e=!1,_n(t,n),xs(t,r,l),Hu(t,r,l,n),Ku(null,t,r,!0,e,n);case 19:return Hs(e,t,n);case 22:return Is(e,t,n)}throw Error(m(156,t.tag))};function da(e,t){return $i(e,t)}function dc(e,t,n,r){this.tag=e,this.key=n,this.sibling=this.child=this.return=this.stateNode=this.type=this.elementType=null,this.index=0,this.ref=null,this.pendingProps=t,this.dependencies=this.memoizedState=this.updateQueue=this.memoizedProps=null,this.mode=r,this.subtreeFlags=this.flags=0,this.deletions=null,this.childLanes=this.lanes=0,this.alternate=null}function Qe(e,t,n,r){return new dc(e,t,n,r)}function di(e){return e=e.prototype,!(!e||!e.isReactComponent)}function pc(e){if(typeof e=="function")return di(e)?1:0;if(e!=null){if(e=e.$$typeof,e===ut)return 11;if(e===it)return 14}return 2}function Ht(e,t){var n=e.alternate;return n===null?(n=Qe(e.tag,t,e.key,e.mode),n.elementType=e.elementType,n.type=e.type,n.stateNode=e.stateNode,n.alternate=e,e.alternate=n):(n.pendingProps=t,n.type=e.type,n.flags=0,n.subtreeFlags=0,n.deletions=null),n.flags=e.flags&14680064,n.childLanes=e.childLanes,n.lanes=e.lanes,n.child=e.child,n.memoizedProps=e.memoizedProps,n.memoizedState=e.memoizedState,n.updateQueue=e.updateQueue,t=e.dependencies,n.dependencies=t===null?null:{lanes:t.lanes,firstContext:t.firstContext},n.sibling=e.sibling,n.index=e.index,n.ref=e.ref,n}function gl(e,t,n,r,l,u){var i=2;if(r=e,typeof e=="function")di(e)&&(i=1);else if(typeof e=="string")i=5;else e:switch(e){case ke:return rn(n.children,l,u,t);case Ve:i=8,l|=8;break;case _t:return e=Qe(12,n,t,l|2),e.elementType=_t,e.lanes=u,e;case Le:return e=Qe(13,n,t,l),e.elementType=Le,e.lanes=u,e;case Ze:return e=Qe(19,n,t,l),e.elementType=Ze,e.lanes=u,e;case J:return wl(n,l,u,t);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case vt:i=10;break e;case $t:i=9;break e;case ut:i=11;break e;case it:i=14;break e;case Ee:i=16,r=null;break e}throw Error(m(130,e==null?e:typeof e,""))}return t=Qe(i,n,t,l),t.elementType=e,t.type=r,t.lanes=u,t}function rn(e,t,n,r){return e=Qe(7,e,r,t),e.lanes=n,e}function wl(e,t,n,r){return e=Qe(22,e,r,t),e.elementType=J,e.lanes=n,e.stateNode={isHidden:!1},e}function pi(e,t,n){return e=Qe(6,e,null,t),e.lanes=n,e}function mi(e,t,n){return t=Qe(4,e.children!==null?e.children:[],e.key,t),t.lanes=n,t.stateNode={containerInfo:e.containerInfo,pendingChildren:null,implementation:e.implementation},t}function mc(e,t,n,r,l){this.tag=t,this.containerInfo=e,this.finishedWork=this.pingCache=this.current=this.pendingChildren=null,this.timeoutHandle=-1,this.callbackNode=this.pendingContext=this.context=null,this.callbackPriority=0,this.eventTimes=Bl(0),this.expirationTimes=Bl(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=Bl(0),this.identifierPrefix=r,this.onRecoverableError=l,this.mutableSourceEagerHydrationData=null}function vi(e,t,n,r,l,u,i,o,s){return e=new mc(e,t,n,o,s),t===1?(t=1,u===!0&&(t|=8)):t=0,u=Qe(3,null,null,t),e.current=u,u.stateNode=e,u.memoizedState={element:r,isDehydrated:n,cache:null,transitions:null,pendingSuspenseBoundaries:null},zu(u),e}function vc(e,t,n){var r=3"u"||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!="function"))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(R)}catch(b){console.error(b)}}return R(),Si.exports=xc(),Si.exports}export{Nc as a,Pc as g,Ec as r}; diff --git a/frontend/dist/index.html b/frontend/dist/index.html new file mode 100644 index 0000000000..74c81d9bd7 --- /dev/null +++ b/frontend/dist/index.html @@ -0,0 +1,31 @@ + + + + + + + ENTity app + + + + + + +
+ + diff --git a/frontend/dist/tree.png b/frontend/dist/tree.png new file mode 100644 index 0000000000..278908c907 Binary files /dev/null and b/frontend/dist/tree.png differ diff --git a/frontend/dist/vite.svg b/frontend/dist/vite.svg new file mode 100644 index 0000000000..e7b8dfb1b2 --- /dev/null +++ b/frontend/dist/vite.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/frontend/index.html b/frontend/index.html index 664410b5b9..3a994cb8f2 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -2,9 +2,25 @@ - + - Technigo React Vite Boiler Plate + Nanwa +
diff --git a/frontend/package.json b/frontend/package.json index 7b2747e949..87259f321b 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -7,20 +7,50 @@ "dev": "vite", "build": "vite build", "lint": "eslint . --ext js,jsx --report-unused-disable-directives --max-warnings 0", - "preview": "vite preview" + "lint:fix": "eslint . --ext js,jsx --fix", + "format": "prettier --write \"src/**/*.{js,jsx,ts,tsx,json,css,md}\"", + "format:check": "prettier --check \"src/**/*.{js,jsx,ts,tsx,json,css,md}\"", + "preview": "vite preview", + "test": "vitest --config config/vitest.config.js", + "test:ui": "vitest --ui --config config/vitest.config.js" }, "dependencies": { + "autoprefixer": "^10.4.14", + "clsx": "^2.1.1", + "leaflet": "^1.9.4", + "leaflet.markercluster": "^1.5.3", + "papaparse": "^5.5.3", + "postcss": "^8.4.24", "react": "^18.2.0", - "react-dom": "^18.2.0" + "react-datepicker": "^8.4.0", + "react-dom": "^18.2.0", + "react-leaflet": "^4.2.1", + "react-router-dom": "^6.14.0", + "recharts": "^3.1.0", + "styled-components": "^6.1.19", + "tailwind-merge": "^3.3.1", + "tailwindcss": "^3.3.0", + "xlsx": "^0.18.5", + "zustand": "^4.4.0" }, "devDependencies": { + "@testing-library/jest-dom": "^5.16.5", + "@testing-library/react": "^13.4.0", "@types/react": "^18.2.15", "@types/react-dom": "^18.2.7", "@vitejs/plugin-react": "^4.0.3", + "concurrently": "^9.2.0", "eslint": "^8.45.0", "eslint-plugin-react": "^7.32.2", "eslint-plugin-react-hooks": "^4.6.0", "eslint-plugin-react-refresh": "^0.4.3", - "vite": "^6.3.5" + "jsdom": "^22.1.0", + "lint-staged": "^13.2.3", + "prettier": "^3.0.0", + "vite": "^6.3.5", + "vitest": "^0.34.0" + }, + "msw": { + "workerDirectory": "public" } } diff --git a/frontend/public/_redirects b/frontend/public/_redirects new file mode 100644 index 0000000000..50a463356b --- /dev/null +++ b/frontend/public/_redirects @@ -0,0 +1 @@ +/* /index.html 200 \ No newline at end of file diff --git a/frontend/public/tree.png b/frontend/public/tree.png new file mode 100644 index 0000000000..278908c907 Binary files /dev/null and b/frontend/public/tree.png differ diff --git a/frontend/src/App.jsx b/frontend/src/App.jsx index 0a24275e6e..7aec6388b8 100644 --- a/frontend/src/App.jsx +++ b/frontend/src/App.jsx @@ -1,8 +1,92 @@ -export const App = () => { +import { BrowserRouter as Router, Routes, Route } from 'react-router-dom'; +import { Suspense, lazy } from 'react'; +import { ProtectedRoute } from './components/auth/ProtectedRoute'; +import { ToastProvider } from './contexts/ToastContext'; +import { DarkModeProvider } from './contexts/DarkModeContext'; +import { ToastNotifications } from './components/ui/Toast'; +import ErrorBoundary from './components/ui/ErrorBoundary'; +import LoadingSpinner from './components/ui/LoadingSpinner'; +import { KeepAliveProvider } from './components/KeepAliveProvider'; + +// Lazy load components +const LandingPage = lazy(() => import('./pages/LandingPage').then(module => ({ default: module.LandingPage }))); +const LoginPage = lazy(() => import('./pages/LoginPage').then(module => ({ default: module.LoginPage }))); +const RegisterPage = lazy(() => import('./pages/RegisterPage').then(module => ({ default: module.RegisterPage }))); +const OverviewDashboardPage = lazy(() => import('./pages/OverviewDashboardPage').then(module => ({ default: module.OverviewDashboardPage }))); +const FinancialDashboardPage = lazy(() => import('./pages/FinancialDashboardPage').then(module => ({ default: module.FinancialDashboardPage }))); +const EcologicalDashboardPage = lazy(() => import('./pages/EcologicalDashboardPage').then(module => ({ default: module.EcologicalDashboardPage }))); +const MapPage = lazy(() => import('./pages/MapPage').then(module => ({ default: module.MapPage }))); +const ExportPage = lazy(() => import('./pages/ExportPage').then(module => ({ default: module.ExportPage }))); +const NotFoundPage = lazy(() => import('./pages/ErrorPages').then(module => ({ default: module.NotFoundPage }))); +export const App = () => { return ( - <> -

Welcome to Final Project!

- + + + + + + }> + {/* Skip link for keyboard users */} + Skip to content + + + } /> + } /> + } /> + +
+ + } + /> + +
+ + } + /> + +
+ + } + /> + +
+ + } + /> + +
+ + } + /> + } /> +
+
+ +
+
+
+
+
); }; diff --git a/frontend/src/components/KeepAliveProvider.jsx b/frontend/src/components/KeepAliveProvider.jsx new file mode 100644 index 0000000000..2a0a82c2d2 --- /dev/null +++ b/frontend/src/components/KeepAliveProvider.jsx @@ -0,0 +1,10 @@ +import { useKeepAlive } from '../hooks/useKeepAlive'; + +// This component initializes the keep-alive service +export const KeepAliveProvider = ({ children }) => { + useKeepAlive(); // Initialize the keep-alive hook + + return children; +}; + +export default KeepAliveProvider; \ No newline at end of file diff --git a/frontend/src/components/auth/DemoCredentials.jsx b/frontend/src/components/auth/DemoCredentials.jsx new file mode 100644 index 0000000000..74043ae812 --- /dev/null +++ b/frontend/src/components/auth/DemoCredentials.jsx @@ -0,0 +1,10 @@ +export const DemoCredentials = () => { + return ( +
+

Demo credentials:

+
+

Admin: admin@nanwa.com / admin123

+
+
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/auth/LoginForm.jsx b/frontend/src/components/auth/LoginForm.jsx new file mode 100644 index 0000000000..de17393a94 --- /dev/null +++ b/frontend/src/components/auth/LoginForm.jsx @@ -0,0 +1,159 @@ +import { Link } from 'react-router-dom'; +import { useFormValidation } from '../../hooks/useFormValidation'; +import { FormField } from '../ui/FormField'; +import { PasswordInput } from '../ui/PasswordInput'; + +// Login form validation rules +const validateLoginForm = (values) => { + const errors = {}; + + if (!values.email) { + errors.email = 'Email is required'; + } else if (!/\S+@\S+\.\S+/.test(values.email)) { + errors.email = 'Email is invalid'; + } + + if (!values.password) { + errors.password = 'Password is required'; + } else if (values.password.length < 6) { + errors.password = 'Password must be at least 6 characters'; + } + + return errors; +}; + +export const LoginForm = ({ onSubmit, isSubmitting, errors: externalErrors }) => { + const { + values, + errors, + handleChange, + handleBlur, + validateForm, + setErrors + } = useFormValidation( + { email: '', password: '' }, + validateLoginForm + ); + + // Merge external errors (like server errors) with form validation errors + const combinedErrors = { ...errors, ...externalErrors }; + + const handleSubmit = (e) => { + e.preventDefault(); + + if (!validateForm()) { + return; + } + + onSubmit(values); + }; + + return ( +
+ {combinedErrors.general && ( +
+ {combinedErrors.general} +
+ )} + + + + + + + + + +
+ + +
+ + + + + + ); +}; + +// Sub-components for better organization +const RememberMeCheckbox = ({ disabled }) => ( +
+ + +
+); + +const ForgotPasswordLink = () => ( + +); + +const SubmitButton = ({ isSubmitting }) => ( + +); + +const SignUpPrompt = () => ( +
+

+ Don't have an account?{' '} + + Sign up + +

+
+); \ No newline at end of file diff --git a/frontend/src/components/auth/PasswordStrengthIndicator.jsx b/frontend/src/components/auth/PasswordStrengthIndicator.jsx new file mode 100644 index 0000000000..141f67e78a --- /dev/null +++ b/frontend/src/components/auth/PasswordStrengthIndicator.jsx @@ -0,0 +1,59 @@ +import styled from 'styled-components'; + +const PasswordStrengthBar = styled.div` + width: 100%; + height: 4px; + background-color: #e5e7eb; + border-radius: 2px; + overflow: hidden; + margin-top: 0.5rem; +`; + +const PasswordStrengthFill = styled.div` + height: 100%; + background-color: ${props => { + if (props.strength === 'weak') return '#ef4444'; + if (props.strength === 'medium') return '#f59e0b'; + if (props.strength === 'strong') return '#10b981'; + return '#e5e7eb'; + }}; + width: ${props => { + if (props.strength === 'weak') return '33%'; + if (props.strength === 'medium') return '66%'; + if (props.strength === 'strong') return '100%'; + return '0%'; + }}; + transition: all 0.3s ease; +`; + +const getPasswordStrengthText = (strength) => { + switch (strength) { + case 'weak': + return 'Weak password'; + case 'medium': + return 'Medium strength password'; + case 'strong': + return 'Strong password'; + default: + return ''; + } +}; + +export const PasswordStrengthIndicator = ({ strength, show }) => { + if (!show) return null; + + return ( +
+ + + +

+ {getPasswordStrengthText(strength)} +

+
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/auth/ProtectedRoute.jsx b/frontend/src/components/auth/ProtectedRoute.jsx new file mode 100644 index 0000000000..f07bfa2729 --- /dev/null +++ b/frontend/src/components/auth/ProtectedRoute.jsx @@ -0,0 +1,23 @@ +import { Navigate } from 'react-router-dom'; +import { useProtectedRoute } from '../../hooks/useProtectedRoute'; +import LoadingSpinner from '../ui/LoadingSpinner'; + +export const ProtectedRoute = ({ children }) => { + const { isAuthenticated, isLoading } = useProtectedRoute(); + + if (isLoading) { + return ( + + ); + } + + if (!isAuthenticated) { + return ; + } + + return children; +}; \ No newline at end of file diff --git a/frontend/src/components/auth/RegisterForm.jsx b/frontend/src/components/auth/RegisterForm.jsx new file mode 100644 index 0000000000..24ade718b2 --- /dev/null +++ b/frontend/src/components/auth/RegisterForm.jsx @@ -0,0 +1,185 @@ +import { Link } from 'react-router-dom'; +import { FormField } from '../ui/FormField'; +import { PasswordInput } from '../ui/PasswordInput'; +import { PasswordStrengthIndicator } from './PasswordStrengthIndicator'; +import { useRegisterForm } from '../../hooks/useRegisterForm'; + +export const RegisterForm = () => { + const { + formData, + errors, + isSubmitting, + passwordStrength, + handleInputChange, + handleSubmit + } = useRegisterForm(); + + return ( +
+ {errors.general && ( +
+ {errors.general} +
+ )} + + + + + + + + + + + + + + + + 0} + /> + + + + + + + + + + + + + ); +}; + +// Sub-components for better organization +const TermsCheckbox = ({ disabled }) => ( +
+ + +
+); + +const SubmitButton = ({ isSubmitting }) => ( + +); + +const SignInPrompt = () => ( +
+

+ Already have an account?{' '} + + Sign in + +

+
+); diff --git a/frontend/src/components/auth/index.js b/frontend/src/components/auth/index.js new file mode 100644 index 0000000000..66df15a894 --- /dev/null +++ b/frontend/src/components/auth/index.js @@ -0,0 +1,6 @@ +// Auth Components +export { ProtectedRoute } from './ProtectedRoute'; +export { LoginForm } from './LoginForm'; +export { DemoCredentials } from './DemoCredentials'; +export { PasswordStrengthIndicator } from './PasswordStrengthIndicator'; +export { RegisterForm } from './RegisterForm'; \ No newline at end of file diff --git a/frontend/src/components/charts/AverageHeightChart.jsx b/frontend/src/components/charts/AverageHeightChart.jsx new file mode 100644 index 0000000000..0acdffff50 --- /dev/null +++ b/frontend/src/components/charts/AverageHeightChart.jsx @@ -0,0 +1,76 @@ +import { LineChart, Line, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle, ChartTooltip } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useChartConfig } from '../../hooks/useChartConfig'; +import { useHeightGrowthData } from '../../hooks/useChartData'; + +export const AverageHeightChart = ({ filters = {} }) => { + const { data: apiData, loading, error } = useHeightGrowthData(filters); + + // Transform API data to chart format + const data = apiData?.chartData?.map(item => ({ + period: item.period, + height: item.species?.reduce((acc, species) => acc + species.avgHeight, 0) / (item.species?.length || 1) || 0 + })) || []; + const chartConfig = useChartConfig('line'); + + if (loading) { + return ( + + + Average Tree Height Over Time + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Average Tree Height Over Time + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + return ( + + + Average Tree Height Over Time + + + + + + + + `Average Height: ${value.toFixed(1)}m`} />} /> + + + + + ); +}; \ No newline at end of file diff --git a/frontend/src/components/charts/BiodiversityTrendsChart.jsx b/frontend/src/components/charts/BiodiversityTrendsChart.jsx new file mode 100644 index 0000000000..63acefe875 --- /dev/null +++ b/frontend/src/components/charts/BiodiversityTrendsChart.jsx @@ -0,0 +1,484 @@ +import { LineChart, Line, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Legend, ComposedChart, Bar, Area, AreaChart } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import { ChartLoader } from '../ui/ChartLoader'; +import { useState, useEffect } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const BiodiversityTrendsChart = ({ filters = {}, chartType = 'line' }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + useEffect(() => { + const fetchBiodiversityData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats to get biodiversity data + const response = await dashboardAPI.getStats(filters); + const stats = response.data; + + // Extract biodiversity and species data + const biodiversityData = stats?.manager?.biodiversity; + const speciesData = stats?.manager?.speciesDiversity; + const soilData = stats?.manager?.soilHealth; + + if (chartType === 'area') { + // Create stacked area chart for biodiversity components + const currentYear = new Date().getFullYear(); + const areaData = []; + + for (let yearOffset = -6; yearOffset <= 0; yearOffset++) { + const year = currentYear + yearOffset; + const progressFactor = (6 + yearOffset) / 6; // 0 to 1 progression + + // Simulate biodiversity components growth over time + const baseSpecies = speciesData?.uniqueSpecies || 25; + const baseEndangered = speciesData?.endangeredSpecies || 3; + + // Species categories with different growth patterns + const mammals = Math.round((baseSpecies * 0.15) * progressFactor * (0.8 + Math.random() * 0.4)); + const birds = Math.round((baseSpecies * 0.35) * progressFactor * (0.9 + Math.random() * 0.2)); + const insects = Math.round((baseSpecies * 0.30) * progressFactor * (1.0 + Math.random() * 0.3)); + const plants = Math.round((baseSpecies * 0.20) * progressFactor * (0.85 + Math.random() * 0.3)); + + // Conservation status + const endangeredSpecies = Math.max(0, Math.round(baseEndangered * (1.2 - progressFactor * 0.4))); // Decreasing + const vulnerableSpecies = Math.round(baseSpecies * 0.1 * progressFactor); + const stableSpecies = mammals + birds + insects + plants - endangeredSpecies - vulnerableSpecies; + + areaData.push({ + year, + yearLabel: year.toString(), + mammals, + birds, + insects, + plants, + endangered: endangeredSpecies, + vulnerable: vulnerableSpecies, + stable: Math.max(0, stableSpecies), + totalSpecies: mammals + birds + insects + plants, + biodiversityIndex: Math.round((biodiversityData?.averageIndex || 5) * progressFactor * 10) / 10, + endemicSpecies: Math.round((baseSpecies * 0.05) * progressFactor), + newSpeciesDiscovered: Math.floor(Math.random() * 3) + (yearOffset === 0 ? 1 : 0) // Random discoveries + }); + } + + setData(areaData); + } else if (chartType === 'composed') { + // Combined biodiversity metrics over time + const currentYear = new Date().getFullYear(); + const composedData = []; + + for (let yearOffset = -5; yearOffset <= 0; yearOffset++) { + const year = currentYear + yearOffset; + const maturityFactor = (5 + yearOffset) / 5; + + // Biodiversity metrics that change over time + const shannonIndex = Math.round(((biodiversityData?.averageIndex || 2.5) * maturityFactor + (Math.random() - 0.5) * 0.3) * 10) / 10; + const simpsonIndex = Math.round((0.8 * maturityFactor + (Math.random() - 0.5) * 0.1) * 100) / 100; + const speciesRichness = Math.round((speciesData?.uniqueSpecies || 20) * maturityFactor * (0.9 + Math.random() * 0.2)); + const evenness = Math.round((0.7 + maturityFactor * 0.2 + (Math.random() - 0.5) * 0.1) * 100) / 100; + + // Ecological connectivity (how well species can move between habitats) + const connectivity = Math.round((60 + maturityFactor * 30 + Math.random() * 10)); + + // Habitat quality score + const habitatQuality = Math.round((50 + maturityFactor * 40 + (Math.random() - 0.5) * 10)); + + // Invasive species pressure (decreases with better management) + const invasiveSpecies = Math.max(0, Math.round(8 - maturityFactor * 5 + (Math.random() - 0.5) * 2)); + + composedData.push({ + year, + yearLabel: year.toString(), + shannonIndex, + simpsonIndex: Math.round(simpsonIndex * 100), // Convert to percentage for display + speciesRichness, + evenness: Math.round(evenness * 100), // Convert to percentage + connectivity, + habitatQuality, + invasiveSpecies, + overallBiodiversityScore: Math.round((shannonIndex * 10 + speciesRichness / 2 + connectivity + habitatQuality) / 4) + }); + } + + setData(composedData); + } else { + // Line chart for key biodiversity trends + const currentYear = new Date().getFullYear(); + const trendData = []; + + for (let yearOffset = -7; yearOffset <= 0; yearOffset++) { + const year = currentYear + yearOffset; + const timeFactor = (7 + yearOffset) / 7; + + // Key biodiversity indicators + const totalSpecies = Math.round((speciesData?.uniqueSpecies || 25) * timeFactor * (0.9 + Math.random() * 0.2)); + const endangeredSpecies = Math.max(0, Math.round((speciesData?.endangeredSpecies || 5) * (1.3 - timeFactor * 0.5))); + const biodiversityIndex = Math.round((biodiversityData?.averageIndex || 3) * timeFactor * 10 + Math.random() * 5) / 10; + + // Native vs non-native species + const nativeSpecies = Math.round(totalSpecies * (0.85 + timeFactor * 0.1)); + const nonNativeSpecies = totalSpecies - nativeSpecies; + + // Population trends for key species groups + const pollinatorPopulation = Math.round(100 * timeFactor * (0.8 + Math.random() * 0.4)); + const predatorPopulation = Math.round(80 * timeFactor * (0.9 + Math.random() * 0.2)); + const herbivorePopulation = Math.round(120 * timeFactor * (0.85 + Math.random() * 0.3)); + + // Habitat connectivity index (0-100) + const connectivityIndex = Math.round(50 + timeFactor * 40 + (Math.random() - 0.5) * 10); + + // Genetic diversity proxy (average heterozygosity) + const geneticDiversity = Math.round((0.65 + timeFactor * 0.2 + (Math.random() - 0.5) * 0.05) * 100); + + trendData.push({ + year, + yearLabel: year.toString(), + totalSpecies, + endangeredSpecies, + biodiversityIndex, + nativeSpecies, + nonNativeSpecies, + pollinatorPopulation, + predatorPopulation, + herbivorePopulation, + connectivityIndex, + geneticDiversity, + // Derived metrics + nativeRatio: Math.round((nativeSpecies / totalSpecies) * 100), + populationHealthIndex: Math.round((pollinatorPopulation + predatorPopulation + herbivorePopulation) / 3), + conservationEffectiveness: Math.round(((totalSpecies - endangeredSpecies) / totalSpecies) * 100) + }); + } + + setData(trendData); + } + + } catch (err) { + console.error('Error fetching biodiversity trends data:', err); + setError(err.message || 'Failed to load biodiversity data'); + } finally { + setLoading(false); + } + }; + + fetchBiodiversityData(); + }, [filters, chartType]); + + const formatNumber = (value) => { + if (!value) return '0'; + return value.toLocaleString(); + }; + + const formatIndex = (value) => { + if (!value) return '0.0'; + return value.toFixed(1); + }; + + const formatPercentage = (value) => { + if (!value) return '0%'; + return `${Math.round(value)}%`; + }; + + const CustomTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + return ( +
+

{label}

+
+ {payload.map((entry, index) => ( +

+ {entry.name}: { + entry.name.includes('Index') || entry.name.includes('index') + ? formatIndex(entry.value) + : entry.name.includes('%') || entry.name.includes('Ratio') || entry.name.includes('Effectiveness') + ? formatPercentage(entry.value) + : formatNumber(entry.value) + } +

+ ))} +
+
+ ); + } + return null; + }; + + const AreaTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + const total = payload.reduce((sum, entry) => sum + entry.value, 0); + return ( +
+

{label}

+
+

+ Total Species: {formatNumber(total)} +

+ {payload.map((entry, index) => ( +

+ {entry.name}: {formatNumber(entry.value)} ({((entry.value / total) * 100).toFixed(0)}%) +

+ ))} +
+

+ Biodiversity Index: {formatIndex(data.biodiversityIndex)} +

+
+
+
+ ); + } + return null; + }; + + if (loading) { + return ; + } + + if (error) { + return ( + + + Biodiversity Trends + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Biodiversity Trends + +
+

No biodiversity data available

+
+
+ ); + } + + if (chartType === 'area') { + return ( + + + Species Composition Trends +

+ Evolution of species diversity by taxonomic groups +

+
+ + + + + + } /> + + + + + + + + +
+ ); + } + + if (chartType === 'composed') { + return ( + + + Biodiversity Metrics Dashboard +

+ Comprehensive biodiversity health indicators +

+
+ + + + + + + } /> + + + {/* Species richness as bars */} + + + {/* Key biodiversity indices as lines */} + + + + + + + +
+ ); + } + + return ( + + + Biodiversity Trends Over Time +

+ Long-term trends in species diversity and conservation status +

+
+ + + + + + } /> + + + {/* Total species trend */} + + + {/* Native vs non-native */} + + + {/* Endangered species (declining hopefully) */} + + + {/* Biodiversity index */} + + + +
+ ); +}; + +export default BiodiversityTrendsChart; \ No newline at end of file diff --git a/frontend/src/components/charts/CO2AbsorptionChart.jsx b/frontend/src/components/charts/CO2AbsorptionChart.jsx new file mode 100644 index 0000000000..876a5c4af6 --- /dev/null +++ b/frontend/src/components/charts/CO2AbsorptionChart.jsx @@ -0,0 +1,92 @@ +import React, { memo } from 'react'; +import { BarChart, Bar, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle, ChartTooltip } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useChartConfig } from '../../hooks/useChartConfig'; +import { useCO2AbsorptionData } from '../../hooks/useChartData'; + +const CO2AbsorptionChartComponent = ({ filters = {}, dashboardData = null }) => { + // Force yearly grouping for this chart + const yearlyFilters = { ...filters, groupBy: 'year' }; + const { data: apiData, loading, error } = useCO2AbsorptionData( + dashboardData ? {} : yearlyFilters, + { skip: !!dashboardData } + ); + + // Use dashboard data first, then API data + const chartData = dashboardData?.charts?.co2Absorption || apiData; + + // Transform data to chart format - show quarters/periods + const data = chartData?.chartData?.map(item => ({ + year: item.period, // Period from data + co2: item.totalAbsorption || 0 + })) || []; + const chartConfig = useChartConfig('bar'); + + // If using dashboard data, no loading state needed + const isLoading = dashboardData ? false : loading; + const hasError = dashboardData ? false : error; + + if (isLoading) { + return ( + + + CO₂ Absorption by Year + +
+ +
+
+ ); + } + + if (hasError) { + return ( + + + CO₂ Absorption by Year + +
+
+

Error loading chart data

+

{hasError}

+
+
+
+ ); + } + + const totalCO2 = data.reduce((sum, item) => sum + item.co2, 0); + + return ( + + + CO₂ Absorption Over Time + + + + + + + + `CO₂ Absorption: ${value.toFixed(1)} tons`} />} /> + + + + ); +}; + +// Memoize component to prevent unnecessary re-renders +export const CO2AbsorptionChart = memo(CO2AbsorptionChartComponent); \ No newline at end of file diff --git a/frontend/src/components/charts/CarbonCreditRevenueChart.jsx b/frontend/src/components/charts/CarbonCreditRevenueChart.jsx new file mode 100644 index 0000000000..269babe042 --- /dev/null +++ b/frontend/src/components/charts/CarbonCreditRevenueChart.jsx @@ -0,0 +1,242 @@ +import { LineChart, Line, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Legend, BarChart, Bar } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const CarbonCreditRevenueChart = ({ filters = {}, chartType = 'line' }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + useEffect(() => { + const fetchRevenueData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats to get carbon credit data + const response = await dashboardAPI.getStats(filters); + const stats = response.data; + + // Create time-series revenue data + // In a real implementation, this would be historical data from the database + const currentYear = new Date().getFullYear(); + const carbonData = stats?.investor?.carbonCredits; + + // Generate quarterly revenue data based on current carbon credits + const baseRevenue = (carbonData?.totalSold || 0) * (carbonData?.averagePrice || 0); + const mockRevenueData = []; + + for (let year = currentYear - 2; year <= currentYear; year++) { + for (let quarter = 1; quarter <= 4; quarter++) { + // Simulate realistic revenue growth over time + const growthFactor = ((year - (currentYear - 2)) * 4 + quarter) / 12; + const seasonalVariation = Math.sin((quarter - 1) * Math.PI / 2) * 0.2 + 0.8; + const randomVariation = Math.random() * 0.3 + 0.85; + + const revenue = baseRevenue * growthFactor * seasonalVariation * randomVariation; + const creditsTraded = Math.floor((carbonData?.totalSold || 100) * growthFactor * randomVariation / 3); + const avgPrice = revenue / (creditsTraded || 1); + + mockRevenueData.push({ + period: `${year} Q${quarter}`, + year, + quarter, + revenue: Math.round(revenue), + creditsTraded, + averagePrice: Math.round(avgPrice * 100) / 100, + cumulativeRevenue: mockRevenueData.reduce((sum, item) => sum + item.revenue, revenue) + }); + } + } + + setData(mockRevenueData); + } catch (err) { + console.error('Error fetching carbon credit revenue data:', err); + setError(err.message || 'Failed to load revenue data'); + } finally { + setLoading(false); + } + }; + + fetchRevenueData(); + }, [filters]); + + const formatCurrency = (value) => { + if (!value) return '0 SEK'; + return new Intl.NumberFormat('sv-SE', { + style: 'currency', + currency: 'SEK', + minimumFractionDigits: 0, + maximumFractionDigits: 0 + }).format(value); + }; + + const formatLargeCurrency = (value) => { + if (!value) return '0 SEK'; + if (value >= 1000000) { + return `${(value / 1000000).toFixed(1)}M SEK`; + } + if (value >= 1000) { + return `${(value / 1000).toFixed(1)}K SEK`; + } + return formatCurrency(value); + }; + + const CustomTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + return ( +
+

{label}

+
+

+ Revenue: {formatLargeCurrency(data.revenue)} +

+

+ Credits Traded: {data.creditsTraded.toLocaleString()} +

+

+ Avg Price: {formatCurrency(data.averagePrice)}/credit +

+ {chartType === 'line' && data.cumulativeRevenue && ( +

+ Cumulative: {formatLargeCurrency(data.cumulativeRevenue)} +

+ )} +
+
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Carbon Credit Revenue Trends + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Carbon Credit Revenue Trends + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Carbon Credit Revenue Trends + +
+

No revenue data available

+
+
+ ); + } + + if (chartType === 'bar') { + return ( + + + Carbon Credit Revenue by Quarter + + + + + + + } /> + + + + + + ); + } + + return ( + + + Carbon Credit Revenue Trends +

+ Quarterly revenue from carbon credit sales +

+
+ + + + + + } /> + + + + + +
+ ); +}; + +export default CarbonCreditRevenueChart; \ No newline at end of file diff --git a/frontend/src/components/charts/EcologicalBenefitsChart.jsx b/frontend/src/components/charts/EcologicalBenefitsChart.jsx new file mode 100644 index 0000000000..e0f6c7b8b6 --- /dev/null +++ b/frontend/src/components/charts/EcologicalBenefitsChart.jsx @@ -0,0 +1,209 @@ +import { ResponsiveContainer, Legend, XAxis, YAxis, CartesianGrid, Tooltip, BarChart, Bar } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const EcologicalBenefitsChart = ({ filters = {} }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + useEffect(() => { + const fetchEcologicalData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats to get ecological data + const response = await dashboardAPI.getStats(filters); + const stats = response.data; + + // Extract relevant data for ecological benefits + const co2Data = stats?.co2; + const biodiversityData = stats?.manager?.biodiversity; + const speciesData = stats?.manager?.speciesDiversity; + const soilData = stats?.manager?.soilHealth; + + // Bar chart data for ecological benefits comparison + const benefits = [ + { + category: 'Carbon Storage', + current: co2Data?.totalAbsorption || 0, + potential: (co2Data?.totalAbsorption || 0) * 1.5, + unit: 'tons CO₂', + description: 'Annual carbon sequestration capacity' + }, + { + category: 'Species Support', + current: speciesData?.uniqueSpecies || 0, + potential: (speciesData?.uniqueSpecies || 0) * 1.3, + unit: 'species', + description: 'Wildlife species supported' + }, + { + category: 'Water Filtration', + current: Math.round((co2Data?.totalAbsorption || 1000) * 2.5), // Estimated from forest size + potential: Math.round((co2Data?.totalAbsorption || 1000) * 3.5), + unit: 'liters/day', + description: 'Daily water filtration capacity' + }, + { + category: 'Oxygen Production', + current: Math.round((co2Data?.totalAbsorption || 1000) * 0.7), // CO2 to O2 conversion + potential: Math.round((co2Data?.totalAbsorption || 1000) * 1.0), + unit: 'tons O₂/year', + description: 'Annual oxygen production' + }, + { + category: 'Soil Protection', + current: Math.round((soilData?.averagePH || 6) * 15), // pH score scaled + potential: Math.round((soilData?.averagePH || 6) * 20), + unit: 'erosion prevented (tons)', + description: 'Annual soil erosion prevention' + }, + { + category: 'Temperature Regulation', + current: Math.round(70 + Math.random() * 20), // Mock temperature regulation + potential: Math.round(85 + Math.random() * 15), + unit: '% cooling effect', + description: 'Microclimate temperature regulation' + } + ]; + + setData(benefits); + + } catch (err) { + console.error('Error fetching ecological benefits data:', err); + setError(err.message || 'Failed to load ecological data'); + } finally { + setLoading(false); + } + }; + + fetchEcologicalData(); + }, [filters]); + + const formatValue = (value, unit) => { + if (!value) return '0'; + if (value >= 1000000) { + return `${(value / 1000000).toFixed(1)}M ${unit}`; + } + if (value >= 1000) { + return `${(value / 1000).toFixed(1)}K ${unit}`; + } + return `${value} ${unit}`; + }; + + const BarTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + return ( +
+

{data.category}

+
+

+ Current: {formatValue(data.current, data.unit)} +

+

+ Potential: {formatValue(data.potential, data.unit)} +

+

+ {data.description} +

+
+
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Ecological Benefits + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Ecological Benefits + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Ecological Benefits + +
+

No ecological data available

+
+
+ ); + } + + return ( + + + Current vs Potential Ecological Benefits +

+ Comparison of current performance with ecological potential +

+
+ + + + + + } /> + + + + + + +
+ ); +}; + +export default EcologicalBenefitsChart; \ No newline at end of file diff --git a/frontend/src/components/charts/EnhancedSurvivalRateChart.jsx b/frontend/src/components/charts/EnhancedSurvivalRateChart.jsx new file mode 100644 index 0000000000..8ffda61cce --- /dev/null +++ b/frontend/src/components/charts/EnhancedSurvivalRateChart.jsx @@ -0,0 +1,326 @@ +import React, { useEffect, useState } from 'react'; +import { + ComposedChart, + Area, + Line, + XAxis, + YAxis, + CartesianGrid, + Tooltip, + Legend, + ResponsiveContainer, + ReferenceLine, + Dot +} from 'recharts'; +import styled from 'styled-components'; +import { TrendingUp, TrendingDown, Minus } from 'lucide-react'; +import { ChartLoader } from '../ui/ChartLoader'; + +const ChartContainer = styled.div` + background: white; + border-radius: 8px; + padding: 20px; + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1); +`; + +const ChartHeader = styled.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 20px; +`; + +const ChartTitle = styled.h3` + font-size: 18px; + font-weight: 600; + color: #1f2937; +`; + +const MetricsRow = styled.div` + display: flex; + gap: 16px; + margin-bottom: 16px; + padding: 12px; + background: #f9fafb; + border-radius: 6px; +`; + +const MetricCard = styled.div` + flex: 1; + text-align: center; +`; + +const MetricValue = styled.div` + font-size: 24px; + font-weight: 700; + color: ${props => props.$positive ? '#10b981' : props.$negative ? '#ef4444' : '#1f2937'}; +`; + +const MetricLabel = styled.div` + font-size: 12px; + color: #6b7280; + margin-top: 4px; +`; + +const TrendIndicator = styled.div` + display: flex; + align-items: center; + gap: 4px; + padding: 4px 8px; + border-radius: 4px; + font-size: 12px; + font-weight: 500; + background: ${props => props.$trend === 'up' ? '#d1fae5' : + props.$trend === 'down' ? '#fee2e2' : + '#f3f4f6'}; + color: ${props => props.$trend === 'up' ? '#065f46' : + props.$trend === 'down' ? '#991b1b' : + '#6b7280'}; +`; + +const CustomTooltip = ({ active, payload, label }) => { + if (!active || !payload?.length) return null; + + const data = payload[0]?.payload; + if (!data) return null; + + return ( +
+

{label}

+

+ Survival Rate: {data.survivalRate?.toFixed(1)}% +

+

+ Trees Planted: {data.totalPlanted} +

+

+ Trees Surviving: {data.surviving} +

+ {data.benchmark && ( +

+ Benchmark: {data.benchmark}% +

+ )} +
+ ); +}; + +const EnhancedSurvivalRateChart = ({ filters = {} }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + const [metrics, setMetrics] = useState({ + currentRate: 0, + trend: 'stable', + trendValue: 0, + avgRate: 0, + benchmark: 85 + }); + + useEffect(() => { + const fetchSurvivalData = async () => { + try { + setLoading(true); + setError(null); + + const params = new URLSearchParams({ + groupBy: 'month' + }); + + if (filters?.forestId) params.append('forestId', filters.forestId); + if (filters?.dateRange?.startDate) params.append('startDate', filters.dateRange.startDate); + if (filters?.dateRange?.endDate) params.append('endDate', filters.dateRange.endDate); + + const response = await fetch( + `${import.meta.env.VITE_API_URL}/charts/survival-rate?${params}`, + { + headers: { + 'Authorization': `Bearer ${localStorage.getItem('authToken')}`, + 'Content-Type': 'application/json' + } + } + ); + + if (!response.ok) { + throw new Error('Failed to fetch survival rate data'); + } + + const result = await response.json(); + + if (result.success && result.data?.chartData) { + const chartData = result.data.chartData; + + // Add benchmark to each data point + const processedData = chartData.map(item => ({ + ...item, + benchmark: 85 // Industry standard benchmark + })); + + setData(processedData); + + // Calculate metrics + if (processedData.length > 0) { + const currentRate = processedData[processedData.length - 1].survivalRate; + const previousRate = processedData.length > 1 ? + processedData[processedData.length - 2].survivalRate : currentRate; + const avgRate = processedData.reduce((sum, item) => sum + item.survivalRate, 0) / processedData.length; + + const trendValue = currentRate - previousRate; + let trend = 'stable'; + if (trendValue > 0.5) trend = 'up'; + else if (trendValue < -0.5) trend = 'down'; + + setMetrics({ + currentRate, + trend, + trendValue, + avgRate, + benchmark: 85 + }); + } + } + } catch (err) { + console.error('Error fetching survival rate data:', err); + setError(err.message); + } finally { + setLoading(false); + } + }; + + fetchSurvivalData(); + }, [filters]); + + const getTrendIcon = (trend) => { + if (trend === 'up') return ; + if (trend === 'down') return ; + return ; + }; + + if (loading) { + return ; + } + + if (error) { + return ( +
+

+ Tree Survival Rate +

+
+

Error: {error}

+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Tree Survival Rate Analysis + +
+

No survival data available for the selected period

+
+
+ ); + } + + return ( + + + Tree Survival Rate Analysis + + {getTrendIcon(metrics.trend)} + {Math.abs(metrics.trendValue).toFixed(1)}% + + + + + + = metrics.benchmark}> + {metrics.currentRate.toFixed(1)}% + + Current Rate + + + {metrics.avgRate.toFixed(1)}% + Average Rate + + + {metrics.benchmark}% + Industry Benchmark + + + + + + + + + } /> + + + + + + + + + + +
= metrics.benchmark ? '#d1fae5' : '#fef3c7', + borderRadius: '6px', + fontSize: '14px', + color: metrics.currentRate >= metrics.benchmark ? '#065f46' : '#92400e' + }}> + {metrics.currentRate >= metrics.benchmark ? ( + ✓ Survival rate is above industry benchmark. Forest health is excellent. + ) : ( + ⚠ Survival rate is below industry benchmark. Consider investigating potential issues. + )} +
+
+ ); +}; + +export default EnhancedSurvivalRateChart; \ No newline at end of file diff --git a/frontend/src/components/charts/EnvironmentalRiskMatrixChart.jsx b/frontend/src/components/charts/EnvironmentalRiskMatrixChart.jsx new file mode 100644 index 0000000000..c65a1b935f --- /dev/null +++ b/frontend/src/components/charts/EnvironmentalRiskMatrixChart.jsx @@ -0,0 +1,279 @@ +import { ScatterChart, Scatter, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Cell, BarChart, Bar } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect, useMemo } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const EnvironmentalRiskMatrixChart = ({ filters = {}, chartType = 'scatter' }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + // Risk level color mapping + const riskColors = useMemo(() => ({ + low: '#10b981', // green-500 + moderate: '#f59e0b', // amber-500 + high: '#ef4444', // red-500 + extreme: '#dc2626' // red-600 + }), []); + + // Risk level numeric mapping for scatter plot + const riskLevels = useMemo(() => ({ + low: 1, + moderate: 2, + high: 3, + extreme: 4 + }), []); + + useEffect(() => { + const fetchRiskData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats which includes fire risk data + const response = await dashboardAPI.getStats(filters); + const fireRiskData = response.data?.manager?.fireRisk?.distribution || []; + + if (chartType === 'scatter') { + // Transform data for scatter plot (risk level vs area affected) + const scatterData = fireRiskData + .filter(item => item.riskLevel && item.totalArea > 0) + .map((item) => ({ + x: riskLevels[item.riskLevel?.toLowerCase()] || 0, + y: item.totalArea, + riskLevel: item.riskLevel, + forestCount: item.forestCount, + fill: riskColors[item.riskLevel?.toLowerCase()] || '#6b7280', + // Vary bubble size based on forest count + r: Math.max(8, Math.min(20, item.forestCount * 3)) + })); + + setData(scatterData); + } else { + // Transform data for bar chart + const barData = fireRiskData + .filter(item => item.riskLevel) + .map(item => ({ + riskLevel: item.riskLevel, + forestCount: item.forestCount || 0, + totalArea: item.totalArea || 0, + fill: riskColors[item.riskLevel?.toLowerCase()] || '#6b7280' + })); + + // Sort by risk level + const riskOrder = ['low', 'moderate', 'high', 'extreme']; + barData.sort((a, b) => { + const aIndex = riskOrder.indexOf(a.riskLevel?.toLowerCase()); + const bIndex = riskOrder.indexOf(b.riskLevel?.toLowerCase()); + return aIndex - bIndex; + }); + + setData(barData); + } + } catch (err) { + console.error('Error fetching environmental risk data:', err); + setError(err.message || 'Failed to load environmental risk data'); + } finally { + setLoading(false); + } + }; + + fetchRiskData(); + }, [filters, chartType, riskColors, riskLevels]); + + const formatRiskLevel = (level) => { + if (typeof level === 'number') { + const levelNames = { 1: 'Low', 2: 'Moderate', 3: 'High', 4: 'Extreme' }; + return levelNames[level] || 'Unknown'; + } + return level ? level.charAt(0).toUpperCase() + level.slice(1) : 'Unknown'; + }; + + const formatArea = (area) => { + return `${area.toFixed(1)} ha`; + }; + + const ScatterTooltip = ({ active, payload }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + return ( +
+

+ {`Risk Level: ${formatRiskLevel(data.riskLevel)}`} +

+

+ {`Area: ${formatArea(data.y)}`} +

+

+ {`Forests: ${data.forestCount}`} +

+
+ ); + } + return null; + }; + + const BarTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0]; + return ( +
+

+ {`Risk Level: ${formatRiskLevel(label)}`} +

+

+ {`Forests: ${data.payload.forestCount}`} +

+

+ {`Total Area: ${formatArea(data.payload.totalArea)}`} +

+
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Environmental Risk Matrix + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Environmental Risk Matrix + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Environmental Risk Matrix + +
+

No environmental risk data available

+
+
+ ); + } + + if (chartType === 'scatter') { + return ( + + + Environmental Risk Matrix +

+ Risk Level vs Affected Area (bubble size = forest count) +

+
+ + + + + + } /> + + {data.map((entry, index) => ( + + ))} + + + + + {/* Legend */} +
+ {Object.entries(riskColors).map(([level, color]) => ( +
+
+ + {level} + +
+ ))} +
+
+ ); + } + + return ( + + + Fire Risk Distribution + + + + + + + } /> + + {data.map((entry, index) => ( + + ))} + + + + + {/* Legend */} +
+ {data.map((item, index) => ( +
+
+ + {formatRiskLevel(item.riskLevel)} ({item.forestCount}) + +
+ ))} +
+
+ ); +}; + +export default EnvironmentalRiskMatrixChart; \ No newline at end of file diff --git a/frontend/src/components/charts/ForestValueAppreciationChart.jsx b/frontend/src/components/charts/ForestValueAppreciationChart.jsx new file mode 100644 index 0000000000..8b8a0ddee4 --- /dev/null +++ b/frontend/src/components/charts/ForestValueAppreciationChart.jsx @@ -0,0 +1,194 @@ +import { LineChart, Line, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Legend } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const ForestValueAppreciationChart = ({ filters = {}, dashboardData = null }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(!dashboardData); + const [error, setError] = useState(null); + + useEffect(() => { + // If dashboard data is provided, use it directly + if (dashboardData?.charts?.forestValue) { + setData(dashboardData.charts.forestValue.chartData); + setLoading(false); + return; + } + + const fetchValueData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats which now includes investor metrics + const response = await dashboardAPI.getStats(filters); + const stats = response.data; + + // Create mock time series data for demonstration + // In a real implementation, you'd have historical value tracking + const mockData = [ + { + period: '2020', + currentValue: (stats?.investor?.portfolio?.totalCurrentValue || 0) * 0.7, + acquisitionCost: stats?.investor?.portfolio?.totalAcquisitionCost || 0 + }, + { + period: '2021', + currentValue: (stats?.investor?.portfolio?.totalCurrentValue || 0) * 0.8, + acquisitionCost: stats?.investor?.portfolio?.totalAcquisitionCost || 0 + }, + { + period: '2022', + currentValue: (stats?.investor?.portfolio?.totalCurrentValue || 0) * 0.9, + acquisitionCost: stats?.investor?.portfolio?.totalAcquisitionCost || 0 + }, + { + period: '2023', + currentValue: (stats?.investor?.portfolio?.totalCurrentValue || 0) * 0.95, + acquisitionCost: stats?.investor?.portfolio?.totalAcquisitionCost || 0 + }, + { + period: '2024', + currentValue: stats?.investor?.portfolio?.totalCurrentValue || 0, + acquisitionCost: stats?.investor?.portfolio?.totalAcquisitionCost || 0 + } + ]; + + setData(mockData); + } catch (err) { + console.error('Error fetching forest value data:', err); + setError(err.message || 'Failed to load forest value data'); + } finally { + setLoading(false); + } + }; + + if (!dashboardData) { + fetchValueData(); + } + }, [filters, dashboardData]); + + const formatCurrency = (value) => { + if (!value) return '0'; + return new Intl.NumberFormat('sv-SE', { + style: 'currency', + currency: 'SEK', + minimumFractionDigits: 0, + maximumFractionDigits: 0 + }).format(value); + }; + + const CustomTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + return ( +
+

{`Year: ${label}`}

+ {payload.map((entry, index) => ( +

+ {`${entry.dataKey === 'currentValue' ? 'Current Value' : 'Acquisition Cost'}: ${formatCurrency(entry.value)}`} +

+ ))} + {payload.length === 2 && ( +

+ Appreciation: {formatCurrency(payload[0].value - payload[1].value)} +

+ )} +
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Forest Value Appreciation + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Forest Value Appreciation + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Forest Value Appreciation + +
+

No forest value data available

+
+
+ ); + } + + const currentValue = data[data.length - 1]?.currentValue || 0; + const initialValue = data[0]?.acquisitionCost || 0; + const appreciation = currentValue - initialValue; + + return ( + + + Forest Value Appreciation + + + + + + + } /> + + + + + ); +}; + +export default ForestValueAppreciationChart; \ No newline at end of file diff --git a/frontend/src/components/charts/GrowthPerformancePredictionsChart.jsx b/frontend/src/components/charts/GrowthPerformancePredictionsChart.jsx new file mode 100644 index 0000000000..812c10181e --- /dev/null +++ b/frontend/src/components/charts/GrowthPerformancePredictionsChart.jsx @@ -0,0 +1,350 @@ +import { LineChart, Line, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Legend, ComposedChart, Area, AreaChart } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const GrowthPerformancePredictionsChart = ({ filters = {}, chartType = 'line' }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + useEffect(() => { + const fetchGrowthData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats to get current growth data + const response = await dashboardAPI.getStats(filters); + const stats = response.data; + + // Generate growth vs prediction data + const heightData = stats?.height; + const currentAvgHeight = heightData?.average || 15; // Default 15m + const currentYear = new Date().getFullYear(); + + // Generate historical and future data (5 years past, 5 years future) + const growthData = []; + + for (let yearOffset = -5; yearOffset <= 5; yearOffset++) { + const year = currentYear + yearOffset; + const isHistorical = yearOffset <= 0; + const isCurrent = yearOffset === 0; + + // Base growth models for different scenarios + const optimalGrowthRate = 0.8; // 0.8m per year under optimal conditions + const averageGrowthRate = 0.6; // 0.6m per year under average conditions + const poorGrowthRate = 0.4; // 0.4m per year under poor conditions + + // Historical actual data (with some variation from prediction) + let actualHeight = null; + let predictedHeight = null; + + if (isHistorical || isCurrent) { + // Historical data - actual measurements with seasonal variation + const baseHeight = currentAvgHeight + (yearOffset * averageGrowthRate); + const seasonalVariation = Math.sin(yearOffset * Math.PI / 3) * 0.2; + const randomVariation = (Math.random() - 0.5) * 0.4; + actualHeight = Math.max(0, baseHeight + seasonalVariation + randomVariation); + + // What was predicted at that time (slightly different from actual) + const predictionError = (Math.random() - 0.5) * 0.3; + predictedHeight = baseHeight + predictionError; + } else { + // Future predictions only + predictedHeight = currentAvgHeight + (yearOffset * averageGrowthRate); + } + + // Confidence intervals for predictions + const confidenceRange = Math.abs(yearOffset) * 0.1; // Uncertainty increases with time + const upperBound = predictedHeight + confidenceRange; + const lowerBound = Math.max(0, predictedHeight - confidenceRange); + + // Environmental factors affecting growth + const environmentalFactors = { + temperature: 18 + Math.sin(yearOffset * 0.5) * 2, // Temperature variation + rainfall: 800 + Math.cos(yearOffset * 0.3) * 100, // Rainfall variation + soilQuality: 7.5 + (Math.random() - 0.5) * 0.5, // pH variation + co2Level: 420 + yearOffset * 2 // Increasing CO2 levels + }; + + // Growth rate based on environmental factors + const temperatureOptimal = Math.max(0, 1 - Math.abs(environmentalFactors.temperature - 18) * 0.05); + const rainfallOptimal = Math.max(0, 1 - Math.abs(environmentalFactors.rainfall - 800) * 0.001); + const soilOptimal = Math.max(0, environmentalFactors.soilQuality / 10); + + const environmentalMultiplier = (temperatureOptimal + rainfallOptimal + soilOptimal) / 3; + const adjustedGrowthRate = averageGrowthRate * environmentalMultiplier; + + growthData.push({ + year, + yearLabel: year.toString(), + actualHeight: actualHeight ? Math.round(actualHeight * 10) / 10 : null, + predictedHeight: Math.round(predictedHeight * 10) / 10, + upperBound: Math.round(upperBound * 10) / 10, + lowerBound: Math.round(lowerBound * 10) / 10, + growthRate: Math.round(adjustedGrowthRate * 100) / 100, + isHistorical, + isFuture: yearOffset > 0, + isCurrent, + environmentalScore: Math.round(environmentalMultiplier * 100), + temperature: Math.round(environmentalFactors.temperature * 10) / 10, + rainfall: Math.round(environmentalFactors.rainfall), + soilQuality: Math.round(environmentalFactors.soilQuality * 10) / 10, + // Calculate accuracy for historical data + accuracy: actualHeight && predictedHeight ? + Math.round((1 - Math.abs(actualHeight - predictedHeight) / actualHeight) * 100) : null + }); + } + + setData(growthData); + } catch (err) { + console.error('Error fetching growth prediction data:', err); + setError(err.message || 'Failed to load growth data'); + } finally { + setLoading(false); + } + }; + + fetchGrowthData(); + }, [filters, chartType]); + + const formatHeight = (value) => { + if (!value) return '0m'; + return `${value}m`; + }; + + const formatGrowthRate = (value) => { + if (!value) return '0m/yr'; + return `${value}m/yr`; + }; + + const CustomTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + return ( +
+

{label}

+
+ {data.actualHeight && ( +

+ Actual Height: {formatHeight(data.actualHeight)} +

+ )} +

+ Predicted Height: {formatHeight(data.predictedHeight)} +

+ {data.upperBound && data.lowerBound && ( +

+ Range: {formatHeight(data.lowerBound)} - {formatHeight(data.upperBound)} +

+ )} +

+ Growth Rate: {formatGrowthRate(data.growthRate)} +

+ {data.accuracy && ( +

+ Prediction Accuracy: {data.accuracy}% +

+ )} +

+ Environmental Score: {data.environmentalScore}% +

+
+

Temp: {data.temperature}°C | Rainfall: {data.rainfall}mm

+

Soil pH: {data.soilQuality}

+
+
+
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Growth Performance vs Predictions + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Growth Performance vs Predictions + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Growth Performance vs Predictions + +
+

No growth data available

+
+
+ ); + } + + if (chartType === 'area') { + return ( + + + Growth Prediction Confidence Range +

+ Predicted height range with confidence intervals +

+
+ + + + + + } /> + + + {/* Confidence range area */} + + + + {/* Predicted height line */} + + + {/* Actual height line (only for historical data) */} + + + +
+ ); + } + + return ( + + + Growth Performance vs Predictions +

+ Historical actual vs predicted growth with future forecasts +

+
+ + + + + + } /> + + + {/* Actual height line (blue, solid) */} + + + {/* Predicted height line (green, dashed for future) */} + + + {/* Upper confidence bound (light) */} + + + {/* Lower confidence bound (light) */} + + + +
+ ); +}; + +export default GrowthPerformancePredictionsChart; \ No newline at end of file diff --git a/frontend/src/components/charts/HealthStatusDistributionChart.jsx b/frontend/src/components/charts/HealthStatusDistributionChart.jsx new file mode 100644 index 0000000000..cdded150d7 --- /dev/null +++ b/frontend/src/components/charts/HealthStatusDistributionChart.jsx @@ -0,0 +1,221 @@ +import { BarChart, Bar, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, PieChart, Pie, Cell } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect, useMemo } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const HealthStatusDistributionChart = ({ filters = {}, chartType = 'bar' }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + // Health status color mapping + const healthColors = useMemo(() => ({ + excellent: '#10b981', // green-500 + good: '#3b82f6', // blue-500 + fair: '#f59e0b', // amber-500 + poor: '#ef4444', // red-500 + critical: '#dc2626' // red-600 + }), []); + + useEffect(() => { + const fetchHealthData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats which includes health distribution + const response = await dashboardAPI.getStats(filters); + const healthDistribution = response.data?.distributions?.health || []; + + // Transform the data for chart display + const transformedData = healthDistribution.map(item => ({ + status: item._id || 'Unknown', + count: item.count || 0, + fill: healthColors[item._id?.toLowerCase()] || '#6b7280' + })); + + // Sort by status priority (excellent -> critical) + const statusOrder = ['excellent', 'good', 'fair', 'poor', 'critical']; + transformedData.sort((a, b) => { + const aIndex = statusOrder.indexOf(a.status.toLowerCase()); + const bIndex = statusOrder.indexOf(b.status.toLowerCase()); + return aIndex - bIndex; + }); + + setData(transformedData); + } catch (err) { + console.error('Error fetching health distribution data:', err); + setError(err.message || 'Failed to load health distribution data'); + } finally { + setLoading(false); + } + }; + + fetchHealthData(); + }, [filters, healthColors]); + + const formatStatus = (status) => { + return status.charAt(0).toUpperCase() + status.slice(1); + }; + + const CustomTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0]; + return ( +
+

+ {`${formatStatus(label || data.payload.status)}: ${data.value} trees`} +

+

+ {`${((data.value / data.payload.total) * 100).toFixed(1)}%`} +

+
+ ); + } + return null; + }; + + const PieTooltip = ({ active, payload }) => { + if (active && payload && payload.length) { + const data = payload[0]; + const totalTrees = data.payload.totalTrees || + data.payload.data?.reduce((sum, item) => sum + item.count, 0) || 0; + + return ( +
+

+ {`${formatStatus(data.payload.status)}: ${data.value} trees`} +

+

+ {totalTrees > 0 ? `${((data.value / totalTrees) * 100).toFixed(1)}%` : '0%'} +

+
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Tree Health Status Distribution + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Tree Health Status Distribution + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Tree Health Status Distribution + +
+

No health status data available

+
+
+ ); + } + + // Calculate total for percentage calculations + const totalTrees = data.reduce((sum, item) => sum + item.count, 0); + const dataWithTotal = data.map(item => ({ ...item, total: totalTrees })); + + if (chartType === 'pie') { + return ( + + + Tree Health Status Distribution + + + + + `${formatStatus(status)}: ${(percent * 100).toFixed(0)}%` + } + outerRadius={80} + fill="#8884d8" + dataKey="count" + > + {dataWithTotal.map((entry, index) => ( + + ))} + + } /> + + + + ); + } + + return ( + + + Tree Health Status Distribution + + + + + + + } /> + + {dataWithTotal.map((entry, index) => ( + + ))} + + + + + {/* Legend */} +
+ {dataWithTotal.map((item, index) => ( +
+
+ + {formatStatus(item.status)} ({item.count}) + +
+ ))} +
+
+ ); +}; + +export default HealthStatusDistributionChart; \ No newline at end of file diff --git a/frontend/src/components/charts/MaintenanceCostAnalysisChart.jsx b/frontend/src/components/charts/MaintenanceCostAnalysisChart.jsx new file mode 100644 index 0000000000..419b1e2e52 --- /dev/null +++ b/frontend/src/components/charts/MaintenanceCostAnalysisChart.jsx @@ -0,0 +1,309 @@ +import { ComposedChart, Bar, Line, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Legend, PieChart, Pie, Cell } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const MaintenanceCostAnalysisChart = ({ filters = {}, chartType = 'composed' }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + useEffect(() => { + const fetchMaintenanceData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats to get maintenance budget data + const response = await dashboardAPI.getStats(filters); + const stats = response.data; + + const maintenanceData = stats?.investor?.maintenance; + + if (chartType === 'pie') { + // Create budget breakdown pie chart data + const totalBudget = maintenanceData?.totalBudget || 100000; + const totalSpent = maintenanceData?.totalSpent || 0; + const remaining = totalBudget - totalSpent; + + const pieData = [ + { + name: 'Spent', + value: totalSpent, + fill: '#ef4444' + }, + { + name: 'Remaining', + value: Math.max(0, remaining), + fill: '#10b981' + } + ]; + + if (remaining < 0) { + pieData.push({ + name: 'Over Budget', + value: Math.abs(remaining), + fill: '#dc2626' + }); + } + + setData(pieData); + } else { + // Create time-series maintenance data by forest/category + const mockMaintenanceData = []; + const categories = ['Equipment', 'Labor', 'Materials', 'Infrastructure', 'Emergency']; + const colors = ['#3b82f6', '#10b981', '#f59e0b', '#8b5cf6', '#ef4444']; + + // Generate monthly data for the last 12 months + const currentDate = new Date(); + for (let i = 11; i >= 0; i--) { + const date = new Date(currentDate); + date.setMonth(date.getMonth() - i); + const monthName = date.toLocaleString('default', { month: 'short', year: '2-digit' }); + + const monthlyBudget = (maintenanceData?.totalBudget || 100000) / 12; + const seasonalFactor = Math.sin((date.getMonth() + 3) * Math.PI / 6) * 0.3 + 1; // Higher in spring/summer + + let totalSpent = 0; + const monthData = { + period: monthName, + budget: Math.round(monthlyBudget), + utilization: 0 + }; + + categories.forEach((category, index) => { + const baseAmount = monthlyBudget * (0.1 + Math.random() * 0.3); // 10-40% of monthly budget per category + const categorySpent = Math.round(baseAmount * seasonalFactor * (0.7 + Math.random() * 0.6)); + monthData[category.toLowerCase()] = categorySpent; + totalSpent += categorySpent; + }); + + monthData.totalSpent = totalSpent; + monthData.utilization = Math.round((totalSpent / monthlyBudget) * 100); + monthData.efficiency = Math.round((Math.random() * 20 + 80)); // 80-100% efficiency + + mockMaintenanceData.push(monthData); + } + + setData(mockMaintenanceData); + } + + } catch (err) { + console.error('Error fetching maintenance cost data:', err); + setError(err.message || 'Failed to load maintenance data'); + } finally { + setLoading(false); + } + }; + + fetchMaintenanceData(); + }, [filters, chartType]); + + const formatCurrency = (value) => { + if (!value) return '0'; + return new Intl.NumberFormat('sv-SE', { + style: 'currency', + currency: 'SEK', + minimumFractionDigits: 0, + maximumFractionDigits: 0 + }).format(value); + }; + + const formatPercentage = (value) => { + return `${value}%`; + }; + + const CustomTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + return ( +
+

{label}

+
+ {payload.map((entry, index) => ( +

+ {entry.name}: { + entry.name.includes('%') || entry.name === 'utilization' || entry.name === 'efficiency' + ? formatPercentage(entry.value) + : formatCurrency(entry.value) + } +

+ ))} +
+
+ ); + } + return null; + }; + + const PieTooltip = ({ active, payload }) => { + if (active && payload && payload.length) { + const data = payload[0]; + const total = payload[0].payload.totalValue || data.payload.value * 2; // Estimate total + return ( +
+

+ {data.payload.name}: {formatCurrency(data.value)} +

+

+ {((data.value / total) * 100).toFixed(1)}% of budget +

+
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Maintenance Cost Analysis + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Maintenance Cost Analysis + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Maintenance Cost Analysis + +
+

No maintenance data available

+
+
+ ); + } + + if (chartType === 'pie') { + const totalBudget = data.reduce((sum, item) => sum + item.value, 0); + + return ( + + + Budget Utilization +

+ Total Budget: {formatCurrency(totalBudget)} +

+
+ + + `${name}: ${(percent * 100).toFixed(0)}%`} + outerRadius={80} + fill="#8884d8" + dataKey="value" + > + {data.map((entry, index) => ( + + ))} + + } /> + + +
+ ); + } + + return ( + + + Maintenance Costs & Budget Utilization +

+ Monthly spend vs budget with utilization trends +

+
+ + + + + + + } /> + + + {/* Budget line */} + + + {/* Actual spending */} + + + {/* Utilization percentage line */} + + + {/* Efficiency line */} + + + +
+ ); +}; + +export default MaintenanceCostAnalysisChart; \ No newline at end of file diff --git a/frontend/src/components/charts/SpeciesEconomicPerformanceChart.jsx b/frontend/src/components/charts/SpeciesEconomicPerformanceChart.jsx new file mode 100644 index 0000000000..a20ab639e6 --- /dev/null +++ b/frontend/src/components/charts/SpeciesEconomicPerformanceChart.jsx @@ -0,0 +1,313 @@ +import { BarChart, Bar, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Legend, ScatterChart, Scatter, Cell } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useState, useEffect } from 'react'; +import { dashboardAPI } from '../../lib/api'; + +export const SpeciesEconomicPerformanceChart = ({ filters = {}, chartType = 'bar' }) => { + const [data, setData] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + useEffect(() => { + const fetchSpeciesData = async () => { + try { + setLoading(true); + setError(null); + + // Fetch dashboard stats to get timber and carbon credit data + const response = await dashboardAPI.getStats(filters); + const stats = response.data; + + // Create species economic performance data + const timberData = stats?.investor?.timber; + const carbonData = stats?.investor?.carbonCredits; + + // Generate species-based economic performance + const species = [ + 'Pine', 'Oak', 'Birch', 'Spruce', 'Maple', + 'Beech', 'Fir', 'Cedar', 'Willow', 'Ash' + ]; + + const performanceData = species.map((speciesName, index) => { + // Base values from actual data + const baseTimberValue = (timberData?.averageValuePerTree || 5000) * (0.8 + Math.random() * 0.4); + const baseCarbonValue = (carbonData?.averagePrice || 500) * (10 + Math.random() * 20); // 10-30 credits per tree + + // Species-specific multipliers based on real forestry economics + const multipliers = { + 'Pine': { timber: 1.0, carbon: 0.9, growth: 1.2 }, + 'Oak': { timber: 1.4, carbon: 1.3, growth: 0.8 }, + 'Birch': { timber: 0.8, carbon: 1.0, growth: 1.5 }, + 'Spruce': { timber: 1.1, carbon: 1.1, growth: 1.0 }, + 'Maple': { timber: 1.3, carbon: 1.2, growth: 0.9 }, + 'Beech': { timber: 1.2, carbon: 1.1, growth: 0.85 }, + 'Fir': { timber: 1.0, carbon: 1.0, growth: 1.0 }, + 'Cedar': { timber: 1.5, carbon: 1.4, growth: 0.7 }, + 'Willow': { timber: 0.6, carbon: 0.8, growth: 1.8 }, + 'Ash': { timber: 1.25, carbon: 1.15, growth: 0.95 } + }; + + const speciesMultiplier = multipliers[speciesName] || { timber: 1.0, carbon: 1.0, growth: 1.0 }; + + const timberValue = Math.round(baseTimberValue * speciesMultiplier.timber); + const carbonValue = Math.round(baseCarbonValue * speciesMultiplier.carbon); + const totalValue = timberValue + carbonValue; + const roi = Math.round(((totalValue / 3000) - 1) * 100); // Assuming 3000 SEK average cost + const growthRate = Math.round(speciesMultiplier.growth * 100) / 10; // Growth rate as %/year + + // Risk factors (higher is riskier) + const riskFactors = { + 'Pine': 2, 'Oak': 1, 'Birch': 3, 'Spruce': 2, 'Maple': 1, + 'Beech': 1, 'Fir': 2, 'Cedar': 1, 'Willow': 4, 'Ash': 2 + }; + + return { + species: speciesName, + timberValue, + carbonValue, + totalValue, + roi, + growthRate, + riskLevel: riskFactors[speciesName] || 2, + treeCount: Math.floor(Math.random() * 500) + 100, // Mock tree count + averageAge: Math.floor(Math.random() * 30) + 10, // 10-40 years + // Performance metrics + timberValuePerYear: Math.round(timberValue / 25), // Assuming 25-year rotation + carbonValuePerYear: Math.round(carbonValue / 10), // Annual carbon sequestration + totalROI: roi + }; + }).sort((a, b) => b.totalValue - a.totalValue); // Sort by total value descending + + setData(performanceData); + } catch (err) { + console.error('Error fetching species performance data:', err); + setError(err.message || 'Failed to load species data'); + } finally { + setLoading(false); + } + }; + + fetchSpeciesData(); + }, [filters, chartType]); + + const formatCurrency = (value) => { + if (!value) return '0 SEK'; + return new Intl.NumberFormat('sv-SE', { + style: 'currency', + currency: 'SEK', + minimumFractionDigits: 0, + maximumFractionDigits: 0 + }).format(value); + }; + + const formatPercentage = (value) => { + return `${value}%`; + }; + + const getROIColor = (roi) => { + if (roi >= 15) return '#10b981'; // Green for high ROI + if (roi >= 5) return '#f59e0b'; // Yellow for medium ROI + return '#ef4444'; // Red for low ROI + }; + + const getRiskColor = (riskLevel) => { + const colors = ['#10b981', '#f59e0b', '#ef4444', '#dc2626', '#991b1b']; + return colors[riskLevel - 1] || '#6b7280'; + }; + + const CustomTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + return ( +
+

{label}

+
+

+ Total Value: {formatCurrency(data.totalValue)} +

+

+ Timber Value: {formatCurrency(data.timberValue)} +

+

+ Carbon Value: {formatCurrency(data.carbonValue)} +

+

+ ROI: {formatPercentage(data.roi)} +

+

+ Growth Rate: {data.growthRate}%/year +

+

+ Trees: {data.treeCount.toLocaleString()} +

+

+ Risk Level: {data.riskLevel}/5 +

+
+
+ ); + } + return null; + }; + + const ScatterTooltip = ({ active, payload, label }) => { + if (active && payload && payload.length) { + const data = payload[0].payload; + return ( +
+

{data.species}

+
+

+ ROI: {formatPercentage(data.roi)} +

+

+ Growth Rate: {data.growthRate}%/year +

+

+ Total Value: {formatCurrency(data.totalValue)} +

+
+
+ ); + } + return null; + }; + + if (loading) { + return ( + + + Species Economic Performance + +
+ +
+
+ ); + } + + if (error) { + return ( + + + Species Economic Performance + +
+
+

Error loading chart data

+

{error}

+
+
+
+ ); + } + + if (!data || data.length === 0) { + return ( + + + Species Economic Performance + +
+

No species data available

+
+
+ ); + } + + if (chartType === 'scatter') { + return ( + + + ROI vs Growth Rate by Species +

+ Investment return vs growth performance analysis +

+
+ + + + + + } /> + + {data.map((entry, index) => ( + + ))} + + + +
+ ); + } + + return ( + + + Species Economic Performance +

+ Timber and carbon value potential by tree species +

+
+ + + + + `${(value / 1000).toFixed(0)}k`} + /> + } /> + + + {/* Timber value bar */} + + + {/* Carbon value bar */} + + + +
+ ); +}; + +export default SpeciesEconomicPerformanceChart; \ No newline at end of file diff --git a/frontend/src/components/charts/SurvivalRateChart.jsx b/frontend/src/components/charts/SurvivalRateChart.jsx new file mode 100644 index 0000000000..993a3b4f59 --- /dev/null +++ b/frontend/src/components/charts/SurvivalRateChart.jsx @@ -0,0 +1,101 @@ +import React, { memo } from 'react'; +import { PieChart, Pie, Cell, ResponsiveContainer, Tooltip } from 'recharts'; +import { ChartContainer, ChartHeader, ChartTitle, ChartTooltip, ChartPercentageDisplay, ChartLegend } from '../ui/ChartComponents'; +import LoadingSpinner from '../ui/LoadingSpinner'; +import { useChartConfig } from '../../hooks/useChartConfig'; +import { useSurvivalRateData } from '../../hooks/useChartData'; + + +const SurvivalRateChartComponent = ({ filters = {}, dashboardData = null }) => { + // Use dashboard data if available, otherwise fetch from API + const { data: apiData, loading, error } = useSurvivalRateData( + dashboardData ? {} : filters, // Skip API call if we have dashboard data + { skip: !!dashboardData } + ); + + // Use dashboard data first, then API data + const rawChartData = dashboardData?.charts?.survivalRate || apiData; + + // Calculate survival rate from available data + const survivalRate = rawChartData?.chartData?.length > 0 + ? rawChartData.chartData[rawChartData.chartData.length - 1]?.survivalRate || 0 + : 0; + + // If using dashboard data, no loading state needed + const isLoading = dashboardData ? false : loading; + const hasError = dashboardData ? false : error; + + const data = { + survived: survivalRate, + lost: 100 - survivalRate + }; + const chartConfig = useChartConfig('pie'); + const chartData = [ + { name: 'Survived', value: data.survived, color: '#10b981' }, + { name: 'Lost', value: data.lost, color: '#ef4444' } + ]; + + if (isLoading) { + return ( + + + Tree Survival Rate + +
+ +
+
+ ); + } + + if (hasError) { + return ( + + + Tree Survival Rate + +
+
+

Error loading chart data

+

{hasError}

+
+
+
+ ); + } + + return ( + + + Tree Survival Rate + + + + + + + + `${value.toFixed(1)}%`} />} /> + + + + + + ); +}; + +// Memoize component to prevent unnecessary re-renders +export const SurvivalRateChart = memo(SurvivalRateChartComponent); \ No newline at end of file diff --git a/frontend/src/components/charts/index.js b/frontend/src/components/charts/index.js new file mode 100644 index 0000000000..bf0305a404 --- /dev/null +++ b/frontend/src/components/charts/index.js @@ -0,0 +1,14 @@ +// Chart Components +export { SurvivalRateChart } from './SurvivalRateChart'; +export { AverageHeightChart } from './AverageHeightChart'; +export { CO2AbsorptionChart } from './CO2AbsorptionChart'; + +// Shared Chart Utilities +export { + ChartContainer, + ChartHeader, + ChartTitle, + ChartTooltip, + CustomTooltipContainer +} from '../ui/ChartComponents'; +export { useChartConfig } from '../../hooks/useChartConfig'; \ No newline at end of file diff --git a/frontend/src/components/filters/DateInput.jsx b/frontend/src/components/filters/DateInput.jsx new file mode 100644 index 0000000000..60ae170c97 --- /dev/null +++ b/frontend/src/components/filters/DateInput.jsx @@ -0,0 +1,36 @@ +import { DateInputWrapper, DateLabel, StyledDatePicker } from './DateRangePicker.styles'; + +export const DateInput = ({ + label, + selected, + onChange, + selectsStart, + selectsEnd, + startDate, + endDate, + minDate, + maxDate, + placeholderText +}) => { + const inputId = `date-input-${label.toLowerCase().replace(/\s+/g, '-')}`; + + return ( + + {label} + + + ); +}; \ No newline at end of file diff --git a/frontend/src/components/filters/DateRangePicker.jsx b/frontend/src/components/filters/DateRangePicker.jsx new file mode 100644 index 0000000000..6a842481f0 --- /dev/null +++ b/frontend/src/components/filters/DateRangePicker.jsx @@ -0,0 +1,62 @@ +import 'react-datepicker/dist/react-datepicker.css'; +import { useDateRange } from '../../hooks/useDateRange'; +import { DateInput } from './DateInput'; +import { + FilterContainer, + FilterHeader, + FilterTitle, + DateInputGroup, + DateSeparator, + ResetButton +} from './DateRangePicker.styles'; + +export const DateRangePicker = ({ onDateChange, initialStartDate, initialEndDate }) => { + // TODO: Consider moving date range state to Zustand store for global access + const { + startDate, + endDate, + handleStartDateChange, + handleEndDateChange, + handleReset + } = useDateRange(onDateChange, initialStartDate, initialEndDate); + + return ( + + + Date Range + + Reset + + + + + + + + to + + + + + + ); +}; \ No newline at end of file diff --git a/frontend/src/components/filters/DateRangePicker.styles.js b/frontend/src/components/filters/DateRangePicker.styles.js new file mode 100644 index 0000000000..a7b858a7b1 --- /dev/null +++ b/frontend/src/components/filters/DateRangePicker.styles.js @@ -0,0 +1,93 @@ +import styled from 'styled-components'; +import DatePicker from 'react-datepicker'; + +export const FilterContainer = styled.div` + background: white; + border-radius: 0.75rem; + padding: 1.5rem; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; +`; + +export const FilterHeader = styled.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; +`; + +export const FilterTitle = styled.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`; + +export const DateInputGroup = styled.div` + display: flex; + gap: 1rem; + align-items: center; + flex-wrap: wrap; +`; + +export const DateInputWrapper = styled.div` + display: flex; + flex-direction: column; + min-width: 150px; +`; + +export const DateLabel = styled.label` + font-size: 0.875rem; + font-weight: 500; + color: #374151; + margin-bottom: 0.5rem; +`; + +export const StyledDatePicker = styled(DatePicker)` + width: 100%; + padding: 0.5rem 0.75rem; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + color: #111827; + background-color: white; + + &:focus { + outline: none; + border-color: #10b981; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } + + &:hover { + border-color: #9ca3af; + } +`; + +export const ResetButton = styled.button` + padding: 0.5rem 1rem; + background-color: #f3f4f6; + color: #374151; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + font-weight: 500; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background-color: #e5e7eb; + border-color: #9ca3af; + } + + &:focus { + outline: none; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } +`; + +export const DateSeparator = styled.div` + display: flex; + align-items: center; + color: #6b7280; + font-size: 0.875rem; +`; \ No newline at end of file diff --git a/frontend/src/components/filters/FilterErrors.jsx b/frontend/src/components/filters/FilterErrors.jsx new file mode 100644 index 0000000000..4c5568df80 --- /dev/null +++ b/frontend/src/components/filters/FilterErrors.jsx @@ -0,0 +1,21 @@ +export const FilterErrors = ({ errors }) => { + if (!errors || Object.keys(errors).length === 0) { + return null; + } + + return ( +
+
+ + + + Filter Validation Errors +
+
    + {Object.entries(errors).map(([field, error]) => ( +
  • • {error}
  • + ))} +
+
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/filters/ForestSelector.jsx b/frontend/src/components/filters/ForestSelector.jsx new file mode 100644 index 0000000000..910bc42b9c --- /dev/null +++ b/frontend/src/components/filters/ForestSelector.jsx @@ -0,0 +1,285 @@ +import { useState, useEffect } from 'react'; +import styled from 'styled-components'; +import { forestAPI } from '../../lib/api'; + +const FilterContainer = styled.div` + background: white; + border-radius: 0.75rem; + padding: 1.5rem; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; +`; + +const FilterHeader = styled.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; +`; + +const FilterTitle = styled.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`; + +const SearchInput = styled.input` + width: 100%; + padding: 0.5rem 0.75rem; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + color: #111827; + background-color: white; + margin-bottom: 1rem; + + &:focus { + outline: none; + border-color: #10b981; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } + + &:hover { + border-color: #9ca3af; + } +`; + +const ForestList = styled.div` + max-height: 200px; + overflow-y: auto; + border: 1px solid #e5e7eb; + border-radius: 0.5rem; + background-color: white; +`; + +const ForestItem = styled.label` + display: flex; + align-items: center; + padding: 0.75rem; + cursor: pointer; + border-bottom: 1px solid #f3f4f6; + transition: background-color 0.2s; + + &:last-child { + border-bottom: none; + } + + &:hover { + background-color: #f9fafb; + } + + &:focus-within { + background-color: #f0fdf4; + } +`; + +const Checkbox = styled.input` + margin-right: 0.75rem; + width: 1rem; + height: 1rem; + accent-color: #10b981; +`; + +const ForestInfo = styled.div` + flex: 1; +`; + +const ForestName = styled.div` + font-weight: 500; + color: #111827; + font-size: 0.875rem; +`; + +const ForestDetails = styled.div` + font-size: 0.75rem; + color: #6b7280; + margin-top: 0.25rem; +`; + +const SelectedCount = styled.div` + font-size: 0.75rem; + color: #6b7280; + margin-top: 0.5rem; +`; + +const ResetButton = styled.button` + padding: 0.5rem 1rem; + background-color: #f3f4f6; + color: #374151; + border: 1px solid #d1d5db; + border-radius: 0.5rem; + font-size: 0.875rem; + font-weight: 500; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background-color: #e5e7eb; + border-color: #9ca3af; + } + + &:focus { + outline: none; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } +`; + +export const ForestSelector = ({ + selectedForests = [], + onChange +}) => { + const [searchTerm, setSearchTerm] = useState(''); + const [forests, setForests] = useState([]); + const [filteredForests, setFilteredForests] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + // Fetch forests from API on component mount + useEffect(() => { + const fetchForests = async () => { + try { + setLoading(true); + setError(null); + const response = await forestAPI.getAll(); + + // Transform the forest data to match our component's expectations + const transformedForests = response.data.forests.map(forest => ({ + id: forest.id, + name: forest.name, + region: forest.region, + treeCount: forest.treeCount || 0, + area: `${Math.round(forest.area)} ha`, + areaNumeric: forest.area, // Store numeric value for sorting if needed + isActive: forest.isActive + })).filter(forest => forest.isActive); // Only show active forests + + setForests(transformedForests); + } catch (err) { + console.error('Error fetching forests:', err); + setError('Failed to load forests'); + setForests([]); // Fallback to empty array + } finally { + setLoading(false); + } + }; + + fetchForests(); + }, []); + + // Filter forests based on search term + useEffect(() => { + const filtered = forests.filter(forest => + forest.name.toLowerCase().includes(searchTerm.toLowerCase()) || + forest.region.toLowerCase().includes(searchTerm.toLowerCase()) + ); + setFilteredForests(filtered); + }, [searchTerm, forests]); + + const handleForestToggle = (forestId) => { + if (onChange) { + if (selectedForests.includes(forestId)) { + onChange(selectedForests.filter(id => id !== forestId)); + } else { + onChange([...selectedForests, forestId]); + } + } + }; + + const handleSelectAll = () => { + if (onChange) { + onChange(forests.map(forest => forest.id)); + } + }; + + const handleSelectNone = () => { + if (onChange) { + onChange([]); + } + }; + + const handleReset = () => { + if (onChange) { + onChange([]); + } + setSearchTerm(''); + }; + + return ( + + + Forest Selection + + Reset + + + + {loading ? ( +
+
Loading forests...
+
+ ) : error ? ( +
+
{error}
+ +
+ ) : ( + <> + setSearchTerm(e.target.value)} + /> + +
+ + +
+ + + {filteredForests.map(forest => ( + + handleForestToggle(forest.id)} + /> + + {forest.name} + + {forest.region} • {forest.treeCount} trees • {forest.area} + + + + ))} + {filteredForests.length === 0 && !loading && ( +
+ {forests.length === 0 ? 'No forests available.' : 'No forests found matching your search.'} +
+ )} +
+ + )} + + + {selectedForests.length} of {forests.length} forests selected + +
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/filters/GlobalFilters.jsx b/frontend/src/components/filters/GlobalFilters.jsx new file mode 100644 index 0000000000..bf4d86d982 --- /dev/null +++ b/frontend/src/components/filters/GlobalFilters.jsx @@ -0,0 +1,380 @@ +import { useState, useEffect, useRef, useCallback } from 'react'; +import { useSearchParams } from 'react-router-dom'; +import { DateRangePicker } from './DateRangePicker'; +import { ForestSelector } from './ForestSelector'; +import { FilterErrors } from './FilterErrors'; +import { formatDateForInput } from '@/utils/dateUtils'; +import { validateFilters, createDefaultFilters } from '@/utils/filterValidation'; +import { useFiltersStore } from '../../lib/stores/filtersStore'; + + +export const GlobalFilters = ({ onFiltersChange, initialFilters = {} }) => { + const [searchParams, setSearchParams] = useSearchParams(); + const onFiltersChangeRef = useRef(onFiltersChange); + const hasMounted = useRef(false); + const didInitURL = useRef(false); + const [isCollapsed, setIsCollapsed] = useState(true); + const [activePreset, setActivePreset] = useState(null); + const [filterErrors, setFilterErrors] = useState({}); + + // Get filter state and actions from store + const { + filters, + setFilters, + clearFilters, + setDateRange, + setForestFilter, + getActiveFiltersCount, + hasActiveFilters, + } = useFiltersStore(); + + // Ensure filters has default structure + const safeFilters = filters || { + dateRange: { start: null, end: null }, + forests: [], + regions: [], + species: [], + status: 'all', + soilCondition: '', + sunlightExposure: '', + search: '' + }; + + // Keep the ref up to date + useEffect(() => { + onFiltersChangeRef.current = onFiltersChange; + }, [onFiltersChange]); + + // Initialize from URL parameters only once on mount + useEffect(() => { + if (!didInitURL.current) { + const startDateParam = searchParams.get('startDate'); + const endDateParam = searchParams.get('endDate'); + const forestsParam = searchParams.get('forests'); + + let needsUpdate = false; + const urlFilters = { ...safeFilters }; + + // Parse date range from URL + if (startDateParam && endDateParam) { + try { + const startDate = new Date(startDateParam); + const endDate = new Date(endDateParam); + + if (!isNaN(startDate.getTime()) && !isNaN(endDate.getTime()) && startDate <= endDate) { + urlFilters.dateRange = { start: startDate, end: endDate }; + needsUpdate = true; + } + } catch (error) { + console.warn('Invalid date parameters in URL:', error); + } + } + + // Parse selected forests from URL + if (forestsParam) { + try { + const forestIds = forestsParam.split(',').map(id => parseInt(id, 10)).filter(id => !isNaN(id)); + if (forestIds.length > 0) { + urlFilters.forests = forestIds; + needsUpdate = true; + } + } catch (error) { + console.warn('Invalid forest parameters in URL:', error); + } + } + + // Apply URL filters to store if found + if (needsUpdate) { + setFilters(urlFilters); + } + + // Apply any initial filters passed as props + if (Object.keys(initialFilters).length > 0) { + setFilters({ ...urlFilters, ...initialFilters }); + } + + didInitURL.current = true; + } + }, [searchParams, setFilters, initialFilters, safeFilters]); + + // Update URL parameters when filters change + const updateURLParams = useCallback((currentFilters) => { + const params = new URLSearchParams(); + + // Add date range to URL + if (currentFilters.dateRange?.start && currentFilters.dateRange?.end) { + params.set('startDate', formatDateForInput(currentFilters.dateRange.start)); + params.set('endDate', formatDateForInput(currentFilters.dateRange.end)); + } + + // Add selected forests to URL + if (currentFilters.forests && currentFilters.forests.length > 0) { + params.set('forests', currentFilters.forests.join(',')); + } + + // Update URL without triggering navigation + setSearchParams(params, { replace: true }); + }, [setSearchParams]); + + // Debounced notify parent of filter changes with validation and URL persistence + useEffect(() => { + // Don't call callback on initial mount + if (!hasMounted.current) { + hasMounted.current = true; + return; + } + + const timer = setTimeout(() => { + // Create compatible filter format for validation + const validationFilters = { + dateRange: { + startDate: safeFilters.dateRange.start, + endDate: safeFilters.dateRange.end, + }, + selectedForests: safeFilters.forests, + }; + + // Validate filters + const errors = validateFilters(validationFilters); + setFilterErrors(errors); + + const isValid = Object.keys(errors).length === 0; + + if (isValid) { + // Update URL parameters + updateURLParams(safeFilters); + + // Notify parent component with store's getApiParams format + if (onFiltersChangeRef.current) { + onFiltersChangeRef.current(safeFilters); + } + } + }, 300); // Reduced debounce for better responsiveness + + return () => clearTimeout(timer); + }, [safeFilters, updateURLParams]); + + // Calculate active filters for display + const activeFilters = []; + if (safeFilters.dateRange.start && safeFilters.dateRange.end) { + const startDate = safeFilters.dateRange.start.toLocaleDateString(); + const endDate = safeFilters.dateRange.end.toLocaleDateString(); + activeFilters.push(`Date: ${startDate} - ${endDate}`); + } + if (safeFilters.forests.length > 0) { + activeFilters.push(`${safeFilters.forests.length} forests selected`); + } + + const handleDateChange = useCallback((dateRange) => { + // Convert from old format to store format + setDateRange({ + start: dateRange.startDate, + end: dateRange.endDate + }); + }, [setDateRange]); + + const handleForestChange = useCallback((selectedForests) => { + setForestFilter(selectedForests); + }, [setForestFilter]); + + const handleClearAll = useCallback(() => { + clearFilters(); + setFilterErrors({}); + setActivePreset(null); + + // Clear URL parameters + setSearchParams({}, { replace: true }); + }, [clearFilters, setSearchParams]); + + const applyDatePreset = useCallback((presetName) => { + const now = new Date(); + let startDate, endDate; + + switch (presetName) { + case 'mtd': // Month to date + startDate = new Date(now.getFullYear(), now.getMonth(), 1); + endDate = now; + break; + case 'ytd': // Year to date + startDate = new Date(now.getFullYear(), 0, 1); + endDate = now; + break; + case '30days': + startDate = new Date(now.getTime() - 30 * 24 * 60 * 60 * 1000); + endDate = now; + break; + case '90days': + startDate = new Date(now.getTime() - 90 * 24 * 60 * 60 * 1000); + endDate = now; + break; + case '1year': + startDate = new Date(now.getFullYear() - 1, now.getMonth(), now.getDate()); + endDate = now; + break; + case '3years': + startDate = new Date(now.getFullYear() - 3, now.getMonth(), now.getDate()); + endDate = now; + break; + case '5years': + startDate = new Date(now.getFullYear() - 5, now.getMonth(), now.getDate()); + endDate = now; + break; + case 'all': + startDate = null; + endDate = null; + break; + default: + return; + } + + setActivePreset(presetName); + handleDateChange({ startDate, endDate }); + }, [handleDateChange]); + + const toggleCollapsed = useCallback(() => { + setIsCollapsed(prev => !prev); + }, []); + + // Reset active preset when dates are manually changed + useEffect(() => { + // Only reset if dates were changed manually (not by preset) + if (activePreset && safeFilters.dateRange.start && safeFilters.dateRange.end) { + // For now, simply reset preset when dates change + // This could be enhanced to detect if dates match specific presets + setActivePreset(null); + } + }, [safeFilters.dateRange, activePreset]); + + return ( +
+
+
+
+

+ Global Filters + {isCollapsed && activeFilters.length > 0 && ( + + ({activeFilters.length} active) + + )} +

+ {!isCollapsed && ( +

+ Filter your data by date range and forest selection +

+ )} +
+
+
+ + +
+
+ +
+ + +
+ + + + + + + + +
+ +
+ + +
+ + {activeFilters.length > 0 && ( +
+ {activeFilters.map((filter, index) => ( + + {filter} + + ))} +
+ )} +
+
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/filters/index.js b/frontend/src/components/filters/index.js new file mode 100644 index 0000000000..d31ec0391a --- /dev/null +++ b/frontend/src/components/filters/index.js @@ -0,0 +1,8 @@ +// Main filter components +export { DateRangePicker } from './DateRangePicker'; +export { ForestSelector } from './ForestSelector'; +export { GlobalFilters } from './GlobalFilters'; + +// Shared filter components +export { FilterErrors } from './FilterErrors'; +export { DateInput } from './DateInput'; \ No newline at end of file diff --git a/frontend/src/components/map/ClusterIcon.jsx b/frontend/src/components/map/ClusterIcon.jsx new file mode 100644 index 0000000000..67abacd186 --- /dev/null +++ b/frontend/src/components/map/ClusterIcon.jsx @@ -0,0 +1,39 @@ +import L from 'leaflet'; +import { CLUSTER_SIZES, CLUSTER_ICON_CONFIG } from '@/constants/mapConstants'; + +export const createClusterIcon = (count) => { + const sizeCategory = getClusterSizeCategory(count); + const { dimension, fontSize } = CLUSTER_SIZES[sizeCategory]; + const { backgroundColor, className, iconSize } = CLUSTER_ICON_CONFIG; + + return L.divIcon({ + html: getClusterIconHTML(backgroundColor, dimension, fontSize, count), + className, + iconSize + }); +}; + +const getClusterSizeCategory = (count) => { + if (count > CLUSTER_SIZES.large.threshold) return 'large'; + if (count > CLUSTER_SIZES.medium.threshold) return 'medium'; + return 'small'; +}; + +const getClusterIconHTML = (backgroundColor, dimension, fontSize, count) => ` +
${count}
+`; + +export default { createClusterIcon }; \ No newline at end of file diff --git a/frontend/src/components/map/ForestMap.jsx b/frontend/src/components/map/ForestMap.jsx new file mode 100644 index 0000000000..c31dbc2a72 --- /dev/null +++ b/frontend/src/components/map/ForestMap.jsx @@ -0,0 +1,146 @@ +import { useState, useCallback } from 'react'; +import { MapContainer, TileLayer } from 'react-leaflet'; +import L from 'leaflet'; +import 'leaflet.markercluster/dist/MarkerCluster.css'; +import 'leaflet.markercluster/dist/MarkerCluster.Default.css'; +import 'leaflet/dist/leaflet.css'; +import LoadingSpinner from '@/components/ui/LoadingSpinner'; +import MarkerCluster from './MarkerCluster'; +import MapLoadingHandler from './MapLoadingHandler'; +import MapController from './MapController'; +import { + MapContainerStyled, + LoadingOverlay, + ErrorOverlay, + ErrorContent +} from './ForestMap.styles'; + +// Fix for default markers in react-leaflet +delete L.Icon.Default.prototype._getIconUrl; +L.Icon.Default.mergeOptions({ + iconRetinaUrl: 'https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.7.1/images/marker-icon-2x.png', + iconUrl: 'https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.7.1/images/marker-icon.png', + shadowUrl: 'https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.7.1/images/marker-shadow.png', +}); + +export const ForestMap = ({ + trees = [], + onTreeSelect, + loading = false, + error = null +}) => { + const [zoom, setZoom] = useState(5); + const [selectedTree, setSelectedTree] = useState(null); + const [mapLoading, setMapLoading] = useState(true); + const [mapError, setMapError] = useState(null); + + // Trees are already filtered by the parent component + const filteredTrees = trees; + + const handleTreeClick = useCallback((tree) => { + setSelectedTree(tree); + if (onTreeSelect) { + onTreeSelect(tree); + } + }, [onTreeSelect]); + + const handleZoomChange = useCallback((newZoom) => { + setZoom(newZoom); + }, []); + + const handleMapReady = useCallback(() => { + setMapLoading(false); + setMapError(null); + }, []); + + const handleMapError = useCallback((error) => { + setMapLoading(false); + setMapError(error?.message || 'Failed to load map'); + }, []); + + const retryMap = useCallback(() => { + setMapLoading(true); + setMapError(null); + // In a real implementation, this would reload the map + setTimeout(() => { + setMapLoading(false); + }, 1000); + }, []); + + return ( +
+ + + + + + + + + + {/* Loading overlay */} + {(mapLoading || loading) && ( + +
+ +

Loading map...

+
+
+ )} + + {/* Error overlay */} + {(mapError || error) && ( + + + + + +

Map Error

+

{mapError || error}

+ +
+
+ )} +
+ + {/* Map Legend */} +
+

Legend

+
+
+
+ Healthy Trees +
+
+
+ Warning +
+
+
+ Critical +
+
+ Showing {filteredTrees.length} of {trees.length} trees +
+
+ Zoom level: {zoom} + {zoom < 12 && (Clustering enabled)} +
+
+
+
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/map/ForestMap.styles.js b/frontend/src/components/map/ForestMap.styles.js new file mode 100644 index 0000000000..475846511c --- /dev/null +++ b/frontend/src/components/map/ForestMap.styles.js @@ -0,0 +1,85 @@ +import styled from 'styled-components'; + +export const MapContainerStyled = styled.div` + height: 600px; + width: 100%; + border-radius: 0.75rem; + overflow: hidden; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; + position: relative; +`; + +export const MapHeader = styled.div` + background: white; + padding: 1rem 1.5rem; + border-bottom: 1px solid #e5e7eb; + display: flex; + justify-content: space-between; + align-items: center; +`; + +export const MapTitle = styled.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`; + +export const MapControls = styled.div` + display: flex; + gap: 0.5rem; +`; + +export const ControlButton = styled.button` + padding: 0.5rem; + background: white; + border: 1px solid #d1d5db; + border-radius: 0.375rem; + cursor: pointer; + transition: all 0.2s; + + &:hover { + background: #f9fafb; + border-color: #9ca3af; + } + + &:focus { + outline: none; + box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1); + } +`; + +export const LoadingOverlay = styled.div` + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: rgba(255, 255, 255, 0.9); + display: flex; + align-items: center; + justify-content: center; + z-index: 1000; + border-radius: 0.75rem; +`; + +export const ErrorOverlay = styled.div` + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: rgba(255, 255, 255, 0.95); + display: flex; + align-items: center; + justify-content: center; + z-index: 1000; + border-radius: 0.75rem; +`; + +export const ErrorContent = styled.div` + text-align: center; + padding: 2rem; + max-width: 300px; +`; \ No newline at end of file diff --git a/frontend/src/components/map/MapController.jsx b/frontend/src/components/map/MapController.jsx new file mode 100644 index 0000000000..43038eb5c5 --- /dev/null +++ b/frontend/src/components/map/MapController.jsx @@ -0,0 +1,23 @@ +import { useEffect } from 'react'; +import { useMap } from 'react-leaflet'; + +const MapController = ({ onZoomChange }) => { + const map = useMap(); + + useEffect(() => { + const handleZoomEnd = () => { + if (onZoomChange) { + onZoomChange(map.getZoom()); + } + }; + + map.on('zoomend', handleZoomEnd); + return () => { + map.off('zoomend', handleZoomEnd); + }; + }, [map]); + + return null; +}; + +export default MapController; \ No newline at end of file diff --git a/frontend/src/components/map/MapLoadingHandler.jsx b/frontend/src/components/map/MapLoadingHandler.jsx new file mode 100644 index 0000000000..1d579fb63a --- /dev/null +++ b/frontend/src/components/map/MapLoadingHandler.jsx @@ -0,0 +1,37 @@ +import { useEffect } from 'react'; +import { useMap } from 'react-leaflet'; + +const MapLoadingHandler = ({ onMapReady, onMapError }) => { + const map = useMap(); + + useEffect(() => { + const handleMapReady = () => { + if (onMapReady) { + onMapReady(); + } + }; + + const handleMapError = (error) => { + console.error('Map error:', error); + if (onMapError) { + onMapError(error); + } + }; + + // Map is ready when tiles are loaded + map.whenReady(() => { + handleMapReady(); + }); + + // Handle tile errors + map.on('tileerror', handleMapError); + + return () => { + map.off('tileerror', handleMapError); + }; + }, [map, onMapReady, onMapError]); + + return null; +}; + +export default MapLoadingHandler; \ No newline at end of file diff --git a/frontend/src/components/map/MarkerCluster.jsx b/frontend/src/components/map/MarkerCluster.jsx new file mode 100644 index 0000000000..aaf8b63aaa --- /dev/null +++ b/frontend/src/components/map/MarkerCluster.jsx @@ -0,0 +1,74 @@ +import { useEffect } from 'react'; +import { useMap } from 'react-leaflet'; +import L from 'leaflet'; +import 'leaflet.markercluster'; +import { createTreeIcon, createClusterIcon, getTreePopupContent } from '@/utils/mapIcons'; + +const MarkerCluster = ({ trees, onTreeClick, zoom }) => { + const map = useMap(); + + useEffect(() => { + if (!map || !trees.length) return; + + // Clear prior marker layers (cluster groups and layer groups) + map.eachLayer(layer => { + if (layer instanceof L.MarkerClusterGroup || layer instanceof L.LayerGroup) { + map.removeLayer(layer); + } + }); + + // Only use clustering for zoom levels < 12 + if (zoom < 12) { + // Create cluster group + const clusterGroup = L.markerClusterGroup({ + chunkedLoading: true, + spiderfyOnMaxZoom: true, + showCoverageOnHover: false, + zoomToBoundsOnClick: true, + maxClusterRadius: 50, + iconCreateFunction: (cluster) => { + const count = cluster.getChildCount(); + return createClusterIcon(count); + } + }); + + // Add markers to cluster group + trees.forEach(tree => { + const marker = L.marker([tree.lat, tree.lng], { + icon: createTreeIcon(tree.health) + }); + + marker.bindPopup(getTreePopupContent(tree)); + marker.on('click', () => onTreeClick(tree)); + clusterGroup.addLayer(marker); + }); + + map.addLayer(clusterGroup); + } else { + // For zoom >= 12, show individual markers inside a layer group + const layerGroup = L.layerGroup(); + trees.forEach(tree => { + const marker = L.marker([tree.lat, tree.lng], { + icon: createTreeIcon(tree.health) + }); + marker.bindPopup(getTreePopupContent(tree)); + marker.on('click', () => onTreeClick(tree)); + layerGroup.addLayer(marker); + }); + layerGroup.addTo(map); + } + + // Cleanup function + return () => { + map.eachLayer(layer => { + if (layer instanceof L.MarkerClusterGroup || layer instanceof L.LayerGroup) { + map.removeLayer(layer); + } + }); + }; + }, [map, trees, onTreeClick, zoom]); + + return null; +}; + +export default MarkerCluster; \ No newline at end of file diff --git a/frontend/src/components/map/TreeIcon.jsx b/frontend/src/components/map/TreeIcon.jsx new file mode 100644 index 0000000000..1e5d691048 --- /dev/null +++ b/frontend/src/components/map/TreeIcon.jsx @@ -0,0 +1,35 @@ +import L from 'leaflet'; +import { TREE_HEALTH_COLORS, TREE_ICON_CONFIG } from '@/constants/mapConstants'; + +export const createTreeIcon = (healthType = 'healthy') => { + const color = TREE_HEALTH_COLORS[healthType] || TREE_HEALTH_COLORS.healthy; + const { size, emoji, className } = TREE_ICON_CONFIG; + + return L.divIcon({ + className, + html: getTreeIconHTML(color, size, emoji), + iconSize: [size, size], + iconAnchor: [size / 2, size / 2] + }); +}; + +const getTreeIconHTML = (color, size, emoji) => ` +
+ ${emoji} +
+`; + +export default { createTreeIcon }; \ No newline at end of file diff --git a/frontend/src/components/map/TreePopup.jsx b/frontend/src/components/map/TreePopup.jsx new file mode 100644 index 0000000000..2b9b48782e --- /dev/null +++ b/frontend/src/components/map/TreePopup.jsx @@ -0,0 +1,55 @@ +import { TREE_HEALTH_STYLES } from '@/constants/mapConstants'; + +export const getTreePopupContent = (tree) => { + const healthStyle = TREE_HEALTH_STYLES[tree.health] || TREE_HEALTH_STYLES.healthy; + + return getPopupHTML(tree, healthStyle); +}; + +const getPopupHTML = (tree, healthStyle) => ` +
+

${tree.name}

+

Species: ${tree.species}

+

Height: ${tree.height}m

+

Health: + ${tree.health} +

+
+`; + +export const TreePopupContent = ({ tree }) => { + const healthStyle = TREE_HEALTH_STYLES[tree.health] || TREE_HEALTH_STYLES.healthy; + + return ( +
+

{tree.name}

+

Species: {tree.species}

+

Height: {tree.height}m

+

+ Health: + + {tree.health} + +

+
+ ); +}; + +const parseStyleString = (styleString) => { + const styles = {}; + styleString.split(';').forEach(style => { + const [property, value] = style.split(':').map(s => s.trim()); + if (property && value) { + const camelCaseProperty = property.replace(/-([a-z])/g, (g) => g[1].toUpperCase()); + styles[camelCaseProperty] = value; + } + }); + return styles; +}; + +export default { getTreePopupContent, TreePopupContent }; \ No newline at end of file diff --git a/frontend/src/components/map/index.js b/frontend/src/components/map/index.js new file mode 100644 index 0000000000..e233971147 --- /dev/null +++ b/frontend/src/components/map/index.js @@ -0,0 +1,12 @@ +// Map Components + +// Components with default exports +export { default as MarkerCluster } from './MarkerCluster'; +export { default as MapLoadingHandler } from './MapLoadingHandler'; +export { default as MapController } from './MapController'; +export { default as ClusterIcon } from './ClusterIcon'; +export { default as TreeIcon } from './TreeIcon'; +export { default as TreePopup } from './TreePopup'; + +// Component with named export +export { ForestMap } from './ForestMap'; \ No newline at end of file diff --git a/frontend/src/components/ui/AboutSection.jsx b/frontend/src/components/ui/AboutSection.jsx new file mode 100644 index 0000000000..d80cdd00f1 --- /dev/null +++ b/frontend/src/components/ui/AboutSection.jsx @@ -0,0 +1,31 @@ +import { Link } from 'react-router-dom'; +import { StatsGrid } from './StatsGrid'; + +export const AboutSection = () => { + return ( +
+
+
+
+

+ Empowering Sustainable Forestry +

+

+ Nanwa is dedicated to providing comprehensive tree monitoring solutions that help investors, growers, and environmental analysts make data-driven decisions for sustainable forestry projects. +

+

+ Our platform offers real-time insights, advanced analytics, and seamless data export capabilities to support your forestry management needs. +

+ + Join Nanwa Today + +
+ +
+
+
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/ui/Button.jsx b/frontend/src/components/ui/Button.jsx new file mode 100644 index 0000000000..8a560c3b7b --- /dev/null +++ b/frontend/src/components/ui/Button.jsx @@ -0,0 +1,62 @@ +import React from 'react'; +import { cn } from '@/lib/utils'; + +const buttonVariants = { + primary: 'btn-primary', + secondary: 'btn-secondary', + outline: 'btn-outline', + ghost: 'btn-ghost', + link: 'btn-link', + destructive: 'btn-destructive', + success: 'btn-success', +}; + +const buttonSizes = { + sm: 'h-8 px-3 text-xs', + md: 'h-10 px-4 py-2', + lg: 'h-12 px-8', + xl: 'h-14 px-10 text-lg', + icon: 'h-10 w-10', +}; + +export const Button = React.forwardRef( + ({ + className, + variant = 'primary', + size = 'md', + disabled = false, + loading = false, + type = 'button', + children, + ...props + }, ref) => { + return ( + + ); + } +); + +Button.displayName = 'Button'; \ No newline at end of file diff --git a/frontend/src/components/ui/ChartComponents.jsx b/frontend/src/components/ui/ChartComponents.jsx new file mode 100644 index 0000000000..fca66903bb --- /dev/null +++ b/frontend/src/components/ui/ChartComponents.jsx @@ -0,0 +1,127 @@ +import styled from 'styled-components'; + +// Common styled components for charts +export const ChartContainer = styled.div` + background: white; + border-radius: 0.75rem; + padding: 1.5rem; + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + border: 1px solid #e5e7eb; + height: 450px; + display: flex; + flex-direction: column; + overflow: hidden; +`; + +export const ChartHeader = styled.div` + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; +`; + +export const ChartTitle = styled.h3` + font-size: 1.125rem; + font-weight: 600; + color: #111827; + margin: 0; +`; + +export const CustomTooltipContainer = styled.div` + background: white; + border: 1px solid #e5e7eb; + border-radius: 0.5rem; + padding: 0.75rem; + box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.1); +`; + +// Chart percentage display components +export const PercentageDisplay = styled.div` + text-align: center; + margin-bottom: 1rem; +`; + +export const PercentageValue = styled.div` + font-size: 2.5rem; + font-weight: 700; + color: #16a34a; + line-height: 1; +`; + +export const PercentageLabel = styled.div` + font-size: 0.875rem; + color: #6b7280; + margin-top: 0.25rem; +`; + +// Chart legend components +export const LegendContainer = styled.div` + display: flex; + justify-content: center; + gap: 1rem; + margin-top: 1rem; + flex-wrap: wrap; +`; + +export const LegendItem = styled.div` + display: flex; + align-items: center; +`; + +export const LegendDot = styled.div` + width: 0.75rem; + height: 0.75rem; + border-radius: 50%; + margin-right: 0.5rem; + background-color: ${props => props.color}; +`; + +export const LegendText = styled.span` + font-size: 0.875rem; + color: #374151; +`; + +// Helper function to format tooltip values +const formatTooltipValue = (value, valueFormatter, unit) => { + if (valueFormatter) { + return valueFormatter(value); + } + return `${value}${unit}`; +}; + +// Reusable tooltip component with enhanced multi-series support +export const ChartTooltip = ({ active, payload, label, valueFormatter, unit = '' }) => { + if (active && payload && payload.length) { + return ( + +

{label}

+ {payload.map((entry, index) => ( +

+ {formatTooltipValue(entry.value, valueFormatter, unit)} +

+ ))} +
+ ); + } + return null; +}; + +// Reusable percentage display component +export const ChartPercentageDisplay = ({ value, label }) => ( + + {value}% + {label} + +); + +// Reusable chart legend component +export const ChartLegend = ({ data }) => ( + + {data.map((item, index) => ( + + + {item.name} ({item.value}%) + + ))} + +); \ No newline at end of file diff --git a/frontend/src/components/ui/ChartLoader.jsx b/frontend/src/components/ui/ChartLoader.jsx new file mode 100644 index 0000000000..32f30553b2 --- /dev/null +++ b/frontend/src/components/ui/ChartLoader.jsx @@ -0,0 +1,30 @@ +import React from 'react'; +import { StandardLoadingSpinner } from './StandardLoadingSpinner'; +import { SkeletonChart } from './SkeletonLoader'; +import { cn } from '@/lib/utils'; + +export const ChartLoader = ({ + variant = 'spinner', // 'spinner', 'skeleton' + title, + className = '' +}) => { + if (variant === 'skeleton') { + return ; + } + + return ( +
+ {title && ( +

+ {title} +

+ )} + +
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/ui/CloseIcon.jsx b/frontend/src/components/ui/CloseIcon.jsx new file mode 100644 index 0000000000..fabd709ef8 --- /dev/null +++ b/frontend/src/components/ui/CloseIcon.jsx @@ -0,0 +1,21 @@ +import React from 'react'; + +export const CloseIcon = ({ className = '', ...props }) => { + return ( + + ); +}; \ No newline at end of file diff --git a/frontend/src/components/ui/ColdStartLoader.jsx b/frontend/src/components/ui/ColdStartLoader.jsx new file mode 100644 index 0000000000..87a7896328 --- /dev/null +++ b/frontend/src/components/ui/ColdStartLoader.jsx @@ -0,0 +1,135 @@ +import { useEffect, useState } from 'react'; + +export const ColdStartLoader = ({ + connectionState, + onCancel, + showCancel = true +}) => { + const [progress, setProgress] = useState(0); + const [timeElapsed, setTimeElapsed] = useState(0); + + const { + isConnecting, + isColdStart, + retryAttempt, + totalAttempts, + estimatedWaitTime, + message + } = connectionState; + + // Progress bar animation + useEffect(() => { + if (!isConnecting) { + setProgress(0); + setTimeElapsed(0); + return; + } + + const interval = setInterval(() => { + setTimeElapsed(prev => prev + 1); + + if (isColdStart) { + // For cold starts, show more realistic progress + const maxTime = 60; // 60 seconds max expected time + const currentProgress = Math.min((timeElapsed / maxTime) * 100, 90); + setProgress(currentProgress); + } else { + // For normal connections, show faster progress + const maxTime = 10; + const currentProgress = Math.min((timeElapsed / maxTime) * 100, 90); + setProgress(currentProgress); + } + }, 1000); + + return () => clearInterval(interval); + }, [isConnecting, isColdStart, timeElapsed]); + + // Complete progress on success + useEffect(() => { + if (!isConnecting && progress > 0) { + setProgress(100); + } + }, [isConnecting, progress]); + + if (!isConnecting) return null; + + return ( +
+
+ {/* Header */} +
+
+ + + + +
+

+ {isColdStart ? 'Server Starting Up' : 'Connecting'} +

+
+ + {/* Message */} +
+

{message}

+ + {isColdStart && ( +
+

The server needs to start up from sleep mode.

+

This usually takes 30-60 seconds.

+
+ )} +
+ + {/* Progress Bar */} +
+
+ Progress + {Math.round(progress)}% +
+
+
+
+
+ + {/* Retry Info */} + {retryAttempt > 0 && ( +
+
+ + Attempt {retryAttempt} of {totalAttempts} + + + {timeElapsed}s elapsed + +
+
+ )} + + {/* Estimated Time */} + {estimatedWaitTime > 0 && ( +
+ Estimated wait time: {estimatedWaitTime} seconds +
+ )} + + {/* Cancel Button */} + {showCancel && onCancel && ( +
+ +
+ )} +
+
+ ); +}; + +export default ColdStartLoader; \ No newline at end of file diff --git a/frontend/src/components/ui/DarkModeToggle.jsx b/frontend/src/components/ui/DarkModeToggle.jsx new file mode 100644 index 0000000000..71046584ab --- /dev/null +++ b/frontend/src/components/ui/DarkModeToggle.jsx @@ -0,0 +1,46 @@ +import { useDarkMode } from '../../contexts/DarkModeContext'; +import { IconButton } from './IconButton'; +import { SunIcon } from './SunIcon'; +import { MoonIcon } from './MoonIcon'; +import { cn } from '@/lib/utils'; + +export const DarkModeToggle = ({ className = '', size = 'md' }) => { + const { isDarkMode, toggleDarkMode } = useDarkMode(); + + const iconSizes = { + sm: 'w-4 h-4', + md: 'w-5 h-5', + lg: 'w-6 h-6', + }; + + return ( + + {/* Sun icon for light mode */} + + + {/* Moon icon for dark mode */} + + + ); +}; \ No newline at end of file diff --git a/frontend/src/components/ui/DashboardHeader.jsx b/frontend/src/components/ui/DashboardHeader.jsx new file mode 100644 index 0000000000..85881b49de --- /dev/null +++ b/frontend/src/components/ui/DashboardHeader.jsx @@ -0,0 +1,52 @@ +import { useAuth } from '../../hooks/useAuth'; +import { DarkModeToggle } from './DarkModeToggle'; + +export const DashboardHeader = ({ onToggleSidebar }) => { + const { user, logout, isAdmin } = useAuth(); + + const handleLogout = async () => { + await logout(); + }; + + return ( +
+
+
+ +

+ Nanwa Dashboard +

+
+
+ + Welcome, {user?.firstName || user?.name} + {isAdmin() && ( + + Admin + + )} + + + +
+
+
+ ); +}; \ No newline at end of file diff --git a/frontend/src/components/ui/DashboardSidebar.jsx b/frontend/src/components/ui/DashboardSidebar.jsx new file mode 100644 index 0000000000..cb76e4736d --- /dev/null +++ b/frontend/src/components/ui/DashboardSidebar.jsx @@ -0,0 +1,145 @@ +import { useAuth } from '../../hooks/useAuth'; +import { useLocation, Link } from 'react-router-dom'; + +export const DashboardSidebar = ({ isOpen, onClose }) => { + const { isAdmin } = useAuth(); + const location = useLocation(); + + return ( + <> + {/* Mobile/Tablet Overlay */} +