Merge pull request #1725 from soudy/develop
Implemented #1721: see users who forked/starred/watched a repository
This commit is contained in:
		
						commit
						d86c785410
					
				| @ -527,6 +527,9 @@ func runWeb(ctx *cli.Context) { | ||||
| 			m.Get("/raw/*", repo.SingleDownload) | ||||
| 			m.Get("/commits/*", repo.RefCommits) | ||||
| 			m.Get("/commit/*", repo.Diff) | ||||
| 			m.Get("/stars", repo.Stars) | ||||
| 			m.Get("/watchers", repo.Watchers) | ||||
| 			m.Get("/forks", repo.Forks) | ||||
| 		}, middleware.RepoRef()) | ||||
| 
 | ||||
| 		m.Get("/compare/:before([a-z0-9]{40})...:after([a-z0-9]{40})", repo.CompareDiff) | ||||
|  | ||||
| @ -46,6 +46,9 @@ var ( | ||||
| 
 | ||||
| var ( | ||||
| 	Gitignores, Licenses, Readmes []string | ||||
| 
 | ||||
| 	// Maximum items per page in forks, watchers and stars of a repo
 | ||||
| 	ItemsPerPage = 54 | ||||
| ) | ||||
| 
 | ||||
| func LoadRepoConfig() { | ||||
| @ -1612,6 +1615,16 @@ func GetWatchers(rid int64) ([]*Watch, error) { | ||||
| 	return getWatchers(x, rid) | ||||
| } | ||||
| 
 | ||||
| // Repository.GetWatchers returns all users watching given repository.
 | ||||
| func (repo *Repository) GetWatchers(offset int) ([]*User, error) { | ||||
| 	users := make([]*User, 0, 10) | ||||
| 	offset = (offset - 1) * ItemsPerPage | ||||
| 
 | ||||
| 	err := x.Limit(ItemsPerPage, offset).Where("repo_id=?", repo.ID).Join("LEFT", "watch", "user.id=watch.user_id").Find(&users) | ||||
| 
 | ||||
| 	return users, err | ||||
| } | ||||
| 
 | ||||
| func notifyWatchers(e Engine, act *Action) error { | ||||
| 	// Add feeds for user self and all watchers.
 | ||||
| 	watches, err := getWatchers(e, act.RepoID) | ||||
| @ -1689,6 +1702,15 @@ func IsStaring(uid, repoId int64) bool { | ||||
| 	return has | ||||
| } | ||||
| 
 | ||||
| func (repo *Repository) GetStars(offset int) ([]*User, error) { | ||||
| 	users := make([]*User, 0, 10) | ||||
| 	offset = (offset - 1) * ItemsPerPage | ||||
| 
 | ||||
| 	err := x.Limit(ItemsPerPage, offset).Where("repo_id=?", repo.ID).Join("LEFT", "star", "user.id=star.uid").Find(&users) | ||||
| 
 | ||||
| 	return users, err | ||||
| } | ||||
| 
 | ||||
| // ___________           __
 | ||||
| // \_   _____/__________|  | __
 | ||||
| //  |    __)/  _ \_  __ \  |/ /
 | ||||
| @ -1756,3 +1778,11 @@ func ForkRepository(u *User, oldRepo *Repository, name, desc string) (_ *Reposit | ||||
| 
 | ||||
| 	return repo, sess.Commit() | ||||
| } | ||||
| 
 | ||||
| func (repo *Repository) GetForks() ([]*Repository, error) { | ||||
| 	forks := make([]*Repository, 0, 10) | ||||
| 
 | ||||
| 	err := x.Find(&forks, &Repository{ForkID: repo.ID}) | ||||
| 
 | ||||
| 	return forks, err | ||||
| } | ||||
|  | ||||
| @ -1947,6 +1947,94 @@ The register and sign-in page style | ||||
| #release #release-new-form { | ||||
|   padding-top: 15px; | ||||
| } | ||||
| #stars h4, | ||||
| #watchers h4, | ||||
| #forks h4 { | ||||
|   font-size: 18px; | ||||
|   padding-bottom: 20px; | ||||
|   text-transform: capitalize; | ||||
|   border-bottom: 1px solid #DDD; | ||||
| } | ||||
| #stars h3, | ||||
| #watchers h3, | ||||
| #forks h3 { | ||||
|   margin: -4px 0 0 0; | ||||
|   padding: 0; | ||||
| } | ||||
| #stars .avatar, | ||||
| #watchers .avatar, | ||||
| #forks .avatar { | ||||
|   width: 75px; | ||||
|   height: 75px; | ||||
|   float: left; | ||||
|   display: block; | ||||
|   margin-right: 10px; | ||||
| } | ||||
| #stars .avatar-small, | ||||
| #watchers .avatar-small, | ||||
| #forks .avatar-small { | ||||
|   width: 24px; | ||||
|   height: 24px; | ||||
|   float: left; | ||||
|   display: block; | ||||
|   margin-right: 10px; | ||||
| } | ||||
| #stars ol, | ||||
| #watchers ol, | ||||
| #forks ol { | ||||
|   margin-top: 10px; | ||||
|   list-style: none; | ||||
|   width: 100%; | ||||
|   overflow: hidden; | ||||
| } | ||||
| #stars li, | ||||
| #watchers li, | ||||
| #forks li { | ||||
|   width: 32.25%; | ||||
|   margin: 10px 10px 10px 0; | ||||
|   border-bottom: 1px solid #DDD; | ||||
|   float: left; | ||||
|   padding-bottom: 10px; | ||||
| } | ||||
| #stars .pagination, | ||||
| #watchers .pagination, | ||||
| #forks .pagination { | ||||
|   width: 100%; | ||||
|   text-align: center; | ||||
|   text-transform: capitalize; | ||||
| } | ||||
| #stars .pagination a, | ||||
| #watchers .pagination a, | ||||
| #forks .pagination a { | ||||
|   border-radius: 3px; | ||||
|   border: 1px solid #399ADE; | ||||
|   padding: 8px; | ||||
|   margin: 0; | ||||
| } | ||||
| #stars .pagination .active, | ||||
| #watchers .pagination .active, | ||||
| #forks .pagination .active { | ||||
|   border-radius: 3px; | ||||
|   border: 1px solid #399ADE; | ||||
|   background: #399ADE; | ||||
|   cursor: default; | ||||
|   padding: 8px; | ||||
|   margin: 0; | ||||
|   color: #FFFFFF; | ||||
| } | ||||
| #stars .pagination .disabled, | ||||
| #watchers .pagination .disabled, | ||||
| #forks .pagination .disabled { | ||||
|   border-radius: 3px; | ||||
|   border: 1px solid #DDD; | ||||
|   color: #D3D3D3; | ||||
|   cursor: default; | ||||
|   padding: 8px; | ||||
|   margin: 0; | ||||
| } | ||||
| #forks p { | ||||
|   padding: 5px 0; | ||||
| } | ||||
| #admin-wrapper, | ||||
| #setting-wrapper { | ||||
|   padding-bottom: 100px; | ||||
|  | ||||
| @ -795,3 +795,84 @@ | ||||
|         padding-top: 15px; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #stars, #watchers, #forks { | ||||
|     h4 { | ||||
|         font-size: 18px; | ||||
|         padding-bottom: 20px; | ||||
|         text-transform: capitalize; | ||||
|         border-bottom: 1px solid #DDD; | ||||
|     } | ||||
| 
 | ||||
