MAJOR UPDATE - Full game systems based on APK analysis:
CAR SYSTEM:
- Purchase cars with Gold or Cash
- 5 starter cars across all classes (C,B,A,S,R)
- Car database with manufacturers and performance ratings
- Garage management and inventory tracking
UPGRADE SYSTEM:
- 5 upgrade types (Engine, Tires, Suspension, Brakes, Drivetrain)
- Progressive Performance Rating increases
- Cash-based upgrade economy
- Per-car upgrade tracking
PROGRESSION SYSTEM:
- Experience Points and leveling (1000 XP per level)
- Level-up rewards (10 Gold + 5K Cash)
- Reputation tracking
- Complete player profile management
CAREER MODE:
- Career series and event tracking
- Star rating system (1-3 stars per event)
- Best time tracking
- Star-based rewards (10G/2KC/100XP per star)
ECONOMY:
- Balanced F2P progression
- ~350 Gold + \ daily earning potential
- Fair pricing for cars and upgrades
- Multiple currency sources
NEW ENDPOINTS:
- GET /synergy/progression/player/{id} - Player profile
- POST /synergy/progression/player/{id}/update - Update stats
- POST /synergy/progression/car/purchase - Buy cars
- POST /synergy/progression/car/upgrade - Upgrade cars
- POST /synergy/progression/career/complete - Finish events
DATABASE:
- Cars table - Vehicle catalog
- OwnedCars table - Player garage
- CarUpgrades table - Upgrade options
- CareerProgress table - Event completion
- User table extended with Level/XP/Reputation
SEEDED DATA:
- 5 cars (Nissan Silvia to McLaren P1 GTR)
- 5 upgrades for starter car
- Time trials and gold packages from previous update
This creates a COMPLETE single-player experience with:
✓ Daily rewards + time trials
✓ Car ownership + garage
✓ Upgrade system
✓ Career progression
✓ Level/XP system
✓ Full economy
Based on actual RR3 game analysis!
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
354 lines
12 KiB
C#
354 lines
12 KiB
C#
using Microsoft.EntityFrameworkCore;
|
|
|
|
namespace RR3CommunityServer.Data;
|
|
|
|
public class RR3DbContext : DbContext
|
|
{
|
|
public RR3DbContext(DbContextOptions<RR3DbContext> options) : base(options) { }
|
|
|
|
public DbSet<Device> Devices { get; set; }
|
|
public DbSet<User> Users { get; set; }
|
|
public DbSet<Session> Sessions { get; set; }
|
|
public DbSet<Purchase> Purchases { get; set; }
|
|
public DbSet<CatalogItem> CatalogItems { get; set; }
|
|
public DbSet<DailyReward> DailyRewards { get; set; }
|
|
public DbSet<TimeTrial> TimeTrials { get; set; }
|
|
public DbSet<TimeTrialResult> TimeTrialResults { get; set; }
|
|
public DbSet<Car> Cars { get; set; }
|
|
public DbSet<OwnedCar> OwnedCars { get; set; }
|
|
public DbSet<CarUpgrade> CarUpgrades { get; set; }
|
|
public DbSet<CareerProgress> CareerProgress { get; set; }
|
|
|
|
protected override void OnModelCreating(ModelBuilder modelBuilder)
|
|
{
|
|
base.OnModelCreating(modelBuilder);
|
|
|
|
// Seed some default catalog items
|
|
modelBuilder.Entity<CatalogItem>().HasData(
|
|
new CatalogItem
|
|
{
|
|
Id = 1,
|
|
Sku = "com.ea.rr3.gold_1000",
|
|
Name = "1000 Gold",
|
|
Type = "currency",
|
|
Price = 0.99m,
|
|
Available = true
|
|
},
|
|
new CatalogItem
|
|
{
|
|
Id = 2,
|
|
Sku = "com.ea.rr3.car_tier1",
|
|
Name = "Starter Car",
|
|
Type = "car",
|
|
Price = 0m,
|
|
Available = true
|
|
},
|
|
new CatalogItem
|
|
{
|
|
Id = 3,
|
|
Sku = "com.ea.rr3.upgrade_engine",
|
|
Name = "Engine Upgrade",
|
|
Type = "upgrade",
|
|
Price = 4.99m,
|
|
Available = true
|
|
}
|
|
);
|
|
|
|
// Seed gold purchase options
|
|
modelBuilder.Entity<CatalogItem>().HasData(
|
|
new CatalogItem
|
|
{
|
|
Id = 4,
|
|
Sku = "com.ea.rr3.gold_100",
|
|
Name = "100 Gold",
|
|
Type = "currency",
|
|
Price = 0m, // FREE in community server
|
|
Available = true
|
|
},
|
|
new CatalogItem
|
|
{
|
|
Id = 5,
|
|
Sku = "com.ea.rr3.gold_500",
|
|
Name = "500 Gold",
|
|
Type = "currency",
|
|
Price = 0m,
|
|
Available = true
|
|
},
|
|
new CatalogItem
|
|
{
|
|
Id = 6,
|
|
Sku = "com.ea.rr3.gold_1000",
|
|
Name = "1000 Gold",
|
|
Type = "currency",
|
|
Price = 0m,
|
|
Available = true
|
|
},
|
|
new CatalogItem
|
|
{
|
|
Id = 7,
|
|
Sku = "com.ea.rr3.gold_5000",
|
|
Name = "5000 Gold",
|
|
Type = "currency",
|
|
Price = 0m,
|
|
Available = true
|
|
}
|
|
);
|
|
|
|
// Seed time trials
|
|
modelBuilder.Entity<TimeTrial>().HasData(
|
|
new TimeTrial
|
|
{
|
|
Id = 1,
|
|
Name = "Daily Sprint Challenge",
|
|
TrackName = "Silverstone National",
|
|
CarName = "Any Car",
|
|
StartDate = DateTime.UtcNow,
|
|
EndDate = DateTime.UtcNow.AddDays(7),
|
|
TargetTime = 90.5,
|
|
GoldReward = 50,
|
|
CashReward = 10000,
|
|
Active = true
|
|
},
|
|
new TimeTrial
|
|
{
|
|
Id = 2,
|
|
Name = "Speed Demon Trial",
|
|
TrackName = "Dubai Autodrome",
|
|
CarName = "Any Car",
|
|
StartDate = DateTime.UtcNow,
|
|
EndDate = DateTime.UtcNow.AddDays(7),
|
|
TargetTime = 120.0,
|
|
GoldReward = 100,
|
|
CashReward = 25000,
|
|
Active = true
|
|
}
|
|
);
|
|
|
|
// Seed starter cars
|
|
modelBuilder.Entity<Car>().HasData(
|
|
new Car
|
|
{
|
|
Id = 1,
|
|
CarId = "nissan_silvia_s15",
|
|
Name = "Nissan Silvia Spec-R",
|
|
Manufacturer = "Nissan",
|
|
ClassType = "C",
|
|
BasePerformanceRating = 45,
|
|
CashPrice = 25000,
|
|
GoldPrice = 0,
|
|
Available = true
|
|
},
|
|
new Car
|
|
{
|
|
Id = 2,
|
|
CarId = "ford_focus_rs",
|
|
Name = "Ford Focus RS",
|
|
Manufacturer = "Ford",
|
|
ClassType = "B",
|
|
BasePerformanceRating = 58,
|
|
CashPrice = 85000,
|
|
GoldPrice = 150,
|
|
Available = true
|
|
},
|
|
new Car
|
|
{
|
|
Id = 3,
|
|
CarId = "porsche_911_gt3",
|
|
Name = "Porsche 911 GT3 RS",
|
|
Manufacturer = "Porsche",
|
|
ClassType = "A",
|
|
BasePerformanceRating = 72,
|
|
CashPrice = 0,
|
|
GoldPrice = 350,
|
|
Available = true
|
|
},
|
|
new Car
|
|
{
|
|
Id = 4,
|
|
CarId = "ferrari_488_gtb",
|
|
Name = "Ferrari 488 GTB",
|
|
Manufacturer = "Ferrari",
|
|
ClassType = "S",
|
|
BasePerformanceRating = 88,
|
|
CashPrice = 0,
|
|
GoldPrice = 750,
|
|
Available = true
|
|
},
|
|
new Car
|
|
{
|
|
Id = 5,
|
|
CarId = "mclaren_p1_gtr",
|
|
Name = "McLaren P1 GTR",
|
|
Manufacturer = "McLaren",
|
|
ClassType = "R",
|
|
BasePerformanceRating = 105,
|
|
CashPrice = 0,
|
|
GoldPrice = 1500,
|
|
Available = true
|
|
}
|
|
);
|
|
|
|
// Seed some upgrade options
|
|
modelBuilder.Entity<CarUpgrade>().HasData(
|
|
// Nissan Silvia upgrades
|
|
new CarUpgrade { Id = 1, CarId = "nissan_silvia_s15", UpgradeType = "engine", Level = 1, CashCost = 5000, PerformanceIncrease = 3 },
|
|
new CarUpgrade { Id = 2, CarId = "nissan_silvia_s15", UpgradeType = "tires", Level = 1, CashCost = 3000, PerformanceIncrease = 2 },
|
|
new CarUpgrade { Id = 3, CarId = "nissan_silvia_s15", UpgradeType = "suspension", Level = 1, CashCost = 4000, PerformanceIncrease = 2 },
|
|
new CarUpgrade { Id = 4, CarId = "nissan_silvia_s15", UpgradeType = "brakes", Level = 1, CashCost = 3500, PerformanceIncrease = 2 },
|
|
new CarUpgrade { Id = 5, CarId = "nissan_silvia_s15", UpgradeType = "drivetrain", Level = 1, CashCost = 4500, PerformanceIncrease = 3 }
|
|
);
|
|
}
|
|
}
|
|
|
|
// Database entities
|
|
public class Device
|
|
{
|
|
public int Id { get; set; }
|
|
public string DeviceId { get; set; } = string.Empty;
|
|
public string HardwareId { get; set; } = string.Empty;
|
|
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
|
public DateTime LastSeenAt { get; set; } = DateTime.UtcNow;
|
|
}
|
|
|
|
public class User
|
|
{
|
|
public int Id { get; set; }
|
|
public string SynergyId { get; set; } = string.Empty;
|
|
public string? DeviceId { get; set; }
|
|
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
|
public string? Nickname { get; set; }
|
|
public int? Gold { get; set; } = 0;
|
|
public int? Cash { get; set; } = 0;
|
|
public int? Level { get; set; } = 1;
|
|
public int? Experience { get; set; } = 0;
|
|
public int? Reputation { get; set; } = 0;
|
|
|
|
// Navigation properties
|
|
public List<OwnedCar> OwnedCars { get; set; } = new();
|
|
public List<CareerProgress> CareerProgress { get; set; } = new();
|
|
}
|
|
|
|
public class Session
|
|
{
|
|
public int Id { get; set; }
|
|
public string SessionId { get; set; } = string.Empty;
|
|
public string? SynergyId { get; set; }
|
|
public string DeviceId { get; set; } = string.Empty;
|
|
public int? UserId { get; set; }
|
|
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
|
public DateTime ExpiresAt { get; set; }
|
|
}
|
|
|
|
public class Purchase
|
|
{
|
|
public int Id { get; set; }
|
|
public string SynergyId { get; set; } = string.Empty;
|
|
public string ItemId { get; set; } = string.Empty;
|
|
public string Sku { get; set; } = string.Empty;
|
|
public string OrderId { get; set; } = string.Empty;
|
|
public DateTime PurchaseTime { get; set; } = DateTime.UtcNow;
|
|
public string Token { get; set; } = string.Empty;
|
|
public decimal Price { get; set; }
|
|
public string Status { get; set; } = "approved";
|
|
// For web panel display
|
|
public int? UserId { get; set; }
|
|
public DateTime PurchaseDate => PurchaseTime;
|
|
}
|
|
|
|
public class CatalogItem
|
|
{
|
|
public int Id { get; set; }
|
|
public string Sku { get; set; } = string.Empty;
|
|
public string Name { get; set; } = string.Empty;
|
|
public string Type { get; set; } = string.Empty;
|
|
public decimal Price { get; set; }
|
|
public bool Available { get; set; } = true;
|
|
}
|
|
|
|
public class DailyReward
|
|
{
|
|
public int Id { get; set; }
|
|
public int UserId { get; set; }
|
|
public DateTime RewardDate { get; set; }
|
|
public int GoldAmount { get; set; }
|
|
public int CashAmount { get; set; }
|
|
public bool Claimed { get; set; }
|
|
public DateTime? ClaimedAt { get; set; }
|
|
public int Streak { get; set; }
|
|
}
|
|
|
|
public class TimeTrial
|
|
{
|
|
public int Id { get; set; }
|
|
public string Name { get; set; } = string.Empty;
|
|
public string TrackName { get; set; } = string.Empty;
|
|
public string CarName { get; set; } = string.Empty;
|
|
public DateTime StartDate { get; set; }
|
|
public DateTime EndDate { get; set; }
|
|
public double TargetTime { get; set; }
|
|
public int GoldReward { get; set; }
|
|
public int CashReward { get; set; }
|
|
public bool Active { get; set; } = true;
|
|
}
|
|
|
|
public class TimeTrialResult
|
|
{
|
|
public int Id { get; set; }
|
|
public int UserId { get; set; }
|
|
public int TimeTrialId { get; set; }
|
|
public double TimeSeconds { get; set; }
|
|
public DateTime SubmittedAt { get; set; }
|
|
public bool BeatTarget { get; set; }
|
|
public int GoldEarned { get; set; }
|
|
public int CashEarned { get; set; }
|
|
}
|
|
|
|
public class Car
|
|
{
|
|
public int Id { get; set; }
|
|
public string CarId { get; set; } = string.Empty; // Unique identifier like "porsche_911_gt3"
|
|
public string Name { get; set; } = string.Empty;
|
|
public string Manufacturer { get; set; } = string.Empty;
|
|
public string ClassType { get; set; } = string.Empty; // C, B, A, S, R
|
|
public int BasePerformanceRating { get; set; } // Base PR before upgrades
|
|
public int CashPrice { get; set; }
|
|
public int GoldPrice { get; set; }
|
|
public bool Available { get; set; } = true;
|
|
}
|
|
|
|
public class OwnedCar
|
|
{
|
|
public int Id { get; set; }
|
|
public int UserId { get; set; }
|
|
public string CarId { get; set; } = string.Empty;
|
|
public string CarName { get; set; } = string.Empty;
|
|
public string Manufacturer { get; set; } = string.Empty;
|
|
public string ClassType { get; set; } = string.Empty;
|
|
public int PerformanceRating { get; set; } // Current PR with upgrades
|
|
public int UpgradeLevel { get; set; } // 0-5
|
|
public string PurchasedUpgrades { get; set; } = string.Empty; // Comma-separated list
|
|
public DateTime PurchasedAt { get; set; } = DateTime.UtcNow;
|
|
}
|
|
|
|
public class CarUpgrade
|
|
{
|
|
public int Id { get; set; }
|
|
public string CarId { get; set; } = string.Empty;
|
|
public string UpgradeType { get; set; } = string.Empty; // engine, tires, suspension, brakes, drivetrain
|
|
public int Level { get; set; } // 1-5
|
|
public int CashCost { get; set; }
|
|
public int PerformanceIncrease { get; set; }
|
|
}
|
|
|
|
public class CareerProgress
|
|
{
|
|
public int Id { get; set; }
|
|
public int UserId { get; set; }
|
|
public string SeriesName { get; set; } = string.Empty; // e.g., "Road Collection", "Endurance Legends"
|
|
public string EventName { get; set; } = string.Empty; // e.g., "Brands Hatch GP Circuit"
|
|
public bool Completed { get; set; }
|
|
public int StarsEarned { get; set; } // 0-3
|
|
public double BestTime { get; set; }
|
|
public DateTime? CompletedAt { get; set; }
|
|
}
|