From a19364798107836460e17297d36f2b470b46e6a7 Mon Sep 17 00:00:00 2001 From: nanaya Date: Mon, 16 Dec 2024 22:52:48 +0900 Subject: [PATCH] Add ability to join team --- .../Teams/ApplicationsController.php | 74 ++++++++++ .../Notifications/TeamApplicationAccept.php | 51 +++++++ .../Notifications/TeamApplicationReject.php | 51 +++++++ app/Libraries/MorphMap.php | 2 + app/Models/Notification.php | 5 + app/Models/Team.php | 15 +++ app/Models/TeamApplication.php | 27 ++++ app/Models/User.php | 6 + app/Singletons/OsuAuthorize.php | 41 ++++++ database/factories/TeamMemberFactory.php | 25 ++++ ..._01_15_000001_create_team_applications.php | 30 +++++ resources/js/notification-maps/category.ts | 3 + resources/lang/en/authorization.php | 9 ++ resources/lang/en/notifications.php | 18 +++ resources/lang/en/teams.php | 23 ++++ resources/views/teams/members/index.blade.php | 71 ++++++++++ resources/views/teams/show.blade.php | 46 +++++++ routes/web.php | 3 + .../Teams/ApplicationsControllerTest.php | 127 ++++++++++++++++++ 19 files changed, 627 insertions(+) create mode 100644 app/Http/Controllers/Teams/ApplicationsController.php create mode 100644 app/Jobs/Notifications/TeamApplicationAccept.php create mode 100644 app/Jobs/Notifications/TeamApplicationReject.php create mode 100644 app/Models/TeamApplication.php create mode 100644 database/factories/TeamMemberFactory.php create mode 100644 database/migrations/2025_01_15_000001_create_team_applications.php create mode 100644 tests/Controllers/Teams/ApplicationsControllerTest.php diff --git a/app/Http/Controllers/Teams/ApplicationsController.php b/app/Http/Controllers/Teams/ApplicationsController.php new file mode 100644 index 00000000000..cbf90fcc27d --- /dev/null +++ b/app/Http/Controllers/Teams/ApplicationsController.php @@ -0,0 +1,74 @@ +. Licensed under the GNU Affero General Public License v3.0. +// See the LICENCE file in the repository root for full licence text. + +declare(strict_types=1); + +namespace App\Http\Controllers\Teams; + +use App\Http\Controllers\Controller; +use App\Jobs\Notifications\TeamApplicationAccept; +use App\Jobs\Notifications\TeamApplicationReject; +use App\Models\Team; +use App\Models\TeamApplication; +use Symfony\Component\HttpFoundation\Response; + +class ApplicationsController extends Controller +{ + public function __construct() + { + parent::__construct(); + + $this->middleware('auth'); + } + + public function accept(string $teamId, string $id): Response + { + $member = \DB::transaction(function () use ($id, $teamId) { + $team = Team::findOrFail($teamId); + $application = $team->applications()->findOrFail($id); + + priv_check('TeamApplicationAccept', $application)->ensureCan(); + + $application->delete(); + + return $team->members()->create(['user_id' => $application->user_id]); + }); + (new TeamApplicationAccept($member, \Auth::user()))->dispatch(); + + \Session::flash('popup', osu_trans('teams.applications.accept.ok')); + + return response(null, 204); + } + + public function destroy(string $teamId, string $id): Response + { + $currentUser = \Auth::user(); + TeamApplication::where('team_id', $teamId)->findOrFail($currentUser->getKey())->delete(); + \Session::flash('popup', osu_trans('teams.applications.destroy.ok')); + + return response(null, 204); + } + + public function reject(string $teamId, string $id): Response + { + $application = TeamApplication::where('team_id', $teamId)->findOrFail($id); + $application->delete(); + \Session::flash('popup', osu_trans('teams.applications.reject.ok')); + (new TeamApplicationReject($application, \Auth::user()))->dispatch(); + + return response(null, 204); + } + + public function store(string $teamId): Response + { + $team = Team::findOrFail($teamId); + priv_check('TeamApplicationStore', $team)->ensureCan(); + + $team->applications()->createOrFirst(['user_id' => \Auth::id()]); + \Session::flash('popup', osu_trans('teams.applications.store.ok')); + + return response(null, 204); + } +} diff --git a/app/Jobs/Notifications/TeamApplicationAccept.php b/app/Jobs/Notifications/TeamApplicationAccept.php new file mode 100644 index 00000000000..4510871ec6b --- /dev/null +++ b/app/Jobs/Notifications/TeamApplicationAccept.php @@ -0,0 +1,51 @@ +. Licensed under the GNU Affero General Public License v3.0. +// See the LICENCE file in the repository root for full licence text. + +namespace App\Jobs\Notifications; + +use App\Models\Notification; +use App\Models\Team; +use App\Models\TeamMember; +use App\Models\User; + +class TeamApplicationAccept extends BroadcastNotificationBase +{ + const DELIVERY_MODE_DEFAULTS = ['mail' => true, 'push' => true]; + + private Team $team; + private int $userId; + + public static function getMailLink(Notification $notification): string + { + return route('teams.show', ['team' => $notification->notifiable_id]); + } + + public function __construct(TeamMember $member, User $source) + { + $this->team = $member->team; + $this->userId = $member->user_id; + + parent::__construct($source); + } + + public function getDetails(): array + { + return [ + 'cover_url' => $this->team->logo()->url(), + 'team_id' => $this->team->getKey(), + 'title' => $this->team->name, + ]; + } + + public function getListeningUserIds(): array + { + return [$this->userId]; + } + + public function getNotifiable() + { + return $this->team; + } +} diff --git a/app/Jobs/Notifications/TeamApplicationReject.php b/app/Jobs/Notifications/TeamApplicationReject.php new file mode 100644 index 00000000000..53e4cbb1e6a --- /dev/null +++ b/app/Jobs/Notifications/TeamApplicationReject.php @@ -0,0 +1,51 @@ +. Licensed under the GNU Affero General Public License v3.0. +// See the LICENCE file in the repository root for full licence text. + +namespace App\Jobs\Notifications; + +use App\Models\Notification; +use App\Models\Team; +use App\Models\TeamApplication; +use App\Models\User; + +class TeamApplicationReject extends BroadcastNotificationBase +{ + const DELIVERY_MODE_DEFAULTS = ['mail' => true, 'push' => true]; + + private Team $team; + private int $userId; + + public static function getMailLink(Notification $notification): string + { + return route('teams.show', ['team' => $notification->notifiable_id]); + } + + public function __construct(TeamApplication $application, User $source) + { + $this->team = $application->team; + $this->userId = $application->user_id; + + parent::__construct($source); + } + + public function getDetails(): array + { + return [ + 'cover_url' => $this->team->logo()->url(), + 'team_id' => $this->team->getKey(), + 'title' => $this->team->name, + ]; + } + + public function getListeningUserIds(): array + { + return [$this->userId]; + } + + public function getNotifiable() + { + return $this->team; + } +} diff --git a/app/Libraries/MorphMap.php b/app/Libraries/MorphMap.php index d79ad41cb88..9f3fcfe40e8 100644 --- a/app/Libraries/MorphMap.php +++ b/app/Libraries/MorphMap.php @@ -18,6 +18,7 @@ use App\Models\NewsPost; use App\Models\Score; use App\Models\Solo; +use App\Models\Team; use App\Models\User; class MorphMap @@ -44,6 +45,7 @@ class MorphMap Score\Osu::class => 'score_osu', Score\Taiko::class => 'score_taiko', Solo\Score::class => 'solo_score', + Team::class => 'team', User::class => 'user', ]; diff --git a/app/Models/Notification.php b/app/Models/Notification.php index 6e9f4ed9614..3c588fdbedf 100644 --- a/app/Models/Notification.php +++ b/app/Models/Notification.php @@ -43,10 +43,13 @@ class Notification extends Model const CHANNEL_MESSAGE = 'channel_message'; const COMMENT_NEW = 'comment_new'; const FORUM_TOPIC_REPLY = 'forum_topic_reply'; + const TEAM_APPLICATION_ACCEPT = 'team_application_accept'; + const TEAM_APPLICATION_REJECT = 'team_application_reject'; const USER_ACHIEVEMENT_UNLOCK = 'user_achievement_unlock'; const USER_BEATMAPSET_NEW = 'user_beatmapset_new'; const USER_BEATMAPSET_REVIVE = 'user_beatmapset_revive'; + // sync with resources/js/notification-maps/category.ts const NAME_TO_CATEGORY = [ self::BEATMAP_OWNER_CHANGE => 'beatmap_owner_change', self::BEATMAPSET_DISCUSSION_LOCK => 'beatmapset_discussion', @@ -65,6 +68,8 @@ class Notification extends Model self::CHANNEL_MESSAGE => 'channel', self::COMMENT_NEW => 'comment', self::FORUM_TOPIC_REPLY => 'forum_topic_reply', + self::TEAM_APPLICATION_ACCEPT => 'team_application', + self::TEAM_APPLICATION_REJECT => 'team_application', self::USER_ACHIEVEMENT_UNLOCK => 'user_achievement_unlock', self::USER_BEATMAPSET_NEW => 'user_beatmapset_new', self::USER_BEATMAPSET_REVIVE => 'user_beatmapset_new', diff --git a/app/Models/Team.php b/app/Models/Team.php index 33437a2774f..41c07198ca1 100644 --- a/app/Models/Team.php +++ b/app/Models/Team.php @@ -92,6 +92,14 @@ public function delete() }); } + public function emptySlots(): int + { + $max = $this->maxMembers(); + $current = $this->members->count(); + + return max(0, $max - $current); + } + public function header(): Uploader { return $this->header ??= new Uploader( @@ -131,4 +139,11 @@ public function logo(): Uploader ['image' => ['maxDimensions' => [512, 256]]], ); } + + public function maxMembers(): int + { + $this->loadMissing('members.user'); + + return 8 + (4 * $this->members->filter(fn ($member) => $member->user?->osu_subscriber ?? false)->count()); + } } diff --git a/app/Models/TeamApplication.php b/app/Models/TeamApplication.php new file mode 100644 index 00000000000..a769a64343f --- /dev/null +++ b/app/Models/TeamApplication.php @@ -0,0 +1,27 @@ +. Licensed under the GNU Affero General Public License v3.0. +// See the LICENCE file in the repository root for full licence text. + +declare(strict_types=1); + +namespace App\Models; + +use Illuminate\Database\Eloquent\Relations\BelongsTo; + +class TeamApplication extends Model +{ + public $incrementing = false; + + protected $primaryKey = 'user_id'; + + public function team(): BelongsTo + { + return $this->belongsTo(Team::class); + } + + public function user(): BelongsTo + { + return $this->belongsTo(User::class, 'user_id'); + } +} diff --git a/app/Models/User.php b/app/Models/User.php index 53088b6d1a3..340caf631ac 100644 --- a/app/Models/User.php +++ b/app/Models/User.php @@ -310,6 +310,11 @@ public function team(): HasOneThrough ); } + public function teamApplication(): HasOne + { + return $this->hasOne(TeamApplication::class); + } + public function getAuthPassword() { return $this->user_password; @@ -967,6 +972,7 @@ public function getAttribute($key) 'supporterTagPurchases', 'supporterTags', 'team', + 'teamApplication', 'tokens', 'topicWatches', 'userAchievements', diff --git a/app/Singletons/OsuAuthorize.php b/app/Singletons/OsuAuthorize.php index de5e189746b..165d2877798 100644 --- a/app/Singletons/OsuAuthorize.php +++ b/app/Singletons/OsuAuthorize.php @@ -30,6 +30,7 @@ use App\Models\Score\Best\Model as ScoreBest; use App\Models\Solo; use App\Models\Team; +use App\Models\TeamApplication; use App\Models\Traits\ReportableInterface; use App\Models\User; use App\Models\UserContestEntry; @@ -1905,6 +1906,46 @@ public function checkScorePin(?User $user, ScoreBest|Solo\Score $score): string return 'ok'; } + public function checkTeamApplicationAccept(?User $user, TeamApplication $application): ?string + { + $this->ensureLoggedIn($user); + + $team = $application->team; + + if ($team->leader_id !== $user->getKey()) { + return null; + } + if ($team->emptySlots() < 1) { + return 'team.member.store.full'; + } + + return 'ok'; + } + + public function checkTeamApplicationStore(?User $user, Team $team): ?string + { + $prefix = 'team.application.store.'; + + $this->ensureLoggedIn($user); + + if ($user->team !== null) { + return $user->team->getKey() === $team->getKey() + ? $prefix.'already_member' + : $prefix.'already_other_member'; + } + if ($user->teamApplication()->exists()) { + return $prefix.'currently_applying'; + } + if (!$team->is_open) { + return $prefix.'team_closed'; + } + if ($team->emptySlots() < 1) { + return $prefix.'team_full'; + } + + return 'ok'; + } + public function checkTeamPart(?User $user, Team $team): ?string { $this->ensureLoggedIn($user); diff --git a/database/factories/TeamMemberFactory.php b/database/factories/TeamMemberFactory.php new file mode 100644 index 00000000000..79a84410423 --- /dev/null +++ b/database/factories/TeamMemberFactory.php @@ -0,0 +1,25 @@ +. Licensed under the GNU Affero General Public License v3.0. +// See the LICENCE file in the repository root for full licence text. + +declare(strict_types=1); + +namespace Database\Factories; + +use App\Models\Team; +use App\Models\TeamMember; +use App\Models\User; + +class TeamMemberFactory extends Factory +{ + protected $model = TeamMember::class; + + public function definition(): array + { + return [ + 'team_id' => Team::factory(), + 'user_id' => User::factory(), + ]; + } +} diff --git a/database/migrations/2025_01_15_000001_create_team_applications.php b/database/migrations/2025_01_15_000001_create_team_applications.php new file mode 100644 index 00000000000..05229bc0b77 --- /dev/null +++ b/database/migrations/2025_01_15_000001_create_team_applications.php @@ -0,0 +1,30 @@ +. Licensed under the GNU Affero General Public License v3.0. +// See the LICENCE file in the repository root for full licence text. + +declare(strict_types=1); + +use Illuminate\Database\Migrations\Migration; +use Illuminate\Database\Schema\Blueprint; +use Illuminate\Support\Facades\Schema; + +return new class extends Migration +{ + public function up(): void + { + Schema::create('team_applications', function (Blueprint $table) { + $table->unsignedBigInteger('user_id')->nullable(false); + $table->unsignedBigInteger('team_id')->nullable(false); + $table->timestampsTz(); + + $table->primary('user_id'); + $table->index('team_id'); + }); + } + + public function down(): void + { + Schema::dropIfExists('team_applications'); + } +}; diff --git a/resources/js/notification-maps/category.ts b/resources/js/notification-maps/category.ts index b7386d65bc8..1b5c3ac9af7 100644 --- a/resources/js/notification-maps/category.ts +++ b/resources/js/notification-maps/category.ts @@ -19,6 +19,7 @@ export function categoryFromName(name: string) { return nameToCategory[name] ?? name; } +// sync with app/Models/Notification.php export const nameToCategory: Partial> = { beatmap_owner_change: 'beatmap_owner_change', beatmapset_discussion_lock: 'beatmapset_discussion', @@ -38,6 +39,8 @@ export const nameToCategory: Partial> = { comment_new: 'comment', comment_reply: 'comment', forum_topic_reply: 'forum_topic_reply', + team_application_accept: 'team_application', + team_application_reject: 'team_application', user_achievement_unlock: 'user_achievement_unlock', user_beatmapset_new: 'user_beatmapset_new', user_beatmapset_revive: 'user_beatmapset_new', diff --git a/resources/lang/en/authorization.php b/resources/lang/en/authorization.php index ba061ef6ccf..404508084fa 100644 --- a/resources/lang/en/authorization.php +++ b/resources/lang/en/authorization.php @@ -192,6 +192,15 @@ ], 'team' => [ + 'application' => [ + 'store' => [ + 'already_member' => "You're already part of the team.", + 'already_other_member' => "You're already part of a different team.", + 'currently_applying' => 'You have pending team join request.', + 'team_closed' => 'The team is currently not accepting any join requests.', + 'team_full' => "The team is full and can't accept any more members.", + ], + ], 'part' => [ 'is_leader' => "Team leader can't leave the team.", 'not_member' => 'Not a member of the team.', diff --git a/resources/lang/en/notifications.php b/resources/lang/en/notifications.php index 780d459296c..1c1da1e4658 100644 --- a/resources/lang/en/notifications.php +++ b/resources/lang/en/notifications.php @@ -159,6 +159,17 @@ ], ], + 'team' => [ + 'team_application' => [ + '_' => 'Team join request', + + 'team_application_accept' => "You're now member of team :title", + 'team_application_accept_compact' => "You're now member of team :title", + 'team_application_reject' => 'Your request to join team :title has been declined', + 'team_application_reject_compact' => 'Your request to join team :title has been declined', + ], + ], + 'user' => [ 'user_beatmapset_new' => [ '_' => 'New beatmap', @@ -243,6 +254,13 @@ ], ], + 'team' => [ + 'team_application' => [ + 'team_application_accept' => "You're now member of team :title", + 'team_application_reject' => 'Your request to join team :title has been declined', + ], + ], + 'user' => [ 'user_beatmapset_new' => [ 'user_beatmapset_new' => ':username has created new beatmaps', diff --git a/resources/lang/en/teams.php b/resources/lang/en/teams.php index a3984e14dff..4a4da7b2907 100644 --- a/resources/lang/en/teams.php +++ b/resources/lang/en/teams.php @@ -4,6 +4,21 @@ // See the LICENCE file in the repository root for full licence text. return [ + 'applications' => [ + 'accept' => [ + 'ok' => 'Added user to team.', + ], + 'destroy' => [ + 'ok' => 'Cancelled join request.', + ], + 'reject' => [ + 'ok' => 'Rejected join request.', + ], + 'store' => [ + 'ok' => 'Requested to join team.', + ], + ], + 'destroy' => [ 'ok' => 'Team removed', ], @@ -50,6 +65,12 @@ 'index' => [ 'title' => 'Manage Members', + 'applications' => [ + 'empty' => 'No join requests at the moment.', + 'title' => 'Join Requests', + 'created_at' => 'Requested At', + ], + 'table' => [ 'status' => 'Status', 'joined_at' => 'Join Date', @@ -71,6 +92,8 @@ 'show' => [ 'bar' => [ 'destroy' => 'Disband Team', + 'join' => 'Request Join', + 'join_cancel' => 'Cancel Join', 'part' => 'Leave Team', ], diff --git a/resources/views/teams/members/index.blade.php b/resources/views/teams/members/index.blade.php index 2befc8268da..99ae3f54c5a 100644 --- a/resources/views/teams/members/index.blade.php +++ b/resources/views/teams/members/index.blade.php @@ -78,6 +78,77 @@ class="btn-osu-big btn-osu-big--rounded-small" +
+

