From c4de253d929d6ff79c07f4e0f2a816e1b625b0b3 Mon Sep 17 00:00:00 2001 From: Rasmus Kjellberg <2277443+kjellberg@users.noreply.github.com> Date: Thu, 25 Apr 2024 13:14:01 +0200 Subject: [PATCH] feat: add a combined user and personal profile settings page - fixes #23 --- app/controllers/application_controller.rb | 6 ---- .../{preferences => settings}/_form.html.erb | 7 ++++- .../{preferences => settings}/edit.html.erb | 0 .../partials/navigations/_protected.html.erb | 2 +- .../partials/navigations/_settings.html.erb | 30 +++++++++--------- config/initializers/kiqr.rb | 5 +++ config/locales/kiqr.en.yml | 14 ++++----- .../controllers/kiqr/accounts_controller.rb | 13 +++++--- .../controllers/kiqr/onboarding_controller.rb | 6 +++- ...s_controller.rb => settings_controller.rb} | 17 +++++----- gems/kiqr/app/models/account.rb | 4 +++ gems/kiqr/app/models/user.rb | 1 + gems/kiqr/config/locales/kiqr/en.yml | 9 +++--- gems/kiqr/lib/kiqr/config.rb | 5 +++ gems/kiqr/lib/kiqr/rails/routes.rb | 6 ++-- .../kiqr/accounts_controller_test.rb | 20 ++++-------- .../kiqr/preferences_controller_test.rb | 20 ------------ .../kiqr/settings_controller_test.rb | 31 +++++++++++++++++++ test/system/accounts/accounts_test.rb | 17 ---------- test/system/users/settings_test.rb | 23 ++++++++++++++ 20 files changed, 135 insertions(+), 101 deletions(-) rename app/views/kiqr/{preferences => settings}/_form.html.erb (70%) rename app/views/kiqr/{preferences => settings}/edit.html.erb (100%) rename gems/kiqr/app/controllers/kiqr/{preferences_controller.rb => settings_controller.rb} (51%) delete mode 100644 test/controllers/kiqr/preferences_controller_test.rb create mode 100644 test/controllers/kiqr/settings_controller_test.rb create mode 100644 test/system/users/settings_test.rb diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb index e58c8b9..79f7715 100644 --- a/app/controllers/application_controller.rb +++ b/app/controllers/application_controller.rb @@ -6,12 +6,6 @@ class ApplicationController < ActionController::Base before_action :ensure_onboarded, unless: :devise_controller? before_action :setup_locale - # Strong parameters for account. - # Used for account creation and update. - def account_permitted_parameters - params.require(:account).permit(:name) - end - private # Automatically include account_id in all URL options if it is already present in the params. diff --git a/app/views/kiqr/preferences/_form.html.erb b/app/views/kiqr/settings/_form.html.erb similarity index 70% rename from app/views/kiqr/preferences/_form.html.erb rename to app/views/kiqr/settings/_form.html.erb index 03d2434..931ece9 100644 --- a/app/views/kiqr/preferences/_form.html.erb +++ b/app/views/kiqr/settings/_form.html.erb @@ -1,4 +1,9 @@ -<%= simple_form_for(user, url: user_preferences_path, method: :patch) do |f| %> +<%= simple_form_for(user, url: settings_path, method: :patch) do |f| %> + + <%= f.simple_fields_for :personal_account, user.personal_account do |pa| %> + <%= pa.input :name %> + <% end %> + <%= f.input :locale, label: t(".locale.label"), placeholder: t(".locale.placeholder"), required: true, autofocus: true, prompt: t(".locale.prompt"), as: :select, collection: options_for_locale %> <%= f.input :time_zone, label: t(".time_zone.label"), placeholder: t(".time_zone.placeholder"), required: true, autofocus: true %>
diff --git a/app/views/kiqr/preferences/edit.html.erb b/app/views/kiqr/settings/edit.html.erb similarity index 100% rename from app/views/kiqr/preferences/edit.html.erb rename to app/views/kiqr/settings/edit.html.erb diff --git a/app/views/partials/navigations/_protected.html.erb b/app/views/partials/navigations/_protected.html.erb index d04333a..9b6752d 100644 --- a/app/views/partials/navigations/_protected.html.erb +++ b/app/views/partials/navigations/_protected.html.erb @@ -2,7 +2,7 @@
<%= render(PageLayouts::Settings::NavigationItem::Component.new( - label: t('.items.preferences.label'), - description: t('.items.preferences.description'), + label: t('.items.user_profile.label'), + description: t('.items.user_profile.description'), icon: "fa fa-sliders", - path: edit_user_preferences_path, - active: current_base_path?(edit_user_preferences_path) + path: edit_settings_path, + active: current_base_path?(edit_settings_path) )) %> <%= render(PageLayouts::Settings::NavigationItem::Component.new( diff --git a/config/initializers/kiqr.rb b/config/initializers/kiqr.rb index e2f477d..e8bac0b 100644 --- a/config/initializers/kiqr.rb +++ b/config/initializers/kiqr.rb @@ -10,6 +10,11 @@ # with default "from" parameter. config.default_from_email = "please-change-me-at-config-initializers@example.com" + # ==> Account attributes + # Strong parameters for account. Used for account creation and update. + # Affects both personal and team accounts. + config.account_attributes = [:name] + # ==> Locales # Configure the available locales in the application. # This is used to validate the locale of the user. diff --git a/config/locales/kiqr.en.yml b/config/locales/kiqr.en.yml index c33353f..e992e09 100644 --- a/config/locales/kiqr.en.yml +++ b/config/locales/kiqr.en.yml @@ -11,21 +11,21 @@ en: sign_in: "Sign in" register: "Create account" settings: - account_settings: "Account settings" - user_settings: "Your personal settings" + account_settings: "Team settings" + user_settings: "User account" items: account: label: "Profile" - description: "Edit your account profile" + description: "Edit your team's profile" members: - label: "Team members" + label: "Members" description: "Manage team members" user: label: "Login credentials" description: "Change user email or password" - preferences: - label: "Preferences" - description: "Change your personal settings" + user_profile: + label: "Settings" + description: "Personal profile & settings" two_factor: label: "Two-factor authentication" description: "One-time passwords" diff --git a/gems/kiqr/app/controllers/kiqr/accounts_controller.rb b/gems/kiqr/app/controllers/kiqr/accounts_controller.rb index 45b9e07..6f48687 100644 --- a/gems/kiqr/app/controllers/kiqr/accounts_controller.rb +++ b/gems/kiqr/app/controllers/kiqr/accounts_controller.rb @@ -1,13 +1,13 @@ module Kiqr class AccountsController < KiqrController - before_action :setup_account, only: %i[edit update] + before_action :ensure_team_and_setup_account, only: %i[edit update] def new @account = Account.new end def create - @account = Account.new(account_permitted_parameters) + @account = Account.new(account_params) if @account.valid? Kiqr::Services::Accounts::Create.call!(account: @account, user: current_user) @@ -19,7 +19,7 @@ def create end def update - @account.assign_attributes(account_permitted_parameters) + @account.assign_attributes(account_params) if @account.valid? Kiqr::Services::Accounts::Update.call!(account: @account, user: current_user) @@ -32,8 +32,13 @@ def update private - def setup_account + def account_params + params.require(:account).permit(Kiqr.config.account_attributes) + end + + def ensure_team_and_setup_account @account = Account.find(current_account.id) + redirect_to edit_settings_path if @account.personal? end def form_submit_path diff --git a/gems/kiqr/app/controllers/kiqr/onboarding_controller.rb b/gems/kiqr/app/controllers/kiqr/onboarding_controller.rb index c0d59c7..e3b0030 100644 --- a/gems/kiqr/app/controllers/kiqr/onboarding_controller.rb +++ b/gems/kiqr/app/controllers/kiqr/onboarding_controller.rb @@ -22,7 +22,7 @@ def new end def create - @account = Account.new(account_permitted_parameters) + @account = Account.new(account_params) if @account.valid? Kiqr::Services::Accounts::Create.call!(account: @account, user: current_user, personal: true) @@ -35,6 +35,10 @@ def create private + def account_params + params.require(:account).permit(Kiqr.config.account_attributes) + end + # This is the path to redirect to after the onboarding process # is completed. By default, it redirects to the dashboard. def after_onboarding_path diff --git a/gems/kiqr/app/controllers/kiqr/preferences_controller.rb b/gems/kiqr/app/controllers/kiqr/settings_controller.rb similarity index 51% rename from gems/kiqr/app/controllers/kiqr/preferences_controller.rb rename to gems/kiqr/app/controllers/kiqr/settings_controller.rb index aedc5da..0afc72a 100644 --- a/gems/kiqr/app/controllers/kiqr/preferences_controller.rb +++ b/gems/kiqr/app/controllers/kiqr/settings_controller.rb @@ -1,13 +1,11 @@ module Kiqr - class PreferencesController < ApplicationController - def edit - @user = current_user - end + class SettingsController < KiqrController + before_action :setup_user, only: %i[edit update] def update - @user = current_user if @user.update(preferences_params) - redirect_to edit_user_preferences_path, notice: t(".updated") + kiqr_flash_message(:notice, :settings_updated) + redirect_to edit_settings_path else render :edit, status: :unprocessable_entity end @@ -15,6 +13,10 @@ def update private + def setup_user + @user = current_user + end + def options_for_locale I18n.available_locales.map do |locale| [I18n.t("languages.#{locale}"), locale] @@ -23,7 +25,8 @@ def options_for_locale helper_method :options_for_locale def preferences_params - params.require(:user).permit(:time_zone, :locale) + personal_account_attributes = Kiqr.config.account_attributes.prepend(:id) + params.require(:user).permit(:time_zone, :locale, personal_account_attributes: personal_account_attributes) end end end diff --git a/gems/kiqr/app/models/account.rb b/gems/kiqr/app/models/account.rb index 46946fa..a1cad03 100644 --- a/gems/kiqr/app/models/account.rb +++ b/gems/kiqr/app/models/account.rb @@ -16,4 +16,8 @@ class Account < ApplicationRecord def has_member?(user) users.include? user end + + def team? + !personal + end end diff --git a/gems/kiqr/app/models/user.rb b/gems/kiqr/app/models/user.rb index 4aad1b4..c47f8d1 100644 --- a/gems/kiqr/app/models/user.rb +++ b/gems/kiqr/app/models/user.rb @@ -8,6 +8,7 @@ class User < ApplicationRecord # User belongs to a personal account. belongs_to :personal_account, class_name: "Account", optional: true, dependent: :destroy validates_associated :personal_account + accepts_nested_attributes_for :personal_account # User can have many team accounts. has_many :account_users, dependent: :destroy diff --git a/gems/kiqr/config/locales/kiqr/en.yml b/gems/kiqr/config/locales/kiqr/en.yml index 8657ecc..d223ffd 100644 --- a/gems/kiqr/config/locales/kiqr/en.yml +++ b/gems/kiqr/config/locales/kiqr/en.yml @@ -10,6 +10,7 @@ en: invitation_expired: "The invitation has expired." account_user_destroyed: "Member has successfully been removed from the team." omniauth_email_taken: "An account with this email already exists. Please sign in to that account first and then link your %{provider} account." + settings_updated: "Your settings have been updated successfully." accounts: new: title: "Create a new team account" @@ -92,10 +93,10 @@ en: new: title: "Setup your personal account" description: "Finish the registration by setting up your user profile." - preferences: + settings: edit: - title: "User preferences" - description: "Update your user preferences." + title: "Settings" + description: "Personal profile & settings" form: time_zone: label: "Time zone" @@ -105,8 +106,6 @@ en: prompt: "Select a language" placeholder: "Select your preferred language" submit_button: "Save changes" - update: - updated: "User preferences have been updated." registrations: new: heading: diff --git a/gems/kiqr/lib/kiqr/config.rb b/gems/kiqr/lib/kiqr/config.rb index 365acde..4af930d 100644 --- a/gems/kiqr/lib/kiqr/config.rb +++ b/gems/kiqr/lib/kiqr/config.rb @@ -13,6 +13,11 @@ class Config # with default "from" parameter. config_accessor :default_from_email, default: "please-change-me-at-config-initializers@example.com" + # ==> Account attributes + # Strong parameters for account. Used for account creation and update. + # Affects both personal and team accounts. + config_accessor :account_attributes, default: [:name] + # ==> Locales # Configure the available locales in the application. # This is used to validate the locale of the user. diff --git a/gems/kiqr/lib/kiqr/rails/routes.rb b/gems/kiqr/lib/kiqr/rails/routes.rb index e877b1a..fd33117 100644 --- a/gems/kiqr/lib/kiqr/rails/routes.rb +++ b/gems/kiqr/lib/kiqr/rails/routes.rb @@ -19,7 +19,7 @@ def kiqr_routes(options = {}) get "two-factor/disable", controller: options[:controllers][:two_factor], action: "disable", as: :disable_two_factor post "two-factor/verify", controller: options[:controllers][:two_factor], action: "verify", as: :verify_two_factor delete "two-factor/destroy", controller: options[:controllers][:two_factor], action: "destroy", as: :destroy_two_factor - resource :preferences, controller: options[:controllers][:preferences], only: %i[edit update], as: :user_preferences + resource :settings, controller: options[:controllers][:settings], only: %i[edit update], as: :settings end end @@ -36,7 +36,7 @@ def default_controllers(options) options[:controllers][:registrations] ||= "kiqr/registrations" options[:controllers][:sessions] ||= "kiqr/sessions" options[:controllers][:two_factor] ||= "kiqr/two_factor" - options[:controllers][:preferences] ||= "kiqr/preferences" + options[:controllers][:settings] ||= "kiqr/settings" options[:controllers][:omniauth] ||= "kiqr/omniauth" options end @@ -59,7 +59,7 @@ def account_routes(options) get "select-account", controller: options[:controllers][:accounts], action: :select, as: :select_account scope "(/team/:account_id)", account_id: %r{[^/]+} do - resource :account, only: [:edit, :update], path: "profile", controller: options[:controllers][:accounts] + resource :account, only: [:edit, :update], controller: options[:controllers][:accounts] resources :account_users, controller: options[:controllers][:account_users], only: [:index, :edit, :update, :destroy], path: "members" end end diff --git a/test/controllers/kiqr/accounts_controller_test.rb b/test/controllers/kiqr/accounts_controller_test.rb index 9f3a634..f6ccea4 100644 --- a/test/controllers/kiqr/accounts_controller_test.rb +++ b/test/controllers/kiqr/accounts_controller_test.rb @@ -7,10 +7,10 @@ class Kiqr::AccountsControllerTest < ActionDispatch::IntegrationTest assert_response :success end - test "can render edit account page" do + test "redirect to profile edit if personal account" do sign_in create(:user) get edit_account_path - assert_response :success + assert_redirected_to edit_settings_path end test "can create new team account" do @@ -33,17 +33,6 @@ class Kiqr::AccountsControllerTest < ActionDispatch::IntegrationTest # assert_template :new end - test "can update personal accounts" do - user = create(:user) - sign_in user - - patch account_path, params: {account: {name: "New name"}} - user.reload - - assert_redirected_to edit_account_path - assert_equal "New name", user.personal_account.name - end - test "can update team accounts" do user = create(:user) account = create(:account, name: "Team account") @@ -59,9 +48,12 @@ class Kiqr::AccountsControllerTest < ActionDispatch::IntegrationTest test "re-renders form if form data is invalid" do user = create(:user) + account = create(:account, name: "Team account") + account.account_users << AccountUser.create(user:, owner: true) + sign_in user - patch account_path, params: {account: {name: "no"}} + patch account_path(account_id: account), params: {account: {name: "no"}} assert_response :unprocessable_entity end end diff --git a/test/controllers/kiqr/preferences_controller_test.rb b/test/controllers/kiqr/preferences_controller_test.rb deleted file mode 100644 index 57e0f8f..0000000 --- a/test/controllers/kiqr/preferences_controller_test.rb +++ /dev/null @@ -1,20 +0,0 @@ -require "test_helper" - -class Kiqr::PreferencesControllerTest < ActionDispatch::IntegrationTest - test "should get edit page" do - user = create(:user) - sign_in(user) - get edit_user_preferences_path - assert_response :success - end - - test "can update user preferences" do - user = create(:user, time_zone: "UTC", locale: "en") - sign_in(user) - - patch user_preferences_path, params: {user: {time_zone: "Stockholm", locale: "sv"}} - assert_redirected_to edit_user_preferences_path - assert_equal "Stockholm", user.reload.time_zone - assert_equal "sv", user.reload.locale - end -end diff --git a/test/controllers/kiqr/settings_controller_test.rb b/test/controllers/kiqr/settings_controller_test.rb new file mode 100644 index 0000000..a16a846 --- /dev/null +++ b/test/controllers/kiqr/settings_controller_test.rb @@ -0,0 +1,31 @@ +require "test_helper" + +class Kiqr::PreferencesControllerTest < ActionDispatch::IntegrationTest + test "should get edit page" do + user = create(:user) + sign_in(user) + get edit_settings_path + assert_response :success + end + + test "can update user fields" do + user = create(:user, time_zone: "UTC", locale: "en") + sign_in(user) + + patch settings_path, params: {user: {time_zone: "Stockholm", locale: "sv"}} + assert_redirected_to edit_settings_path + assert_equal "Stockholm", user.reload.time_zone + assert_equal "sv", user.reload.locale + end + + test "can update personal account fields" do + user = create(:user) + sign_in user + + patch settings_path, params: {user: {personal_account_attributes: {name: "Personal account name"}}} + user.reload + + assert_redirected_to edit_settings_path + assert_equal "Personal account name", user.personal_account.name + end +end diff --git a/test/system/accounts/accounts_test.rb b/test/system/accounts/accounts_test.rb index f367e16..4613b72 100644 --- a/test/system/accounts/accounts_test.rb +++ b/test/system/accounts/accounts_test.rb @@ -1,23 +1,6 @@ require "application_system_test_case" class EditAccountsTest < ApplicationSystemTestCase - test "can edit personal account" do - user = create(:user) - - sign_in(user) - visit edit_account_path - assert_selector("input[name='account[name]']") - - # Fill the personal account form - fill_in_account_fields - - click_on "commit" - assert_text I18n.t("kiqr.flash_messages.account_updated") - - user.personal_account.reload - assert_equal "New name", user.personal_account.name - end - test "can edit team account" do user = create(:user) team_account = create(:account, name: "Team account") diff --git a/test/system/users/settings_test.rb b/test/system/users/settings_test.rb new file mode 100644 index 0000000..28ed6a6 --- /dev/null +++ b/test/system/users/settings_test.rb @@ -0,0 +1,23 @@ +require "application_system_test_case" + +class SettingsTest < ApplicationSystemTestCase + test "can edit user and personal account" do + user = create(:user) + + sign_in(user) + visit edit_settings_path + + # Fill the personal account form + select "Swedish", from: "user[locale]" + fill_in "user[personal_account_attributes][name]", with: "New name" + + click_on "commit" + assert_text I18n.t("kiqr.flash_messages.settings_updated") + + user.reload + + assert_current_path edit_settings_path + assert_equal "New name", user.personal_account.name + assert_equal "sv", user.locale + end +end