1
0
Fork 0
mirror of https://github.com/maybe-finance/maybe.git synced 2025-07-19 05:09:38 +02:00

Add confirmation dialog for balance reconciliation creates and updates (#2457)
Some checks are pending
Publish Docker image / ci (push) Waiting to run
Publish Docker image / Build docker image (push) Blocked by required conditions

This commit is contained in:
Zach Gollwitzer 2025-07-15 18:58:40 -04:00 committed by GitHub
parent c1d98fe73b
commit 89cc64418e
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
13 changed files with 180 additions and 49 deletions

View file

@ -1,15 +1,24 @@
class ValuationsController < ApplicationController
include EntryableResource, StreamExtensions
def confirm_create
@account = Current.family.accounts.find(params.dig(:entry, :account_id))
@entry = @account.entries.build(entry_params.merge(currency: @account.currency))
render :confirm_create
end
def confirm_update
@entry = Current.family.entries.find(params[:id])
@account = @entry.account
@entry.assign_attributes(entry_params.merge(currency: @account.currency))
render :confirm_update
end
def create
account = Current.family.accounts.find(params.dig(:entry, :account_id))
result = account.update_balance(
balance: entry_params[:amount],
date: entry_params[:date],
currency: entry_params[:currency],
notes: entry_params[:notes]
)
result = perform_balance_update(account, entry_params.merge(currency: account.currency))
if result.success?
@success_message = result.updated? ? "Balance updated" : "No changes made. Account is already up to date."
@ -25,12 +34,7 @@ class ValuationsController < ApplicationController
end
def update
result = @entry.account.update_balance(
date: @entry.date,
balance: entry_params[:amount],
currency: entry_params[:currency],
notes: entry_params[:notes]
)
result = perform_balance_update(@entry.account, entry_params.merge(currency: @entry.currency, existing_valuation_id: @entry.id))
if result.success?
@entry.reload
@ -59,4 +63,14 @@ class ValuationsController < ApplicationController
params.require(:entry)
.permit(:date, :amount, :currency, :notes)
end
def perform_balance_update(account, params)
account.update_balance(
balance: params[:amount],
date: params[:date],
currency: params[:currency],
notes: params[:notes],
existing_valuation_id: params[:existing_valuation_id]
)
end
end

View file

@ -115,8 +115,8 @@ class Account < ApplicationRecord
end
def update_balance(balance:, date: Date.current, currency: nil, notes: nil)
Account::BalanceUpdater.new(self, balance:, currency:, date:, notes:).update
def update_balance(balance:, date: Date.current, currency: nil, notes: nil, existing_valuation_id: nil)
Account::BalanceUpdater.new(self, balance:, currency:, date:, notes:, existing_valuation_id:).update
end
def start_date

View file

@ -1,10 +1,11 @@
class Account::BalanceUpdater
def initialize(account, balance:, currency: nil, date: Date.current, notes: nil)
def initialize(account, balance:, currency: nil, date: Date.current, notes: nil, existing_valuation_id: nil)
@account = account
@balance = balance.to_d
@currency = currency
@date = date.to_date
@notes = notes
@existing_valuation_id = existing_valuation_id
end
def update
@ -17,10 +18,15 @@ class Account::BalanceUpdater
account.save!
end
valuation_entry = account.entries.valuations.find_or_initialize_by(date: date) do |entry|
entry.entryable = Valuation.new(kind: "reconciliation")
valuation_entry = if existing_valuation_id
account.entries.find(existing_valuation_id)
else
account.entries.valuations.find_or_initialize_by(date: date) do |entry|
entry.entryable = Valuation.new(kind: "reconciliation")
end
end
valuation_entry.date = date
valuation_entry.amount = balance
valuation_entry.currency = currency if currency.present?
valuation_entry.name = Valuation.build_reconciliation_name(account.accountable_type)
@ -37,7 +43,7 @@ class Account::BalanceUpdater
end
private
attr_reader :account, :balance, :currency, :date, :notes
attr_reader :account, :balance, :currency, :date, :notes, :existing_valuation_id
Result = Struct.new(:success?, :updated?, :error_message)

View file

@ -28,4 +28,19 @@ class Investment < ApplicationRecord
"line-chart"
end
end
def holdings_value_for_date(date)
# Find the most recent holding for each security on or before the given date
# Using a subquery to get the max date for each security
account.holdings
.where(currency: account.currency)
.where("date <= ?", date)
.where("(security_id, date) IN (
SELECT security_id, MAX(date) as max_date
FROM holdings
WHERE account_id = ? AND date <= ?
GROUP BY security_id
)", account.id, date)
.sum(:amount)
end
end

View file

@ -6,7 +6,7 @@
</div>
<div class="flex items-center gap-1 text-secondary">
<%= form_with url: transactions_bulk_deletion_path, data: { turbo_confirm: true, turbo_frame: "_top" } do %>
<%= form_with url: transactions_bulk_deletion_path, data: { turbo_confirm: CustomConfirm.for_resource_deletion("entry").to_data_attribute, turbo_frame: "_top" } do %>
<button type="button" data-bulk-select-scope-param="bulk_delete" data-action="bulk-select#submitBulkRequest" class="p-1.5 group hover:bg-inverse flex items-center justify-center rounded-md" title="Delete">
<%= icon "trash-2", class: "group-hover:text-inverse" %>
</button>

View file

@ -0,0 +1,49 @@
<div class="space-y-4 text-sm text-secondary">
<% if account.investment? %>
<% holdings_value = account.investment.holdings_value_for_date(entry.date) %>
<% brokerage_cash = entry.amount - holdings_value %>
<p>This will <%= action_verb %> the account value on <span class="font-medium text-primary"><%= entry.date.strftime("%B %d, %Y") %></span> to:</p>
<div class="bg-container rounded-lg p-4 space-y-2 border border-primary">
<div class="flex justify-between">
<span>Total account value</span>
<span class="font-medium text-primary"><%= entry.amount_money.format %></span>
</div>
<div class="flex justify-between text-xs">
<span>Holdings value</span>
<span><%= Money.new(holdings_value, account.currency).format %></span>
</div>
<div class="flex justify-between text-xs">
<span>Brokerage cash</span>
<span class="<%= brokerage_cash.negative? ? "text-red-500" : "text-green-500" %>"><%= Money.new(brokerage_cash, account.currency).format %></span>
</div>
</div>
<% else %>
<p><%= action_verb.capitalize %>
<% if account.depository? %>
account balance
<% elsif account.credit_card? %>
credit card balance
<% elsif account.loan? %>
loan balance
<% elsif account.property? %>
property value
<% elsif account.vehicle? %>
vehicle value
<% elsif account.crypto? %>
crypto balance
<% elsif account.other_asset? %>
asset value
<% elsif account.other_liability? %>
liability balance
<% else %>
balance
<% end %>
on <span class="font-medium text-primary"><%= entry.date.strftime("%B %d, %Y") %></span> to
<span class="font-medium text-primary"><%= entry.amount_money.format %></span>.
</p>
<% end %>
<p>All future transactions and balances will be recalculated based on this <%= is_update ? "change" : "update" %>.</p>
</div>

View file

@ -1,17 +0,0 @@
<%# locals: (entry:, error_message:) %>
<%= styled_form_with model: entry, url: valuations_path, class: "space-y-4" do |form| %>
<%= form.hidden_field :account_id %>
<% if error_message.present? %>
<%= render AlertComponent.new(message: error_message, variant: :error) %>
<% end %>
<div class="space-y-3">
<%= form.hidden_field :name, value: "Balance update" %>
<%= form.date_field :date, label: true, required: true, value: Date.current, min: Entry.min_supported_date, max: Date.current %>
<%= form.money_field :amount, label: t(".amount"), required: true %>
</div>
<%= form.submit t(".submit") %>
<% end %>

View file

@ -0,0 +1,20 @@
<%= render DialogComponent.new do |dialog| %>
<% dialog.with_header(title: "Confirm new balance") %>
<% dialog.with_body do %>
<%= styled_form_with model: @entry, url: valuations_path, class: "space-y-4", data: { turbo: false } do |form| %>
<%= form.hidden_field :account_id %>
<%= form.hidden_field :date %>
<%= form.hidden_field :amount %>
<%= form.hidden_field :currency %>
<%= form.hidden_field :notes %>
<%= render "confirmation_contents",
account: @account,
entry: @entry,
action_verb: "set",
is_update: false %>
<%= form.submit "Confirm" %>
<% end %>
<% end %>
<% end %>

View file

@ -0,0 +1,19 @@
<%= render DialogComponent.new do |dialog| %>
<% dialog.with_header(title: "Update balance") %>
<% dialog.with_body do %>
<%= styled_form_with model: @entry, url: valuation_path(@entry), method: :patch, class: "space-y-4", data: { turbo: false } do |form| %>
<%= form.hidden_field :date %>
<%= form.hidden_field :amount %>
<%= form.hidden_field :currency %>
<%= form.hidden_field :notes %>
<%= render "confirmation_contents",
account: @account,
entry: @entry,
action_verb: "update",
is_update: true %>
<%= form.submit "Update" %>
<% end %>
<% end %>
<% end %>

View file

@ -1,6 +1,19 @@
<%= render DialogComponent.new do |dialog| %>
<% dialog.with_header(title: t(".title")) %>
<% dialog.with_body do %>
<%= render "form", entry: @entry, error_message: @error_message %>
<%= styled_form_with model: @entry, url: confirm_create_valuations_path, class: "space-y-4" do |form| %>
<%= form.hidden_field :account_id %>
<% if @error_message.present? %>
<%= render AlertComponent.new(message: @error_message, variant: :error) %>
<% end %>
<div class="space-y-3">
<%= form.date_field :date, label: true, required: true, value: Date.current, min: Entry.min_supported_date, max: Date.current %>
<%= form.money_field :amount, label: t(".amount"), required: true, disable_currency: true %>
</div>
<%= form.submit t(".submit") %>
<% end %>
<% end %>
<% end %>

View file

@ -15,18 +15,25 @@
<% dialog.with_section(title: t(".overview"), open: true) do %>
<div class="pb-4">
<%= styled_form_with model: entry,
url: entry_path(entry),
class: "space-y-2",
data: { controller: "auto-submit-form" } do |f| %>
url: confirm_update_valuation_path(entry),
method: :post,
data: { turbo_frame: :modal },
class: "space-y-4" do |f| %>
<%= f.date_field :date,
label: t(".date_label"),
max: Date.current,
"data-auto-submit-form-target": "auto" %>
max: Date.current %>
<%= f.money_field :amount,
label: t(".amount"),
auto_submit: true,
disable_currency: true %>
<div class="flex justify-end">
<%= render ButtonComponent.new(
text: "Update value",
variant: :primary,
type: "submit"
) %>
</div>
<% end %>
</div>
<% end %>
@ -34,9 +41,13 @@
<% dialog.with_section(title: t(".details")) do %>
<div class="pb-4">
<%= styled_form_with model: entry,
url: entry_path(entry),
url: valuation_path(entry),
method: :patch,
class: "space-y-2",
data: { controller: "auto-submit-form" } do |f| %>
<%= f.hidden_field :date, value: entry.date %>
<%= f.hidden_field :amount, value: entry.amount %>
<%= f.hidden_field :currency, value: entry.currency %>
<%= f.text_area :notes,
label: t(".note_label"),
placeholder: t(".note_placeholder"),
@ -59,7 +70,7 @@
entry_path(entry),
method: :delete,
class: "rounded-lg px-3 py-2 text-red-500 text-sm font-medium border border-secondary",
data: { turbo_confirm: true, turbo_frame: "_top" } %>
data: { turbo_confirm: CustomConfirm.for_resource_deletion("value update").to_data_attribute, turbo_frame: "_top" } %>
</div>
</div>
<% end %>

View file

@ -110,7 +110,10 @@ Rails.application.routes.draw do
resources :holdings, only: %i[index new show destroy]
resources :trades, only: %i[show new create update destroy]
resources :valuations, only: %i[show new create update destroy]
resources :valuations, only: %i[show new create update destroy] do
post :confirm_create, on: :collection
post :confirm_update, on: :member
end
namespace :transactions do
resource :bulk_deletion, only: :create

View file

@ -15,7 +15,6 @@ class ValuationsControllerTest < ActionDispatch::IntegrationTest
post valuations_url, params: {
entry: {
amount: account.balance + 100,
currency: "USD",
date: Date.current.to_s,
account_id: account.id
}
@ -37,7 +36,6 @@ class ValuationsControllerTest < ActionDispatch::IntegrationTest
patch valuation_url(@entry), params: {
entry: {
amount: 20000,
currency: "USD",
date: Date.current
}
}