+ {{ osu_trans('teams.members.index.applications.title') }} +

+ @if ($team->applications->isEmpty()) + {{ osu_trans('teams.members.index.applications.empty') }} + @else +
    +
  • + + {{ osu_trans('teams.members.index.applications.created_at') }} + + + @foreach ($team->applications as $application) + @php + $user = $application->user; + @endphp + @if ($user === null) + @continue + @endif +
  • + + + user_avatar) !!} + > + + + {{ $user->username }} + + + {!! timeago($application->created_at) !!} + + +
    + +
    +
    + +
    + +
    +
    + @endforeach +
+ @endif +
+
diff --git a/resources/views/teams/show.blade.php b/resources/views/teams/show.blade.php index ae9e5927d8e..7630204e4c8 100644 --- a/resources/views/teams/show.blade.php +++ b/resources/views/teams/show.blade.php @@ -21,6 +21,15 @@ if (priv_check('TeamUpdate', $team)->can()) { $buttons->add('destroy'); } + + $currentUser = Auth::user(); + if ($currentUser === null || $currentUser->team === null) { + if ($currentUser !== null && $currentUser->teamApplication?->team_id === $team->getKey()) { + $buttons->add('join_cancel'); + } else { + $buttons->add('join'); + } + } @endphp @extends('master', [ @@ -102,6 +111,43 @@ class="btn-circle btn-circle--page-toggle" @endif + + @if ($buttons->contains('join_cancel')) +
+ + +
+ @endif + @if ($buttons->contains('join')) + @php + $joinPriv = priv_check('TeamApplicationStore', $team); + @endphp +
+ +
+ @endif
@endif