|     h3 { | ||||
|         margin: -4px 0 0 0; | ||||
|         padding: 0; | ||||
|     } | ||||
| 
 | ||||
|     .avatar { | ||||
|         width: 75px; | ||||
|         height: 75px; | ||||
|         float: left; | ||||
|         display: block; | ||||
|         margin-right: 10px; | ||||
|     } | ||||
| 
 | ||||
|     .avatar-small { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         float: left; | ||||
|         display: block; | ||||
|         margin-right: 10px; | ||||
|     } | ||||
| 
 | ||||
|     ol { | ||||
|         margin-top: 10px; | ||||
|         list-style: none; | ||||
|         width: 100%; | ||||
|         overflow: hidden; | ||||
|     } | ||||
| 
 | ||||
|     li { | ||||
|         width: 32.25%; | ||||
|         margin: 10px 10px 10px 0; | ||||
|         border-bottom: 1px solid #DDD; | ||||
|         float: left; | ||||
|         padding-bottom: 10px; | ||||
|     } | ||||
| 
 | ||||
|     .pagination { | ||||
|         width: 100%; | ||||
|         text-align: center; | ||||
|         text-transform: capitalize; | ||||
| 
 | ||||
|         a { | ||||
|             border-radius: 3px; | ||||
|             border: 1px solid #399ADE; | ||||
|             padding: 8px; | ||||
|             margin: 0; | ||||
|         } | ||||
| 
 | ||||
|         .active { | ||||
|             border-radius: 3px; | ||||
|             border: 1px solid #399ADE; | ||||
|             background: #399ADE; | ||||
|             cursor: default; | ||||
|             padding: 8px; | ||||
|             margin: 0; | ||||
|             color: #FFFFFF; | ||||
|         } | ||||
| 
 | ||||
|         .disabled { | ||||
|             border-radius: 3px; | ||||
|             border: 1px solid #DDD; | ||||
|             color: #D3D3D3; | ||||
|             cursor: default; | ||||
|             padding: 8px; | ||||
|             margin: 0; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #forks p { | ||||
|     padding: 5px 0; | ||||
| } | ||||
|  | ||||
							
								
								
									
										37
									
								
								routers/repo/forks.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								routers/repo/forks.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | ||||
| // Copyright 2014 The Gogs Authors. All rights reserved.
 | ||||
| // Use of this source code is governed by a MIT-style
 | ||||
| // license that can be found in the LICENSE file.
 | ||||
| 
 | ||||
| package repo | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gogits/gogs/modules/base" | ||||
| 	"github.com/gogits/gogs/modules/middleware" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	FORKS base.TplName = "repo/forks" | ||||
| ) | ||||
| 
 | ||||
