First cut of a simplified "intro" UI layout (#265)

* First cut of a simplified "intro" UI layout

* Linter

* Add guest role and intro-only access

* Fix guest role UI defaults (#940)

Use enum predicate to avoid missing role helper.

* Remove legacy user role mapping (#941)

Drop the unused user role references in role normalization
and SSO role mapping forms to avoid implying a role that
never existed.

Refs: #0

* Remove role normalization (#942)

Remove role normalization

Roles are now stored directly without legacy mappings.

* Revert role mapping logic

* Remove `normalize_role_settings`

* Remove unnecessary migration

* Make `member` the default

* Broken `.erb`

---------

Signed-off-by: Juan José Mata <juanjo.mata@gmail.com>
This commit is contained in:
Juan José Mata
2026-02-09 11:09:25 +01:00
committed by GitHub
parent ba442d5f26
commit 705b5a8b26
33 changed files with 556 additions and 138 deletions

View File

@@ -9,6 +9,9 @@ class InvitationsControllerTest < ActionDispatch::IntegrationTest
test "should get new" do
get new_invitation_url
assert_response :success
assert_select "option[value=?]", "member"
assert_select "option[value=?]", "guest"
assert_select "option[value=?]", "admin"
end
test "should create invitation for member" do
@@ -89,6 +92,49 @@ class InvitationsControllerTest < ActionDispatch::IntegrationTest
assert_equal @admin, invitation.inviter
end
test "admin can create guest invitation" do
assert_difference("Invitation.count") do
post invitations_url, params: {
invitation: {
email: "intro-invite@example.com",
role: "guest"
}
}
end
invitation = Invitation.order(created_at: :desc).first
assert_equal "guest", invitation.role
assert_equal @admin.family, invitation.family
assert_equal @admin, invitation.inviter
end
test "inviting an existing user as guest applies intro defaults" do
existing_user = users(:empty)
existing_user.update!(
role: :member,
ui_layout: :dashboard,
show_sidebar: true,
show_ai_sidebar: true,
ai_enabled: false
)
assert_difference("Invitation.count") do
post invitations_url, params: {
invitation: {
email: existing_user.email,
role: "guest"
}
}
end
existing_user.reload
assert_equal "guest", existing_user.role
assert existing_user.ui_layout_intro?
assert_not existing_user.show_sidebar?
assert_not existing_user.show_ai_sidebar?
assert existing_user.ai_enabled?
end
test "should handle invalid invitation creation" do
assert_no_difference("Invitation.count") do
post invitations_url, params: {

View File

@@ -5,6 +5,7 @@ class PagesControllerTest < ActionDispatch::IntegrationTest
setup do
sign_in @user = users(:family_admin)
@intro_user = users(:intro_user)
@family = @user.family
end
@@ -13,6 +14,21 @@ class PagesControllerTest < ActionDispatch::IntegrationTest
assert_response :ok
end
test "intro page requires guest role" do
get intro_path
assert_redirected_to root_path
assert_equal "Intro is only available to guest users.", flash[:alert]
end
test "intro page is accessible for guest users" do
sign_in @intro_user
get intro_path
assert_response :ok
end
test "dashboard renders sankey chart with subcategories" do
# Create parent category with subcategory
parent_category = @family.categories.create!(name: "Shopping", classification: "expense", color: "#FF5733")

View File

@@ -67,4 +67,24 @@ class RegistrationsControllerTest < ActionDispatch::IntegrationTest
end
end
end
test "creating account from guest invitation assigns guest role and intro layout" do
invitation = invitations(:one)
invitation.update!(role: "guest", email: "guest-signup@example.com")
assert_difference "User.count", +1 do
post registration_url, params: { user: {
email: invitation.email,
password: "Password1!",
invitation: invitation.token
} }
end
created_user = User.find_by(email: invitation.email)
assert_equal "guest", created_user.role
assert created_user.ui_layout_intro?
assert_not created_user.show_sidebar?
assert_not created_user.show_ai_sidebar?
assert created_user.ai_enabled?
end
end

View File

@@ -4,6 +4,7 @@ class Settings::ProfilesControllerTest < ActionDispatch::IntegrationTest
setup do
@admin = users(:family_admin)
@member = users(:family_member)
@intro_user = users(:intro_user)
end
test "should get show" do
@@ -12,6 +13,19 @@ class Settings::ProfilesControllerTest < ActionDispatch::IntegrationTest
assert_response :success
end
test "intro user sees profile without settings navigation" do
sign_in @intro_user
get settings_profile_path
assert_response :success
assert_select "#mobile-settings-nav", count: 0
assert_select "h2", text: I18n.t("settings.profiles.show.household_title"), count: 0
assert_select "[data-action='app-layout#openMobileSidebar']", count: 0
assert_select "[data-action='app-layout#closeMobileSidebar']", count: 0
assert_select "[data-action='app-layout#toggleLeftSidebar']", count: 0
assert_select "[data-action='app-layout#toggleRightSidebar']", count: 0
end
test "admin can remove a family member" do
sign_in @admin
assert_difference("User.count", -1) do