2015-12-21 20:24:11 +08:00
|
|
|
// Copyright 2015 The Gogs Authors. All rights reserved.
|
2019-02-19 00:00:27 +08:00
|
|
|
// Copyright 2019 The Gitea Authors. All rights reserved.
|
2022-11-28 02:20:29 +08:00
|
|
|
// SPDX-License-Identifier: MIT
|
2015-12-21 20:24:11 +08:00
|
|
|
|
|
|
|
package user
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-04-05 23:30:52 +08:00
|
|
|
"net/http"
|
2024-01-15 16:49:24 +08:00
|
|
|
"path"
|
2015-12-21 20:24:11 +08:00
|
|
|
"strings"
|
|
|
|
|
2022-08-25 10:31:57 +08:00
|
|
|
activities_model "code.gitea.io/gitea/models/activities"
|
2021-09-24 19:32:56 +08:00
|
|
|
"code.gitea.io/gitea/models/db"
|
2021-12-10 09:27:50 +08:00
|
|
|
repo_model "code.gitea.io/gitea/models/repo"
|
2021-11-11 15:03:30 +08:00
|
|
|
user_model "code.gitea.io/gitea/models/user"
|
2024-01-30 22:45:54 +08:00
|
|
|
"code.gitea.io/gitea/modules/base"
|
2023-05-09 13:57:24 +08:00
|
|
|
"code.gitea.io/gitea/modules/git"
|
2023-07-07 02:59:24 +08:00
|
|
|
"code.gitea.io/gitea/modules/log"
|
2021-04-20 06:25:08 +08:00
|
|
|
"code.gitea.io/gitea/modules/markup"
|
2020-08-05 15:48:37 +08:00
|
|
|
"code.gitea.io/gitea/modules/markup/markdown"
|
2024-03-01 02:52:49 +08:00
|
|
|
"code.gitea.io/gitea/modules/optional"
|
2016-11-11 00:24:48 +08:00
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2017-10-27 05:16:13 +08:00
|
|
|
"code.gitea.io/gitea/modules/util"
|
2021-10-16 22:21:16 +08:00
|
|
|
"code.gitea.io/gitea/routers/web/feed"
|
2021-06-09 07:33:54 +08:00
|
|
|
"code.gitea.io/gitea/routers/web/org"
|
2023-07-07 02:59:24 +08:00
|
|
|
shared_user "code.gitea.io/gitea/routers/web/shared/user"
|
2024-02-27 15:12:22 +08:00
|
|
|
"code.gitea.io/gitea/services/context"
|
2015-12-21 20:24:11 +08:00
|
|
|
)
|
|
|
|
|
2024-01-30 22:45:54 +08:00
|
|
|
const (
|
|
|
|
tplProfileBigAvatar base.TplName = "shared/user/profile_big_avatar"
|
2024-02-17 21:42:52 +08:00
|
|
|
tplFollowUnfollow base.TplName = "org/follow_unfollow"
|
2024-01-30 22:45:54 +08:00
|
|
|
)
|
|
|
|
|
2023-07-07 02:59:24 +08:00
|
|
|
// OwnerProfile render profile page for a user or a organization (aka, repo owner)
|
|
|
|
func OwnerProfile(ctx *context.Context) {
|
2022-03-26 17:04:22 +08:00
|
|
|
if strings.Contains(ctx.Req.Header.Get("Accept"), "application/rss+xml") {
|
|
|
|
feed.ShowUserFeedRSS(ctx)
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
2021-04-28 20:35:06 +08:00
|
|
|
}
|
2022-03-26 17:04:22 +08:00
|
|
|
if strings.Contains(ctx.Req.Header.Get("Accept"), "application/atom+xml") {
|
|
|
|
feed.ShowUserFeedAtom(ctx)
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-26 17:04:22 +08:00
|
|
|
if ctx.ContextUser.IsOrganization() {
|
2021-06-27 03:53:14 +08:00
|
|
|
org.Home(ctx)
|
2023-07-07 02:59:24 +08:00
|
|
|
} else {
|
|
|
|
userProfile(ctx)
|
2021-06-27 03:53:14 +08:00
|
|
|
}
|
2023-07-07 02:59:24 +08:00
|
|
|
}
|
2021-06-27 03:53:14 +08:00
|
|
|
|
2023-07-07 02:59:24 +08:00
|
|
|
func userProfile(ctx *context.Context) {
|
2021-06-27 03:53:14 +08:00
|
|
|
// check view permissions
|
2022-05-20 22:08:52 +08:00
|
|
|
if !user_model.IsUserVisibleToViewer(ctx, ctx.ContextUser, ctx.Doer) {
|
2022-03-26 17:04:22 +08:00
|
|
|
ctx.NotFound("user", fmt.Errorf(ctx.ContextUser.Name))
|
2021-10-16 22:21:16 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-26 17:04:22 +08:00
|
|
|
ctx.Data["Title"] = ctx.ContextUser.DisplayName()
|
2015-12-21 20:24:11 +08:00
|
|
|
ctx.Data["PageIsUserProfile"] = true
|
2020-11-19 06:00:16 +08:00
|
|
|
|
2023-07-07 02:59:24 +08:00
|
|
|
// prepare heatmap data
|
2021-03-05 06:59:13 +08:00
|
|
|
if setting.Service.EnableUserHeatmap {
|
2023-09-25 21:17:37 +08:00
|
|
|
data, err := activities_model.GetUserHeatmapDataByUser(ctx, ctx.ContextUser, ctx.Doer)
|
2020-11-19 06:00:16 +08:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetUserHeatmapDataByUser", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["HeatmapData"] = data
|
2023-04-18 02:26:01 +08:00
|
|
|
ctx.Data["HeatmapTotalContributions"] = activities_model.GetTotalContributionsInHeatmap(data)
|
2020-11-19 06:00:16 +08:00
|
|
|
}
|
|
|
|
|
2023-12-27 16:32:27 +08:00
|
|
|
profileDbRepo, profileGitRepo, profileReadmeBlob, profileClose := shared_user.FindUserProfileReadme(ctx, ctx.Doer)
|
2023-07-07 02:59:24 +08:00
|
|
|
defer profileClose()
|
2023-05-09 13:57:24 +08:00
|
|
|
|
2022-03-26 17:04:22 +08:00
|
|
|
showPrivate := ctx.IsSigned && (ctx.Doer.IsAdmin || ctx.Doer.ID == ctx.ContextUser.ID)
|
2023-12-27 16:32:27 +08:00
|
|
|
prepareUserProfileTabData(ctx, showPrivate, profileDbRepo, profileGitRepo, profileReadmeBlob)
|
2023-07-07 02:59:24 +08:00
|
|
|
// call PrepareContextForProfileBigAvatar later to avoid re-querying the NumFollowers & NumFollowing
|
|
|
|
shared_user.PrepareContextForProfileBigAvatar(ctx)
|
|
|
|
ctx.HTML(http.StatusOK, tplProfile)
|
|
|
|
}
|
2016-01-31 05:51:11 +08:00
|
|
|
|
2023-12-27 16:32:27 +08:00
|
|
|
func prepareUserProfileTabData(ctx *context.Context, showPrivate bool, profileDbRepo *repo_model.Repository, profileGitRepo *git.Repository, profileReadme *git.Blob) {
|
2023-07-07 02:59:24 +08:00
|
|
|
// if there is a profile readme, default to "overview" page, otherwise, default to "repositories" page
|
2023-09-20 09:48:44 +08:00
|
|
|
// if there is not a profile readme, the overview tab should be treated as the repositories tab
|
2021-08-11 08:31:13 +08:00
|
|
|
tab := ctx.FormString("tab")
|
2023-09-20 09:48:44 +08:00
|
|
|
if tab == "" || tab == "overview" {
|
2023-07-07 02:59:24 +08:00
|
|
|
if profileReadme != nil {
|
|
|
|
tab = "overview"
|
|
|
|
} else {
|
|
|
|
tab = "repositories"
|
|
|
|
}
|
|
|
|
}
|
2015-12-21 20:24:11 +08:00
|
|
|
ctx.Data["TabName"] = tab
|
2023-07-07 02:59:24 +08:00
|
|
|
ctx.Data["HasProfileReadme"] = profileReadme != nil
|
2017-02-14 15:28:22 +08:00
|
|
|
|
2021-07-29 09:42:15 +08:00
|
|
|
page := ctx.FormInt("page")
|
2017-02-14 15:28:22 +08:00
|
|
|
if page <= 0 {
|
|
|
|
page = 1
|
|
|
|
}
|
|
|
|
|
2023-02-25 05:15:10 +08:00
|
|
|
pagingNum := setting.UI.User.RepoPagingNum
|
2021-07-29 09:42:15 +08:00
|
|
|
topicOnly := ctx.FormBool("topic")
|
2017-02-14 15:28:22 +08:00
|
|
|
var (
|
2021-12-10 09:27:50 +08:00
|
|
|
repos []*repo_model.Repository
|
2017-02-14 15:28:22 +08:00
|
|
|
count int64
|
2019-04-20 12:15:19 +08:00
|
|
|
total int
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy db.SearchOrderBy
|
2017-02-14 15:28:22 +08:00
|
|
|
)
|
|
|
|
|
2021-08-11 08:31:13 +08:00
|
|
|
ctx.Data["SortType"] = ctx.FormString("sort")
|
|
|
|
switch ctx.FormString("sort") {
|
2017-02-14 15:28:22 +08:00
|
|
|
case "newest":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByNewest
|
2017-02-14 15:28:22 +08:00
|
|
|
case "oldest":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByOldest
|
2017-02-14 15:28:22 +08:00
|
|
|
case "recentupdate":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByRecentUpdated
|
2017-02-14 15:28:22 +08:00
|
|
|
case "leastupdate":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByLeastUpdated
|
2017-02-14 15:28:22 +08:00
|
|
|
case "reversealphabetically":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByAlphabeticallyReverse
|
2017-02-14 15:28:22 +08:00
|
|
|
case "alphabetically":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByAlphabetically
|
2018-05-24 09:03:42 +08:00
|
|
|
case "moststars":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByStarsReverse
|
2018-05-24 09:03:42 +08:00
|
|
|
case "feweststars":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByStars
|
2018-05-24 09:03:42 +08:00
|
|
|
case "mostforks":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByForksReverse
|
2018-05-24 09:03:42 +08:00
|
|
|
case "fewestforks":
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByForks
|
2017-02-14 15:28:22 +08:00
|
|
|
default:
|
|
|
|
ctx.Data["SortType"] = "recentupdate"
|
2021-11-24 17:49:20 +08:00
|
|
|
orderBy = db.SearchOrderByRecentUpdated
|
2017-02-14 15:28:22 +08:00
|
|
|
}
|
|
|
|
|
2021-08-11 23:08:52 +08:00
|
|
|
keyword := ctx.FormTrim("q")
|
2017-02-14 15:28:22 +08:00
|
|
|
ctx.Data["Keyword"] = keyword
|
2022-01-28 19:29:04 +08:00
|
|
|
|
|
|
|
language := ctx.FormTrim("language")
|
|
|
|
ctx.Data["Language"] = language
|
|
|
|
|
2023-04-30 03:13:58 +08:00
|
|
|
followers, numFollowers, err := user_model.GetUserFollowers(ctx, ctx.ContextUser, ctx.Doer, db.ListOptions{
|
|
|
|
PageSize: pagingNum,
|
|
|
|
Page: page,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetUserFollowers", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["NumFollowers"] = numFollowers
|
|
|
|
following, numFollowing, err := user_model.GetUserFollowing(ctx, ctx.ContextUser, ctx.Doer, db.ListOptions{
|
|
|
|
PageSize: pagingNum,
|
|
|
|
Page: page,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetUserFollowing", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["NumFollowing"] = numFollowing
|
|
|
|
|
2024-03-03 18:18:34 +08:00
|
|
|
archived := ctx.FormOptionalBool("archived")
|
|
|
|
ctx.Data["IsArchived"] = archived
|
|
|
|
|
|
|
|
fork := ctx.FormOptionalBool("fork")
|
|
|
|
ctx.Data["IsFork"] = fork
|
|
|
|
|
|
|
|
mirror := ctx.FormOptionalBool("mirror")
|
|
|
|
ctx.Data["IsMirror"] = mirror
|
|
|
|
|
|
|
|
template := ctx.FormOptionalBool("template")
|
|
|
|
ctx.Data["IsTemplate"] = template
|
|
|
|
|
|
|
|
private := ctx.FormOptionalBool("private")
|
|
|
|
ctx.Data["IsPrivate"] = private
|
|
|
|
|
2015-12-21 20:24:11 +08:00
|
|
|
switch tab {
|
2020-02-10 04:18:01 +08:00
|
|
|
case "followers":
|
2023-04-30 03:13:58 +08:00
|
|
|
ctx.Data["Cards"] = followers
|
2023-09-19 23:24:54 +08:00
|
|
|
total = int(numFollowers)
|
2020-02-10 04:18:01 +08:00
|
|
|
case "following":
|
2023-04-30 03:13:58 +08:00
|
|
|
ctx.Data["Cards"] = following
|
2023-09-19 23:24:54 +08:00
|
|
|
total = int(numFollowing)
|
2015-12-21 20:24:11 +08:00
|
|
|
case "activity":
|
2023-02-25 05:15:10 +08:00
|
|
|
date := ctx.FormString("date")
|
2023-07-07 02:59:24 +08:00
|
|
|
pagingNum = setting.UI.FeedPagingNum
|
2023-02-25 05:15:10 +08:00
|
|
|
items, count, err := activities_model.GetFeeds(ctx, activities_model.GetFeedsOptions{
|
2022-03-26 17:04:22 +08:00
|
|
|
RequestedUser: ctx.ContextUser,
|
2022-03-22 15:03:22 +08:00
|
|
|
Actor: ctx.Doer,
|
2017-08-23 09:30:54 +08:00
|
|
|
IncludePrivate: showPrivate,
|
|
|
|
OnlyPerformedBy: true,
|
|
|
|
IncludeDeleted: false,
|
2023-02-25 05:15:10 +08:00
|
|
|
Date: date,
|
|
|
|
ListOptions: db.ListOptions{
|
|
|
|
PageSize: pagingNum,
|
|
|
|
Page: page,
|
|
|
|
},
|
2017-08-23 09:30:54 +08:00
|
|
|
})
|
2022-03-14 00:40:47 +08:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetFeeds", err)
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
|
|
|
}
|
2023-02-25 05:15:10 +08:00
|
|
|
ctx.Data["Feeds"] = items
|
|
|
|
ctx.Data["Date"] = date
|
|
|
|
|
|
|
|
total = int(count)
|
2016-12-29 22:58:24 +08:00
|
|
|
case "stars":
|
2017-02-14 15:28:22 +08:00
|
|
|
ctx.Data["PageIsProfileStarList"] = true
|
2022-11-19 16:12:33 +08:00
|
|
|
repos, count, err = repo_model.SearchRepository(ctx, &repo_model.SearchRepoOptions{
|
2021-09-24 19:32:56 +08:00
|
|
|
ListOptions: db.ListOptions{
|
2023-02-25 05:15:10 +08:00
|
|
|
PageSize: pagingNum,
|
2020-01-25 03:00:29 +08:00
|
|
|
Page: page,
|
|
|
|
},
|
2022-03-22 15:03:22 +08:00
|
|
|
Actor: ctx.Doer,
|
2019-08-26 01:06:36 +08:00
|
|
|
Keyword: keyword,
|
|
|
|
OrderBy: orderBy,
|
|
|
|
Private: ctx.IsSigned,
|
2022-03-26 17:04:22 +08:00
|
|
|
StarredByID: ctx.ContextUser.ID,
|
2024-03-01 02:52:49 +08:00
|
|
|
Collaborate: optional.Some(false),
|
2019-08-26 01:06:36 +08:00
|
|
|
TopicOnly: topicOnly,
|
2022-01-28 19:29:04 +08:00
|
|
|
Language: language,
|
2019-08-26 01:06:36 +08:00
|
|
|
IncludeDescription: setting.UI.SearchRepoDescription,
|
2024-03-03 18:18:34 +08:00
|
|
|
Archived: archived,
|
|
|
|
Fork: fork,
|
|
|
|
Mirror: mirror,
|
|
|
|
Template: template,
|
|
|
|
IsPrivate: private,
|
2019-05-15 23:24:39 +08:00
|
|
|
})
|
|
|
|
if err != nil {
|
2019-08-26 01:06:36 +08:00
|
|
|
ctx.ServerError("SearchRepository", err)
|
2019-05-15 23:24:39 +08:00
|
|
|
return
|
2017-02-07 19:54:16 +08:00
|
|
|
}
|
|
|
|
|
2019-04-20 12:15:19 +08:00
|
|
|
total = int(count)
|
2021-04-16 00:53:57 +08:00
|
|
|
case "watching":
|
2022-11-19 16:12:33 +08:00
|
|
|
repos, count, err = repo_model.SearchRepository(ctx, &repo_model.SearchRepoOptions{
|
2021-09-24 19:32:56 +08:00
|
|
|
ListOptions: db.ListOptions{
|
2023-02-25 05:15:10 +08:00
|
|
|
PageSize: pagingNum,
|
2021-04-16 00:53:57 +08:00
|
|
|
Page: page,
|
|
|
|
},
|
2022-03-22 15:03:22 +08:00
|
|
|
Actor: ctx.Doer,
|
2021-04-16 00:53:57 +08:00
|
|
|
Keyword: keyword,
|
|
|
|
OrderBy: orderBy,
|
|
|
|
Private: ctx.IsSigned,
|
2022-03-26 17:04:22 +08:00
|
|
|
WatchedByID: ctx.ContextUser.ID,
|
2024-03-01 02:52:49 +08:00
|
|
|
Collaborate: optional.Some(false),
|
2021-04-16 00:53:57 +08:00
|
|
|
TopicOnly: topicOnly,
|
2022-01-28 19:29:04 +08:00
|
|
|
Language: language,
|
2021-04-16 00:53:57 +08:00
|
|
|
IncludeDescription: setting.UI.SearchRepoDescription,
|
2024-03-03 18:18:34 +08:00
|
|
|
Archived: archived,
|
|
|
|
Fork: fork,
|
|
|
|
Mirror: mirror,
|
|
|
|
Template: template,
|
|
|
|
IsPrivate: private,
|
2021-04-16 00:53:57 +08:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("SearchRepository", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
total = int(count)
|
2023-07-07 02:59:24 +08:00
|
|
|
case "overview":
|
|
|
|
if bytes, err := profileReadme.GetBlobContent(setting.UI.MaxDisplayFileSize); err != nil {
|
|
|
|
log.Error("failed to GetBlobContent: %v", err)
|
|
|
|
} else {
|
2023-07-20 06:22:32 +08:00
|
|
|
if profileContent, err := markdown.RenderString(&markup.RenderContext{
|
2024-01-15 16:49:24 +08:00
|
|
|
Ctx: ctx,
|
|
|
|
GitRepo: profileGitRepo,
|
|
|
|
Links: markup.Links{
|
|
|
|
// Give the repo link to the markdown render for the full link of media element.
|
|
|
|
// the media link usually be like /[user]/[repoName]/media/branch/[branchName],
|
|
|
|
// Eg. /Tom/.profile/media/branch/main
|
|
|
|
// The branch shown on the profile page is the default branch, this need to be in sync with doc, see:
|
|
|
|
// https://docs.gitea.com/usage/profile-readme
|
|
|
|
Base: profileDbRepo.Link(),
|
|
|
|
BranchPath: path.Join("branch", util.PathEscapeSegments(profileDbRepo.DefaultBranch)),
|
|
|
|
},
|
|
|
|
Metas: map[string]string{"mode": "document"},
|
2023-07-20 06:22:32 +08:00
|
|
|
}, bytes); err != nil {
|
2023-07-07 02:59:24 +08:00
|
|
|
log.Error("failed to RenderString: %v", err)
|
|
|
|
} else {
|
|
|
|
ctx.Data["ProfileReadme"] = profileContent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
default: // default to "repositories"
|
2022-11-19 16:12:33 +08:00
|
|
|
repos, count, err = repo_model.SearchRepository(ctx, &repo_model.SearchRepoOptions{
|
2021-09-24 19:32:56 +08:00
|
|
|
ListOptions: db.ListOptions{
|
2023-02-25 05:15:10 +08:00
|
|
|
PageSize: pagingNum,
|
2020-01-25 03:00:29 +08:00
|
|
|
Page: page,
|
|
|
|
},
|
2022-03-22 15:03:22 +08:00
|
|
|
Actor: ctx.Doer,
|
2019-08-26 01:06:36 +08:00
|
|
|
Keyword: keyword,
|
2022-03-26 17:04:22 +08:00
|
|
|
OwnerID: ctx.ContextUser.ID,
|
2019-08-26 01:06:36 +08:00
|
|
|
OrderBy: orderBy,
|
|
|
|
Private: ctx.IsSigned,
|
2024-03-01 02:52:49 +08:00
|
|
|
Collaborate: optional.Some(false),
|
2019-08-26 01:06:36 +08:00
|
|
|
TopicOnly: topicOnly,
|
2022-01-28 19:29:04 +08:00
|
|
|
Language: language,
|
2019-08-26 01:06:36 +08:00
|
|
|
IncludeDescription: setting.UI.SearchRepoDescription,
|
2024-03-03 18:18:34 +08:00
|
|
|
Archived: archived,
|
|
|
|
Fork: fork,
|
|
|
|
Mirror: mirror,
|
|
|
|
Template: template,
|
|
|
|
IsPrivate: private,
|
2019-05-15 23:24:39 +08:00
|
|
|
})
|
|
|
|
if err != nil {
|
2019-08-26 01:06:36 +08:00
|
|
|
ctx.ServerError("SearchRepository", err)
|
2019-05-15 23:24:39 +08:00
|
|
|
return
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
2019-05-15 23:24:39 +08:00
|
|
|
|
|
|
|
total = int(count)
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
2019-04-20 12:15:19 +08:00
|
|
|
ctx.Data["Repos"] = repos
|
|
|
|
ctx.Data["Total"] = total
|
|
|
|
|
2023-08-12 01:08:05 +08:00
|
|
|
err = shared_user.LoadHeaderCount(ctx)
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("LoadHeaderCount", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-02-25 05:15:10 +08:00
|
|
|
pager := context.NewPagination(total, pagingNum, page, 5)
|
2019-04-20 12:15:19 +08:00
|
|
|
pager.SetDefaultParams(ctx)
|
2024-03-16 20:07:56 +08:00
|
|
|
pager.AddParamString("tab", tab)
|
2022-01-28 19:29:04 +08:00
|
|
|
if tab != "followers" && tab != "following" && tab != "activity" && tab != "projects" {
|
2024-03-16 20:07:56 +08:00
|
|
|
pager.AddParamString("language", language)
|
2022-01-28 19:29:04 +08:00
|
|
|
}
|
2023-02-25 05:15:10 +08:00
|
|
|
if tab == "activity" {
|
2024-03-16 20:07:56 +08:00
|
|
|
if ctx.Data["Date"] != nil {
|
|
|
|
pager.AddParamString("date", fmt.Sprint(ctx.Data["Date"]))
|
|
|
|
}
|
2023-02-25 05:15:10 +08:00
|
|
|
}
|
2024-08-16 20:41:45 +08:00
|
|
|
if archived.Has() {
|
|
|
|
pager.AddParamString("archived", fmt.Sprint(archived.Value()))
|
|
|
|
}
|
|
|
|
if fork.Has() {
|
|
|
|
pager.AddParamString("fork", fmt.Sprint(fork.Value()))
|
|
|
|
}
|
|
|
|
if mirror.Has() {
|
|
|
|
pager.AddParamString("mirror", fmt.Sprint(mirror.Value()))
|
|
|
|
}
|
|
|
|
if template.Has() {
|
|
|
|
pager.AddParamString("template", fmt.Sprint(template.Value()))
|
|
|
|
}
|
|
|
|
if private.Has() {
|
|
|
|
pager.AddParamString("private", fmt.Sprint(private.Value()))
|
|
|
|
}
|
2019-04-20 12:15:19 +08:00
|
|
|
ctx.Data["Page"] = pager
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
|
|
|
|
2016-11-18 11:03:03 +08:00
|
|
|
// Action response for follow/unfollow user request
|
2016-03-12 00:56:52 +08:00
|
|
|
func Action(ctx *context.Context) {
|
2015-12-21 20:24:11 +08:00
|
|
|
var err error
|
2021-12-21 01:18:26 +08:00
|
|
|
switch ctx.FormString("action") {
|
2015-12-21 20:24:11 +08:00
|
|
|
case "follow":
|
2024-03-04 16:16:03 +08:00
|
|
|
err = user_model.FollowUser(ctx, ctx.Doer, ctx.ContextUser)
|
2015-12-21 20:24:11 +08:00
|
|
|
case "unfollow":
|
2023-09-16 22:39:12 +08:00
|
|
|
err = user_model.UnfollowUser(ctx, ctx.Doer.ID, ctx.ContextUser.ID)
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2023-07-07 23:27:12 +08:00
|
|
|
log.Error("Failed to apply action %q: %v", ctx.FormString("action"), err)
|
2024-01-30 22:45:54 +08:00
|
|
|
ctx.Error(http.StatusBadRequest, fmt.Sprintf("Action %q failed", ctx.FormString("action")))
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
|
|
|
}
|
2024-01-30 22:45:54 +08:00
|
|
|
|
2024-02-17 13:13:37 +08:00
|
|
|
if ctx.ContextUser.IsIndividual() {
|
|
|
|
shared_user.PrepareContextForProfileBigAvatar(ctx)
|
|
|
|
ctx.HTML(http.StatusOK, tplProfileBigAvatar)
|
|
|
|
return
|
|
|
|
} else if ctx.ContextUser.IsOrganization() {
|
2024-02-23 08:24:57 +08:00
|
|
|
ctx.Data["Org"] = ctx.ContextUser
|
2024-02-17 13:13:37 +08:00
|
|
|
ctx.Data["IsFollowing"] = ctx.Doer != nil && user_model.IsFollowing(ctx, ctx.Doer.ID, ctx.ContextUser.ID)
|
|
|
|
ctx.HTML(http.StatusOK, tplFollowUnfollow)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
log.Error("Failed to apply action %q: unsupport context user type: %s", ctx.FormString("action"), ctx.ContextUser.Type)
|
|
|
|
ctx.Error(http.StatusBadRequest, fmt.Sprintf("Action %q failed", ctx.FormString("action")))
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|