Files
budget/src/Budget.Api/Controllers/IncomesController.cs
T
Spencer Twaddle f429a747d8 Phase 4: Income API and page
- Add FrequencyCalculator static utility (all 9 frequency multipliers)
- Add IncomesController: list, create, update, delete, reorder
- Add Income DTOs with computed Monthly/Annually fields
- Add shared TypeScript types (IncomeDto, OutgoDto, BudgetDto, ShareDto, SummaryDto)
- Add API client with Bearer token injection via setTokenProvider
- Add FrequencySelect, MoneyDisplay, BudgetNav shared components
- Add Income page: sortable table with inline editing, add/delete rows, drag-to-reorder via dnd-kit
- Wire TokenWirer in App.tsx to keep API client in sync with auth state

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-25 07:56:42 -05:00

93 lines
3.3 KiB
C#

using Budget.Api.Data;
using Budget.Api.DTOs;
using Budget.Api.Models;
using Budget.Api.Services;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
namespace Budget.Api.Controllers;
[ApiController]
[Route("api/budgets/{budgetId:guid}/incomes")]
[Authorize]
public class IncomesController(AppDbContext db, BudgetAuthorizationService authz) : ControllerBase
{
private string UserId => User.FindFirst("sub")!.Value;
private static IncomeDto ToDto(Income i) => new(
i.Id, i.BudgetId, i.Name, i.Frequency, i.Amount,
FrequencyCalculator.ToMonthly(i.Amount, i.Frequency),
FrequencyCalculator.ToAnnually(i.Amount, i.Frequency),
i.SortOrder);
[HttpGet]
public async Task<IActionResult> List(Guid budgetId)
{
if (!await authz.CanReadAsync(budgetId, UserId)) return Forbid();
var incomes = await db.Incomes
.Where(i => i.BudgetId == budgetId)
.OrderBy(i => i.SortOrder)
.ToListAsync();
return Ok(incomes.Select(ToDto));
}
[HttpPost]
public async Task<IActionResult> Create(Guid budgetId, [FromBody] CreateIncomeRequest req)
{
if (!await authz.CanWriteAsync(budgetId, UserId)) return Forbid();
var maxOrder = await db.Incomes.Where(i => i.BudgetId == budgetId).MaxAsync(i => (int?)i.SortOrder) ?? -1;
var income = new Income
{
Id = Guid.NewGuid(),
BudgetId = budgetId,
Name = req.Name,
Frequency = req.Frequency,
Amount = req.Amount,
SortOrder = maxOrder + 1,
};
db.Incomes.Add(income);
await db.SaveChangesAsync();
return Ok(ToDto(income));
}
[HttpPut("{incomeId:guid}")]
public async Task<IActionResult> Update(Guid budgetId, Guid incomeId, [FromBody] UpdateIncomeRequest req)
{
if (!await authz.CanWriteAsync(budgetId, UserId)) return Forbid();
var income = await db.Incomes.FirstOrDefaultAsync(i => i.Id == incomeId && i.BudgetId == budgetId);
if (income is null) return NotFound();
income.Name = req.Name;
income.Frequency = req.Frequency;
income.Amount = req.Amount;
await db.SaveChangesAsync();
return Ok(ToDto(income));
}
[HttpDelete("{incomeId:guid}")]
public async Task<IActionResult> Delete(Guid budgetId, Guid incomeId)
{
if (!await authz.CanWriteAsync(budgetId, UserId)) return Forbid();
var income = await db.Incomes.FirstOrDefaultAsync(i => i.Id == incomeId && i.BudgetId == budgetId);
if (income is null) return NotFound();
db.Incomes.Remove(income);
await db.SaveChangesAsync();
return NoContent();
}
[HttpPut("order")]
public async Task<IActionResult> Reorder(Guid budgetId, [FromBody] ReorderIncomesRequest req)
{
if (!await authz.CanWriteAsync(budgetId, UserId)) return Forbid();
var incomes = await db.Incomes.Where(i => i.BudgetId == budgetId).ToListAsync();
var lookup = incomes.ToDictionary(i => i.Id);
for (int idx = 0; idx < req.OrderedIds.Count; idx++)
{
if (lookup.TryGetValue(req.OrderedIds[idx], out var income))
income.SortOrder = idx;
}
await db.SaveChangesAsync();
return NoContent();
}
}