Files
sure/app/controllers/goal_pledges_controller.rb
Guillem Arias e16d9c4a6a Merge origin/feat/goals-v2-architecture; reconcile beta→preview rename
Remote branch added a beta_gated_nav_item helper + 'Gating the main nav'
docs section. Main concurrently renamed the beta-features gate to
preview-features (concern, predicate, JSONB key, locale flash). Rename
the new helper / partial local / pill marker to match preview naming and
port the nav-gating docs into gating-a-preview-feature.md so the
improvement survives the rename.

Resolved conflicts:
- db/schema.rb: take the later schema version (2026_05_19_100000).
- docs/llm-guides/gating-a-beta-feature.md: accept main's deletion;
  port the 'Gating the main nav' section into the preview guide.

Renames carried through to keep the gate wired end-to-end:
- application_helper.rb: beta_gated_nav_item → preview_gated_nav_item;
  beta_features_enabled? → preview_features_enabled?; beta: → preview:.
- _nav_item.html.erb: beta: local → preview: local; shared.beta i18n
  key → shared.preview.
- application.html.erb: caller renamed to preview_gated_nav_item.
- goals/index.html.erb: pill label uses shared.preview.
- shared/en.yml: 'beta: Beta' → 'preview: Preview'.
- goals_controller, goal_pledges_controller: require_beta_features! →
  require_preview_features!.
- goals_controller_test, goal_pledges_controller_test: flip the
  preference key, flash matcher, and test names to 'preview'.
2026-05-20 21:47:27 +02:00

90 lines
2.7 KiB
Ruby

class GoalPledgesController < ApplicationController
before_action :require_preview_features!
before_action :set_goal
before_action :set_pledge, only: %i[renew destroy]
rescue_from ActiveRecord::RecordNotFound, with: :record_not_found
def new
# The form is dialog-only. A direct GET (F5, bookmark, deep-link
# gone stale) lands the user back on the goal show page with the
# modal auto-opened via the catch-up CTA params, rather than
# rendering a freestanding DS::Dialog over an empty page.
unless turbo_frame_request?
redirect_to goal_path(@goal) and return
end
account = preselected_account
@pledge = @goal.goal_pledges.new(
currency: @goal.currency,
account: account,
kind: account&.default_pledge_kind || "transfer",
amount: params[:amount].presence
)
end
def create
@pledge = @goal.goal_pledges.new(pledge_params)
@pledge.account = lookup_account(params.dig(:goal_pledge, :account_id))
@pledge.kind = @pledge.account&.default_pledge_kind || "transfer"
@pledge.currency = @goal.currency
if @pledge.save
flash[:notice] = t(".success")
respond_to do |format|
format.html { redirect_to goal_path(@goal) }
format.turbo_stream do
render turbo_stream: turbo_stream.action(:redirect, goal_path(@goal))
end
end
else
render :new, status: :unprocessable_entity
end
end
def renew
@pledge.extend!
redirect_to goal_path(@goal), notice: t(".success")
rescue GoalPledge::NotOpenError
redirect_to goal_path(@goal), alert: t(".not_open")
end
def destroy
@pledge.cancel!
redirect_to goal_path(@goal), notice: t(".success")
rescue GoalPledge::NotOpenError
redirect_to goal_path(@goal), alert: t(".not_open")
end
private
def set_goal
# Preload linked accounts + their providers so any_connected_account?
# and the new-pledge form's per-account helpers don't trigger N+1
# queries on account_providers.
@goal = Current.family.goals
.includes(:open_pledges, linked_accounts: :account_providers)
.find(params[:goal_id])
end
def set_pledge
@pledge = @goal.goal_pledges.find(params[:id])
end
def pledge_params
params.require(:goal_pledge).permit(:amount)
end
def lookup_account(id)
return nil if id.blank?
@goal.linked_accounts.find_by(id: id)
end
def preselected_account
requested = params[:account_id].presence && @goal.linked_accounts.find_by(id: params[:account_id])
requested || @goal.linked_accounts.first
end
def record_not_found
redirect_to goals_path, alert: t("goals.errors.not_found")
end
end