| func Forks(ctx *middleware.Context) { | ||||
| 	ctx.Data["Title"] = ctx.Tr("repos.forks") | ||||
| 
 | ||||
| 	forks, err := ctx.Repo.Repository.GetForks() | ||||
| 
 | ||||
| 	if err != nil { | ||||
| 		ctx.Handle(500, "GetForks", err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	for _, fork := range forks { | ||||
| 		if err = fork.GetOwner(); err != nil { | ||||
| 			ctx.Handle(500, "GetOwner", fmt.Errorf("%d: %v", fork.ID, err)) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	ctx.Data["Forks"] = forks | ||||
| 
 | ||||
| 	ctx.HTML(200, FORKS) | ||||
| } | ||||
							
								
								
									
										44
									
								
								routers/repo/stars.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								routers/repo/stars.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | ||||
| // Copyright 2014 The Gogs Authors. All rights reserved.
 | ||||
| // Use of this source code is governed by a MIT-style
 | ||||
| // license that can be found in the LICENSE file.
 | ||||
| 
 | ||||
| package repo | ||||
| 
 | ||||
| import ( | ||||
| 	"github.com/Unknwon/paginater" | ||||
| 
 | ||||
| 	"github.com/gogits/gogs/models" | ||||
| 	"github.com/gogits/gogs/modules/base" | ||||
| 	"github.com/gogits/gogs/modules/middleware" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	STARS base.TplName = "repo/stars" | ||||
| ) | ||||
| 
 | ||||
| func Stars(ctx *middleware.Context) { | ||||
| 	ctx.Data["Title"] = ctx.Tr("repos.stars") | ||||
| 
 | ||||
| 	page := ctx.QueryInt("page") | ||||
| 	if page <= 0 { | ||||
| 		page = 1 | ||||
| 	} | ||||
| 
 | ||||
| 	ctx.Data["Page"] = paginater.New(ctx.Repo.Repository.NumStars, models.ItemsPerPage, page, 5) | ||||
| 
 | ||||
| 	stars, err := ctx.Repo.Repository.GetStars(ctx.QueryInt("page")) | ||||
| 
 | ||||
| 	if err != nil { | ||||
| 		ctx.Handle(500, "GetStars", err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	if (ctx.QueryInt("page")-1)*models.ItemsPerPage > ctx.Repo.Repository.NumStars { | ||||
| 		ctx.Handle(404, "ctx.Repo.Repository.NumStars", nil) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	ctx.Data["Stars"] = stars | ||||
| 
 | ||||
| 	ctx.HTML(200, STARS) | ||||
| } | ||||
							
								
								
									
										44
									
								
								routers/repo/watchers.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								routers/repo/watchers.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | ||||
| // Copyright 2014 The Gogs Authors. All rights reserved.
 | ||||
| // Use of this source code is governed by a MIT-style
 | ||||
| // license that can be found in the LICENSE file.
 | ||||
| 
 | ||||
| package repo | ||||
| 
 | ||||
| import ( | ||||
| 	"github.com/Unknwon/paginater" | ||||
| 
 | ||||
| 	"github.com/gogits/gogs/models" | ||||
| 	"github.com/gogits/gogs/modules/base" | ||||
| 	"github.com/gogits/gogs/modules/middleware" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	WATCHERS base.TplName = "repo/watchers" | ||||
| ) | ||||
| 
 | ||||
| func Watchers(ctx *middleware.Context) { | ||||
| 	ctx.Data["Title"] = ctx.Tr("repos.watches") | ||||
| 
 | ||||
| 	page := ctx.QueryInt("page") | ||||
| 	if page <= 0 { | ||||
| 		page = 1 | ||||
| 	} | ||||
| 
 | ||||
| 	ctx.Data["Page"] = paginater.New(ctx.Repo.Repository.NumWatches, models.ItemsPerPage, page, 5) | ||||
| 
 | ||||
| 	watchers, err := ctx.Repo.Repository.GetWatchers(ctx.QueryInt("page")) | ||||
| 
 | ||||
| 	if err != nil { | ||||
| 		ctx.Handle(500, "GetWatchers", err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	if (ctx.QueryInt("page")-1)*models.ItemsPerPage > ctx.Repo.Repository.NumWatches { | ||||
| 		ctx.Handle(404, "ctx.Repo.Repository.NumWatches", nil) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	ctx.Data["Watchers"] = watchers | ||||
| 
 | ||||
| 	ctx.HTML(200, WATCHERS) | ||||
| } | ||||
							
								
								
									
										27
									
								
								templates/repo/forks.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								templates/repo/forks.tmpl
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | ||||
| {{template "ng/base/head" .}} | ||||
| {{template "ng/base/header" .}} | ||||
| <div id="repo-wrapper"> | ||||
|     {{template "repo/header_old" .}} | ||||
|     <div id="repo-content" class="clear container"> | ||||
|         <div id="repo-main" class="left grid-5-6"> | ||||
|             <div id="forks"> | ||||
|                 <h4> | ||||
|                     <strong>{{.i18n.Tr "repos.forks"}}</strong> | ||||
|                 </h4> | ||||
| 
 | ||||
|                 <ol> | ||||
|                 {{range .Forks}} | ||||
|                     <p> | ||||
|                     <img class="avatar-small" src="{{.Owner.AvatarLink}}"> | ||||
|                     <a href="{{AppSubUrl}}/{{.Owner.Name}}">{{.Owner.Name}}</a> | ||||
|                     / | ||||
|                     <a href="{{AppSubUrl}}/{{.Owner.Name}}/{{.Name}}">{{.Name}}</a> | ||||
|                     </p> | ||||
|                 {{end}} | ||||
|             </div> | ||||
|         </div> | ||||
| 
 | ||||
|         {{template "repo/sidebar" .}} | ||||
|     </div> | ||||
| </div> | ||||
| {{template "ng/base/footer" .}} | ||||
							
								
								
									
										61
									
								
								templates/repo/stars.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								templates/repo/stars.tmpl
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,61 @@ | ||||
| {{template "ng/base/head" .}} | ||||
| {{template "ng/base/header" .}} | ||||
| <div id="repo-wrapper"> | ||||
|     {{template "repo/header_old" .}} | ||||
|     <div id="repo-content" class="clear container"> | ||||
|         <div id="repo-main" class="left grid-5-6"> | ||||
|             <div id="stars"> | ||||
|                 <h4> | ||||
|                     <strong>{{.i18n.Tr "repos.stars"}}</strong> | ||||
|                 </h4> | ||||
| 
 | ||||
|                 <ol> | ||||
|                 {{range .Stars}} | ||||
|                 <li> | ||||
|                     <a href="{{AppSubUrl}}/{{.Name}}"> | ||||
|                     <img class="avatar" src="{{.AvatarLink}}" title="{{.Name}}"/> | ||||
| 
 | ||||
|                     <h3>{{.Name}}</h3> | ||||
|                     </a> | ||||
| 
 | ||||
|                     <p> | ||||
|                     {{if .Website}} | ||||
|                     <span class="octicon octicon-link"></span> <a href="{{.Website}}" target="_blank">{{.Website}}</a> | ||||
|                     {{else if .Location}} | ||||
|                     <span class="octicon octicon-location"></span> {{.Location}} | ||||
|                     {{else}} | ||||
|                     <span class="octicon octicon-clock"></span> {{$.i18n.Tr "user.join_on"}} {{DateFmtShort .Created}} | ||||
|                     {{end}} | ||||
|                     </p> | ||||
|                 </li> | ||||
|                 {{end}} | ||||
|                 </ol> | ||||
| 
 | ||||
|                 {{with .Page}} | ||||
|                 {{if gt .TotalPages 1}} | ||||
|                 <div class="pagination"> | ||||
|                     {{if .HasPrevious}} | ||||
|                     <a href="{{$.RepoLink}}/stars?page={{.Previous}}">{{$.i18n.Tr "issues.previous"}}</a> | ||||
|                     {{end}} | ||||
| 
 | ||||
|                     {{range .Pages}} | ||||
|                     {{if eq .Num -1}} | ||||
|                     <a class="disabled item">...</a> | ||||
|                     {{else}} | ||||
|                     <a class="{{if .IsCurrent}}active{{end}} item" {{if not .IsCurrent}}href="{{$.RepoLink}}/stars?page={{.Num}}"{{end}}>{{.Num}}</a> | ||||
|                     {{end}} | ||||
|                     {{end}} | ||||
| 
 | ||||
|                     {{if .HasNext}} | ||||
|                     <a href="{{$.RepoLink}}/stars?page={{.Next}}">{{$.i18n.Tr "issues.next"}}</a> | ||||
|                     {{end}} | ||||
|                 </div> | ||||
|                 {{end}} | ||||
|                 {{end}} | ||||
|             </div> | ||||
|         </div> | ||||
| 
 | ||||
|         {{template "repo/sidebar" .}} | ||||
|     </div> | ||||
| </div> | ||||
| {{template "ng/base/footer" .}} | ||||
							
								
								
									
										61
									
								
								templates/repo/watchers.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								templates/repo/watchers.tmpl
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,61 @@ | ||||
| {{template "ng/base/head" .}} | ||||
| {{template "ng/base/header" .}} | ||||
| <div id="repo-wrapper"> | ||||
|     {{template "repo/header_old" .}} | ||||
|     <div id="repo-content" class="clear container"> | ||||
|         <div id="repo-main" class="left grid-5-6"> | ||||
|             <div id="stars"> | ||||
|                 <h4> | ||||
|                     <strong>{{.i18n.Tr "repos.watches"}}</strong> | ||||
|                 </h4> | ||||
| 
 | ||||
|                 <ol> | ||||
|                 {{range .Watchers}} | ||||
|                 <li> | ||||
|                     <a href="{{AppSubUrl}}/{{.Name}}"> | ||||
|                     <img class="avatar" src="{{.AvatarLink}}" title="{{.Name}}"/> | ||||
| 
 | ||||
|                     <h3>{{.Name}}</h3> | ||||
|                     </a> | ||||
| 
 | ||||
|                     <p> | ||||
|                     {{if .Website}} | ||||
|                     <span class="octicon octicon-link"></span> <a href="{{.Website}}" target="_blank">{{.Website}}</a> | ||||
|                     {{else if .Location}} | ||||
|                     <span class="octicon octicon-location"></span> {{.Location}} | ||||
|                     {{else}} | ||||
|                     <span class="octicon octicon-clock"></span> {{$.i18n.Tr "user.join_on"}} {{DateFmtShort .Created}} | ||||
|                     {{end}} | ||||
|                     </p> | ||||
|                 </li> | ||||
|                 {{end}} | ||||
|                 </ol> | ||||
| 
 | ||||
|                 {{with .Page}} | ||||
|                 {{if gt .TotalPages 1}} | ||||
|                 <div class="pagination"> | ||||
|                     {{if .HasPrevious}} | ||||
|                     <a href="{{$.RepoLink}}/watchers?page={{.Previous}}">{{$.i18n.Tr "issues.previous"}}</a> | ||||
|                     {{end}} | ||||
| 
 | ||||
|                     {{range .Pages}} | ||||
|                     {{if eq .Num -1}} | ||||
|                     <a class="disabled item">...</a> | ||||
|                     {{else}} | ||||
|                     <a class="{{if .IsCurrent}}active{{end}} item" {{if not .IsCurrent}}href="{{$.RepoLink}}/watchers?page={{.Num}}"{{end}}>{{.Num}}</a> | ||||
|                     {{end}} | ||||
|                     {{end}} | ||||
| 
 | ||||
|                     {{if .HasNext}} | ||||
|                     <a href="{{$.RepoLink}}/watchers?page={{.Next}}">{{$.i18n.Tr "issues.next"}}</a> | ||||
|                     {{end}} | ||||
|                 </div> | ||||
|                 {{end}} | ||||
|                 {{end}} | ||||
|             </div> | ||||
|         </div> | ||||
| 
 | ||||
|         {{template "repo/sidebar" .}} | ||||
|     </div> | ||||
| </div> | ||||
| {{template "ng/base/footer" .}} | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user