mirror of
https://github.com/scinote-eln/scinote-web.git
synced 2025-01-04 06:36:27 +08:00
548 lines
14 KiB
Ruby
548 lines
14 KiB
Ruby
class Users::SettingsController < ApplicationController
|
|
include UsersGenerator
|
|
|
|
before_action :load_user, only: [
|
|
:preferences,
|
|
:update_preferences,
|
|
:organizations,
|
|
:organization,
|
|
:create_organization,
|
|
:organization_users_datatable,
|
|
:tutorial,
|
|
:reset_tutorial
|
|
]
|
|
|
|
before_action :check_organization_permission, only: [
|
|
:organization,
|
|
:update_organization,
|
|
:destroy_organization,
|
|
:organization_name,
|
|
:organization_description,
|
|
:search_organization_users,
|
|
:organization_users_datatable,
|
|
:create_user_and_user_organization
|
|
]
|
|
|
|
before_action :check_create_user_organization_permission, only: [
|
|
:create_user_organization
|
|
]
|
|
|
|
before_action :check_user_organization_permission, only: [
|
|
:update_user_organization,
|
|
:leave_user_organization_html,
|
|
:destroy_user_organization_html,
|
|
:destroy_user_organization
|
|
]
|
|
|
|
def preferences
|
|
end
|
|
|
|
def update_preferences
|
|
respond_to do |format|
|
|
if @user.update(update_preferences_params)
|
|
flash[:notice] = t("users.settings.preferences.update_flash")
|
|
format.json {
|
|
flash.keep
|
|
render json: { status: :ok }
|
|
}
|
|
else
|
|
format.json {
|
|
render json: @user.errors,
|
|
status: :unprocessable_entity
|
|
}
|
|
end
|
|
end
|
|
end
|
|
|
|
def organizations
|
|
@user_orgs =
|
|
@user
|
|
.user_organizations
|
|
.includes(organization: :users)
|
|
.order(created_at: :asc)
|
|
@member_of = @user_orgs.count
|
|
end
|
|
|
|
def organization
|
|
@user_org = UserOrganization.find_by(user: @user, organization: @org)
|
|
end
|
|
|
|
def update_organization
|
|
respond_to do |format|
|
|
if @org.update(update_organization_params)
|
|
@org.update(last_modified_by: current_user)
|
|
format.json {
|
|
render json: {
|
|
status: :ok,
|
|
description_label: render_to_string(
|
|
partial: "users/settings/organizations/description_label.html.erb",
|
|
locals: { org: @org }
|
|
)
|
|
}
|
|
}
|
|
else
|
|
format.json {
|
|
render json: @org.errors,
|
|
status: :unprocessable_entity
|
|
}
|
|
end
|
|
end
|
|
end
|
|
|
|
def organization_name
|
|
respond_to do |format|
|
|
format.json {
|
|
render json: {
|
|
html: render_to_string({
|
|
partial: "users/settings/organizations/name_modal_body.html.erb",
|
|
locals: { org: @org }
|
|
})
|
|
}
|
|
}
|
|
end
|
|
end
|
|
|
|
def organization_description
|
|
respond_to do |format|
|
|
format.json {
|
|
render json: {
|
|
html: render_to_string({
|
|
partial: "users/settings/organizations/description_modal_body.html.erb",
|
|
locals: { org: @org }
|
|
})
|
|
}
|
|
}
|
|
end
|
|
end
|
|
|
|
def search_organization_users
|
|
respond_to do |format|
|
|
format.json {
|
|
if params.include? :existing_query and
|
|
(query = params[:existing_query].strip()).present?
|
|
if query.length < 3
|
|
render json: {
|
|
"existing_query": [
|
|
I18n.t("users.settings.organizations.edit.modal_add_user.existing_query_too_short")
|
|
]},
|
|
status: :unprocessable_entity
|
|
else
|
|
# Okay, query exists and is non-blank, find users
|
|
nr_of_results = User.search(true, query, @org).count
|
|
|
|
|
|
users = User.search(false, query, @org).limit(5)
|
|
|
|
nr_of_members = User.organization_search(false, query, @org).count
|
|
|
|
render json: {
|
|
html: render_to_string({
|
|
partial: "users/settings/organizations/existing_users_search_results.html.erb",
|
|
locals: {
|
|
users: users,
|
|
nr_of_results: nr_of_results,
|
|
nr_of_members: nr_of_members,
|
|
org: @org,
|
|
query: query
|
|
}
|
|
})
|
|
}
|
|
end
|
|
else
|
|
render json: {
|
|
"existing_query": [
|
|
I18n.t("users.settings.organizations.edit.modal_add_user.existing_query_blank")
|
|
]},
|
|
status: :unprocessable_entity
|
|
end
|
|
}
|
|
end
|
|
end
|
|
|
|
def organization_users_datatable
|
|
respond_to do |format|
|
|
format.json {
|
|
render json: ::OrganizationUsersDatatable.new(view_context, @org, @user)
|
|
}
|
|
end
|
|
end
|
|
|
|
def new_organization
|
|
@new_org = Organization.new
|
|
end
|
|
|
|
def create_organization
|
|
@new_org = Organization.new(create_organization_params)
|
|
@new_org.created_by = @user
|
|
|
|
if @new_org.save
|
|
# Okay, organization is created, now
|
|
# add the current user as admin
|
|
UserOrganization.create(
|
|
user: @user,
|
|
organization: @new_org,
|
|
role: 2
|
|
)
|
|
|
|
# Redirect to new organization page
|
|
redirect_to action: :organization, organization_id: @new_org.id
|
|
else
|
|
render :new_organization
|
|
end
|
|
end
|
|
|
|
def destroy_organization
|
|
@org.destroy
|
|
|
|
flash[:notice] = I18n.t(
|
|
"users.settings.organizations.edit.modal_destroy_organization.flash_success",
|
|
org: @org.name
|
|
)
|
|
|
|
# Redirect back to all organizations page
|
|
redirect_to action: :organizations
|
|
end
|
|
|
|
def create_user_organization
|
|
@new_user_org = UserOrganization.new(create_user_organization_params)
|
|
|
|
# Check if such association doesn't exist already
|
|
if !UserOrganization.where(
|
|
user: @new_user_org.user,
|
|
organization: @new_user_org.organization
|
|
).exists? && @new_user_org.save
|
|
AppMailer.delay.invitation_to_organization(@new_user_org.user,
|
|
@user_organization.user,
|
|
@new_user_org.organization)
|
|
|
|
generate_notification(@user_organization.user,
|
|
@new_user_org.user,
|
|
@new_user_org.role_str,
|
|
@new_user_org.organization)
|
|
|
|
flash[:notice] = I18n.t(
|
|
'users.settings.organizations.edit.modal_add_user.existing_flash_success',
|
|
user: @new_user_org.user.full_name,
|
|
role: @new_user_org.role_str
|
|
)
|
|
else
|
|
flash[:alert] =
|
|
I18n.t('users.settings.organizations.edit.modal_add_user.existing_flash_error')
|
|
end
|
|
|
|
# Either way, redirect back to organization page
|
|
redirect_to action: :organization,
|
|
organization_id: @new_user_org.organization_id
|
|
end
|
|
|
|
def create_user_and_user_organization
|
|
respond_to do |format|
|
|
# User & organization
|
|
# parameters are already taken care of,
|
|
# so only role needs to be verified
|
|
if !params.include? :role or
|
|
!UserOrganization.roles.keys.include? params[:role]
|
|
format.json {
|
|
render json: "Invalid role provided",
|
|
status: :unprocessable_entity
|
|
}
|
|
else
|
|
password = generate_user_password
|
|
user_params = create_user_params
|
|
full_name = user_params[:full_name]
|
|
email = user_params[:email]
|
|
|
|
# Validate the user data
|
|
errors = validate_user(full_name, email, password)
|
|
|
|
if errors.count == 0
|
|
@user = User.invite!(
|
|
full_name: full_name,
|
|
email: email,
|
|
initials: full_name.split(" ").map{|w| w[0].upcase}.join[0..3],
|
|
skip_invitation: true
|
|
)
|
|
|
|
# Sending email invitation is done in background job to prevent
|
|
# issues with email delivery. Also invite method must be call
|
|
# with :skip_invitation attribute set to true - see above.
|
|
@user.delay.deliver_invitation
|
|
|
|
# Also generate user organization relation
|
|
@user_org = UserOrganization.new(
|
|
user: @user,
|
|
organization: @org,
|
|
role: params[:role]
|
|
)
|
|
@user_org.save
|
|
|
|
# Flash message
|
|
flash[:notice] = t(
|
|
"users.settings.organizations.edit.modal_add_user.new_flash_success",
|
|
user: @user.full_name,
|
|
role: @user_org.role_str,
|
|
email: @user.email
|
|
)
|
|
flash.keep
|
|
|
|
# Return success!
|
|
format.json {
|
|
render json: {
|
|
status: :ok
|
|
}
|
|
}
|
|
else
|
|
format.json {
|
|
render json: errors,
|
|
status: :unprocessable_entity
|
|
}
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def update_user_organization
|
|
respond_to do |format|
|
|
if @user_org.update(update_user_organization_params)
|
|
format.json {
|
|
render json: {
|
|
status: :ok
|
|
}
|
|
}
|
|
else
|
|
format.json {
|
|
render json: @user_org.errors,
|
|
status: :unprocessable_entity
|
|
}
|
|
end
|
|
end
|
|
end
|
|
|
|
def leave_user_organization_html
|
|
respond_to do |format|
|
|
format.json {
|
|
render json: {
|
|
html: render_to_string({
|
|
partial: "users/settings/organizations/leave_user_organization_modal_body.html.erb",
|
|
locals: { user_organization: @user_org }
|
|
}),
|
|
heading: I18n.t(
|
|
"users.settings.organizations.index.leave_uo_heading",
|
|
org: @user_org.organization.name
|
|
)
|
|
}
|
|
}
|
|
end
|
|
end
|
|
|
|
def destroy_user_organization_html
|
|
respond_to do |format|
|
|
format.json {
|
|
render json: {
|
|
html: render_to_string({
|
|
partial: "users/settings/organizations/destroy_user_organization_modal_body.html.erb",
|
|
locals: { user_organization: @user_org }
|
|
}),
|
|
heading: I18n.t(
|
|
"users.settings.organizations.edit.destroy_uo_heading",
|
|
user: @user_org.user.full_name,
|
|
org: @user_org.organization.name
|
|
)
|
|
}
|
|
}
|
|
end
|
|
end
|
|
|
|
def destroy_user_organization
|
|
respond_to do |format|
|
|
# If user is last administrator of organization,
|
|
# he/she cannot be deleted from it.
|
|
invalid =
|
|
@user_org.admin? &&
|
|
@user_org
|
|
.organization
|
|
.user_organizations
|
|
.where(role: 2)
|
|
.count <= 1
|
|
|
|
if !invalid then
|
|
begin
|
|
UserOrganization.transaction do
|
|
# If user leaves on his/her own accord,
|
|
# new owner for projects is the first
|
|
# administrator of organization
|
|
if params[:leave]
|
|
new_owner =
|
|
@user_org
|
|
.organization
|
|
.user_organizations
|
|
.where(role: 2)
|
|
.where.not(id: @user_org.id)
|
|
.first
|
|
.user
|
|
else
|
|
# Otherwise, the new owner for projects is
|
|
# the current user (= an administrator removing
|
|
# the user from the organization)
|
|
new_owner = current_user
|
|
end
|
|
|
|
@user_org.destroy(new_owner)
|
|
end
|
|
rescue Exception
|
|
invalid = true
|
|
end
|
|
end
|
|
|
|
if !invalid
|
|
if params[:leave] then
|
|
flash[:notice] = I18n.t(
|
|
"users.settings.organizations.index.leave_flash",
|
|
org: @user_org.organization.name
|
|
)
|
|
flash.keep(:notice)
|
|
end
|
|
|
|
format.json {
|
|
render json: {
|
|
status: :ok
|
|
}
|
|
}
|
|
else
|
|
format.json {
|
|
render json: @user_org.errors,
|
|
status: :unprocessable_entity
|
|
}
|
|
end
|
|
end
|
|
end
|
|
|
|
def tutorial
|
|
@orgs =
|
|
@user
|
|
.user_organizations
|
|
.includes(organization: :users)
|
|
.where(role: 1..2)
|
|
.order(created_at: :asc)
|
|
.map { |uo| uo.organization }
|
|
@member_of = @orgs.count
|
|
|
|
respond_to do |format|
|
|
format.json {
|
|
render json: {
|
|
status: :ok,
|
|
html: render_to_string({
|
|
partial: "users/settings/repeat_tutorial_modal_body.html.erb"
|
|
})
|
|
}
|
|
}
|
|
end
|
|
end
|
|
|
|
def reset_tutorial
|
|
if @user.update(tutorial_status: 0) && params[:org][:id]
|
|
cookies.delete :tutorial_data
|
|
cookies.delete :current_tutorial_step
|
|
cookies[:repeat_tutorial_org_id] = {
|
|
value: params[:org][:id],
|
|
expires: 1.day.from_now
|
|
}
|
|
|
|
flash[:notice] = t("users.settings.preferences.tutorial.tutorial_reset_flash")
|
|
redirect_to root_path
|
|
else
|
|
flash[:alert] = t("users.settings.preferences.tutorial.tutorial_reset_error")
|
|
redirect_to :back
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def load_user
|
|
@user = current_user
|
|
end
|
|
|
|
def check_organization_permission
|
|
@org = Organization.find_by_id(params[:organization_id])
|
|
unless is_admin_of_organization(@org)
|
|
render_403
|
|
end
|
|
end
|
|
|
|
def check_create_user_organization_permission
|
|
@org = Organization.find_by_id(params[:user_organization][:organization_id])
|
|
unless is_admin_of_organization(@org)
|
|
render_403
|
|
end
|
|
end
|
|
|
|
def check_user_organization_permission
|
|
@user_org = UserOrganization.find_by_id(params[:user_organization_id])
|
|
@org = @user_org.organization
|
|
# Don't allow the user to modify UserOrganization-s if he's not admin,
|
|
# unless he/she is modifying his/her UserOrganization
|
|
if current_user != @user_org.user and
|
|
!is_admin_of_organization(@user_org.organization)
|
|
render_403
|
|
end
|
|
end
|
|
|
|
def update_preferences_params
|
|
params.require(:user).permit(
|
|
:time_zone
|
|
)
|
|
end
|
|
|
|
def create_organization_params
|
|
params.require(:organization).permit(
|
|
:name,
|
|
:description
|
|
)
|
|
end
|
|
|
|
def update_organization_params
|
|
params.require(:organization).permit(
|
|
:name,
|
|
:description
|
|
)
|
|
end
|
|
|
|
def create_user_params
|
|
params.require(:user).permit(
|
|
:full_name,
|
|
:email
|
|
)
|
|
end
|
|
|
|
def create_user_organization_params
|
|
params.require(:user_organization).permit(
|
|
:user_id,
|
|
:organization_id,
|
|
:role
|
|
)
|
|
end
|
|
|
|
def update_user_organization_params
|
|
params.require(:user_organization).permit(
|
|
:role
|
|
)
|
|
end
|
|
|
|
def generate_notification(user, target_user, role, org)
|
|
title = I18n.t('activities.assign_user_to_organization',
|
|
assigned_user: target_user.name,
|
|
role: role,
|
|
organization: org.name,
|
|
assigned_by_user: user.name)
|
|
|
|
message = "#{I18n.t('search.index.organization')} #{org.name}"
|
|
notification = Notification.create(
|
|
type_of: :assignment,
|
|
title:
|
|
ActionController::Base.helpers.sanitize(title),
|
|
message:
|
|
ActionController::Base.helpers.sanitize(message),
|
|
)
|
|
UserNotification.create(notification: notification, user: target_user)
|
|
end
|
|
end
|