mirror of
https://github.com/maybe-finance/maybe.git
synced 2025-07-19 13:19:39 +02:00
* Transfer data model migration * Transfers and payment modeling and UI improvements * Fix CI * Transfer matching flow * Better UI for transfers * Auto transfer matching, approve, reject flow * Mark transfers created from form as confirmed * Account filtering * Excluded rejected transfers from calculations * Calculation tweaks with transfer exclusions * Clean up migration
189 lines
5.8 KiB
Ruby
189 lines
5.8 KiB
Ruby
module ApplicationHelper
|
|
include Pagy::Frontend
|
|
|
|
def date_format_options
|
|
[
|
|
[ "DD-MM-YYYY", "%d-%m-%Y" ],
|
|
[ "DD.MM.YYYY", "%d.%m.%Y" ],
|
|
[ "MM-DD-YYYY", "%m-%d-%Y" ],
|
|
[ "YYYY-MM-DD", "%Y-%m-%d" ],
|
|
[ "DD/MM/YYYY", "%d/%m/%Y" ],
|
|
[ "YYYY/MM/DD", "%Y/%m/%d" ],
|
|
[ "MM/DD/YYYY", "%m/%d/%Y" ],
|
|
[ "D/MM/YYYY", "%e/%m/%Y" ],
|
|
[ "YYYY.MM.DD", "%Y.%m.%d" ]
|
|
]
|
|
end
|
|
|
|
def title(page_title)
|
|
content_for(:title) { page_title }
|
|
end
|
|
|
|
def header_title(page_title)
|
|
content_for(:header_title) { page_title }
|
|
end
|
|
|
|
def family_notifications_stream
|
|
turbo_stream_from [ Current.family, :notifications ] if Current.family
|
|
end
|
|
|
|
def family_stream
|
|
turbo_stream_from Current.family if Current.family
|
|
end
|
|
|
|
def render_flash_notifications
|
|
notifications = flash.flat_map do |type, message_or_messages|
|
|
Array(message_or_messages).map do |message|
|
|
render partial: "shared/notification", locals: { type: type, message: message }
|
|
end
|
|
end
|
|
|
|
safe_join(notifications)
|
|
end
|
|
|
|
##
|
|
# Helper to open a centered and overlayed modal with custom contents
|
|
#
|
|
# @example Basic usage
|
|
# <%= modal classes: "custom-class" do %>
|
|
# <div>Content here</div>
|
|
# <% end %>
|
|
#
|
|
def modal(options = {}, &block)
|
|
content = capture &block
|
|
render partial: "shared/modal", locals: { content:, classes: options[:classes] }
|
|
end
|
|
|
|
##
|
|
# Helper to open a drawer on the right side of the screen with custom contents
|
|
#
|
|
# @example Basic usage
|
|
# <%= drawer do %>
|
|
# <div>Content here</div>
|
|
# <% end %>
|
|
#
|
|
def drawer(reload_on_close: false, &block)
|
|
content = capture &block
|
|
render partial: "shared/drawer", locals: { content:, reload_on_close: }
|
|
end
|
|
|
|
def disclosure(title, default_open: true, &block)
|
|
content = capture &block
|
|
render partial: "shared/disclosure", locals: { title: title, content: content, open: default_open }
|
|
end
|
|
|
|
def sidebar_link_to(name, path, options = {})
|
|
is_current = current_page?(path) || (request.path.start_with?(path) && path != "/")
|
|
|
|
classes = [
|
|
"flex items-center gap-2 px-3 py-2 rounded-xl border text-sm font-medium text-gray-500",
|
|
(is_current ? "bg-white text-gray-900 shadow-xs border-alpha-black-50" : "hover:bg-gray-100 border-transparent")
|
|
].compact.join(" ")
|
|
|
|
link_to path, **options.merge(class: classes), aria: { current: ("page" if current_page?(path)) } do
|
|
concat(lucide_icon(options[:icon], class: "w-5 h-5")) if options[:icon]
|
|
concat(name)
|
|
end
|
|
end
|
|
|
|
def mixed_hex_styles(hex)
|
|
color = hex || "#1570EF" # blue-600
|
|
|
|
<<-STYLE.strip
|
|
background-color: color-mix(in srgb, #{color} 10%, white);
|
|
border-color: color-mix(in srgb, #{color} 30%, white);
|
|
color: #{color};
|
|
STYLE
|
|
end
|
|
|
|
def circle_logo(name, hex: nil, size: "md")
|
|
render partial: "shared/circle_logo", locals: { name: name, hex: hex, size: size }
|
|
end
|
|
|
|
def return_to_path(params, fallback = root_path)
|
|
uri = URI.parse(params[:return_to] || fallback)
|
|
uri.relative? ? uri.path : root_path
|
|
end
|
|
|
|
def trend_styles(trend)
|
|
fallback = { bg_class: "bg-gray-500/5", text_class: "text-gray-500", symbol: "", icon: "minus" }
|
|
return fallback if trend.nil? || trend.direction.flat?
|
|
|
|
bg_class, text_class, symbol, icon = case trend.direction
|
|
when "up"
|
|
trend.favorable_direction.down? ? [ "bg-red-500/5", "text-red-500", "+", "arrow-up" ] : [ "bg-green-500/5", "text-green-500", "+", "arrow-up" ]
|
|
when "down"
|
|
trend.favorable_direction.down? ? [ "bg-green-500/5", "text-green-500", "-", "arrow-down" ] : [ "bg-red-500/5", "text-red-500", "-", "arrow-down" ]
|
|
when "flat"
|
|
[ "bg-gray-500/5", "text-gray-500", "", "minus" ]
|
|
else
|
|
raise ArgumentError, "Invalid trend direction: #{trend.direction}"
|
|
end
|
|
|
|
{ bg_class: bg_class, text_class: text_class, symbol: symbol, icon: icon }
|
|
end
|
|
|
|
# Wrapper around I18n.l to support custom date formats
|
|
def format_date(object, format = :default, options = {})
|
|
date = object.to_date
|
|
|
|
format_code = options[:format_code] || Current.family&.date_format
|
|
|
|
if format_code.present?
|
|
date.strftime(format_code)
|
|
else
|
|
I18n.l(date, format: format, **options)
|
|
end
|
|
end
|
|
|
|
def format_money(number_or_money, options = {})
|
|
return nil unless number_or_money
|
|
|
|
money = Money.new(number_or_money)
|
|
options.reverse_merge!(money.format_options(I18n.locale))
|
|
number_to_currency(money.amount, options)
|
|
end
|
|
|
|
def format_money_without_symbol(number_or_money, options = {})
|
|
return nil unless number_or_money
|
|
|
|
money = Money.new(number_or_money)
|
|
options.reverse_merge!(money.format_options(I18n.locale))
|
|
ActiveSupport::NumberHelper.number_to_delimited(money.amount.round(options[:precision] || 0), { delimiter: options[:delimiter], separator: options[:separator] })
|
|
end
|
|
|
|
def totals_by_currency(collection:, money_method:, separator: " | ", negate: false)
|
|
collection.group_by(&:currency)
|
|
.transform_values { |item| negate ? item.sum(&money_method) * -1 : item.sum(&money_method) }
|
|
.map { |_currency, money| format_money(money) }
|
|
.join(separator)
|
|
end
|
|
|
|
def show_super_admin_bar?
|
|
if params[:admin].present?
|
|
cookies.permanent[:admin] = params[:admin]
|
|
end
|
|
|
|
cookies[:admin] == "true"
|
|
end
|
|
|
|
def custom_pagy_url_for(pagy, page, current_path: nil)
|
|
if current_path.blank?
|
|
pagy_url_for(pagy, page)
|
|
else
|
|
uri = URI.parse(current_path)
|
|
params = URI.decode_www_form(uri.query || "").to_h
|
|
|
|
# Delete existing page param if it exists
|
|
params.delete("page")
|
|
# Add new page param unless it's page 1
|
|
params["page"] = page unless page == 1
|
|
|
|
if params.empty?
|
|
uri.path
|
|
else
|
|
"#{uri.path}?#{URI.encode_www_form(params)}"
|
|
end
|
|
end
|
|
end
|
|
end
|