2017-09-14 08:51:32 +02:00
// Copyright 2017 The Gitea Authors. All rights reserved.
2014-07-26 00:24:27 -04:00
// 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 (
"bytes"
2017-01-29 12:13:57 -08:00
"encoding/base64"
2016-08-09 12:56:00 -07:00
"fmt"
gotemplate "html/template"
2014-07-26 00:24:27 -04:00
"io/ioutil"
"path"
2017-01-29 12:13:57 -08:00
"strconv"
2014-07-26 00:24:27 -04:00
"strings"
2016-11-11 13:11:45 +01:00
"code.gitea.io/git"
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/context"
2016-12-06 18:58:31 +01:00
"code.gitea.io/gitea/modules/highlight"
2016-12-26 02:16:37 +01:00
"code.gitea.io/gitea/modules/lfs"
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/modules/log"
2017-04-21 15:01:08 +08:00
"code.gitea.io/gitea/modules/markup"
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/modules/setting"
2016-12-06 18:58:31 +01:00
"code.gitea.io/gitea/modules/templates"
2018-09-29 11:33:54 +03:00
2016-11-11 13:11:45 +01:00
"github.com/Unknwon/paginater"
2014-07-26 00:24:27 -04:00
)
const (
2019-01-18 00:01:04 +00:00
tplRepoEMPTY base . TplName = "repo/empty"
tplRepoHome base . TplName = "repo/home"
tplWatchers base . TplName = "repo/watchers"
tplForks base . TplName = "repo/forks"
2014-07-26 00:24:27 -04:00
)
2016-08-30 02:08:38 -07:00
func renderDirectory ( ctx * context . Context , treeLink string ) {
tree , err := ctx . Repo . Commit . SubTree ( ctx . Repo . TreePath )
if err != nil {
ctx . NotFoundOrServerError ( "Repo.Commit.SubTree" , git . IsErrNotExist , err )
return
2016-06-27 16:38:35 +08:00
}
2014-07-26 00:24:27 -04:00
2016-08-30 02:08:38 -07:00
entries , err := tree . ListEntries ( )
if err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "ListEntries" , err )
2016-08-30 02:08:38 -07:00
return
2014-07-26 00:24:27 -04:00
}
2017-09-19 11:37:03 +03:00
entries . CustomSort ( base . NaturalSortLess )
2014-07-26 00:24:27 -04:00
2016-08-30 02:08:38 -07:00
ctx . Data [ "Files" ] , err = entries . GetCommitsInfo ( ctx . Repo . Commit , ctx . Repo . TreePath )
2016-08-14 23:02:14 -07:00
if err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "GetCommitsInfo" , err )
2014-07-26 00:24:27 -04:00
return
}
2019-01-14 21:15:06 +02:00
// 3 for the extensions in exts[] in order
// the last one is for a readme that doesn't
// strictly match an extension
var readmeFiles [ 4 ] * git . Blob
var exts = [ ] string { ".md" , ".txt" , "" } // sorted by priority
2016-08-30 02:08:38 -07:00
for _ , entry := range entries {
2017-05-02 16:57:54 +08:00
if entry . IsDir ( ) {
continue
}
2019-01-14 21:15:06 +02:00
for i , ext := range exts {
if markup . IsReadmeFile ( entry . Name ( ) , ext ) {
readmeFiles [ i ] = entry . Blob ( )
}
}
if markup . IsReadmeFile ( entry . Name ( ) ) {
readmeFiles [ 3 ] = entry . Blob ( )
2016-08-30 02:08:38 -07:00
}
2019-01-14 21:15:06 +02:00
}
2016-08-30 02:08:38 -07:00
2019-01-14 21:15:06 +02:00
var readmeFile * git . Blob
for _ , f := range readmeFiles {
if f != nil {
readmeFile = f
2017-05-02 16:57:54 +08:00
break
}
2016-08-30 02:08:38 -07:00
}
if readmeFile != nil {
2016-08-31 13:59:23 -07:00
ctx . Data [ "RawFileLink" ] = ""
2016-08-30 02:08:38 -07:00
ctx . Data [ "ReadmeInList" ] = true
ctx . Data [ "ReadmeExist" ] = true
2017-11-29 02:50:39 +01:00
dataRc , err := readmeFile . DataAsync ( )
2016-08-14 23:02:14 -07:00
if err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "Data" , err )
2014-07-26 00:24:27 -04:00
return
2016-08-14 23:02:14 -07:00
}
2017-11-29 02:50:39 +01:00
defer dataRc . Close ( )
2014-07-26 00:24:27 -04:00
2016-08-14 23:02:14 -07:00
buf := make ( [ ] byte , 1024 )
n , _ := dataRc . Read ( buf )
2016-08-31 13:59:23 -07:00
buf = buf [ : n ]
2016-08-09 20:35:20 +01:00
2016-08-30 02:08:38 -07:00
isTextFile := base . IsTextFile ( buf )
ctx . Data [ "FileIsText" ] = isTextFile
ctx . Data [ "FileName" ] = readmeFile . Name ( )
// FIXME: what happens when README file is an image?
if isTextFile {
2017-11-29 02:50:39 +01:00
if readmeFile . Size ( ) >= setting . UI . MaxDisplayFileSize {
// Pretend that this is a normal text file to display 'This file is too large to be shown'
ctx . Data [ "IsFileTooLarge" ] = true
ctx . Data [ "IsTextFile" ] = true
ctx . Data [ "FileSize" ] = readmeFile . Size ( )
2017-04-21 15:01:08 +08:00
} else {
2017-11-29 02:50:39 +01:00
d , _ := ioutil . ReadAll ( dataRc )
2018-09-29 11:33:54 +03:00
buf = templates . ToUTF8WithFallback ( append ( buf , d ... ) )
2017-11-29 02:50:39 +01:00
if markup . Type ( readmeFile . Name ( ) ) != "" {
ctx . Data [ "IsMarkup" ] = true
ctx . Data [ "FileContent" ] = string ( markup . Render ( readmeFile . Name ( ) , buf , treeLink , ctx . Repo . Repository . ComposeMetas ( ) ) )
} else {
ctx . Data [ "IsRenderedHTML" ] = true
2018-06-10 11:42:16 -07:00
ctx . Data [ "FileContent" ] = strings . Replace (
gotemplate . HTMLEscapeString ( string ( buf ) ) , "\n" , ` <br> ` , - 1 ,
)
2017-11-29 02:50:39 +01:00
}
2014-07-26 00:24:27 -04:00
}
2016-08-14 23:02:14 -07:00
}
2016-08-30 02:08:38 -07:00
}
2016-08-14 23:02:14 -07:00
2016-08-30 02:08:38 -07:00
// Show latest commit info of repository in table header,
// or of directory if not in root directory.
latestCommit := ctx . Repo . Commit
if len ( ctx . Repo . TreePath ) > 0 {
latestCommit , err = ctx . Repo . Commit . GetCommitByPath ( ctx . Repo . TreePath )
2014-07-26 00:24:27 -04:00
if err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "GetCommitByPath" , err )
2014-07-26 00:24:27 -04:00
return
}
2016-08-30 02:08:38 -07:00
}
ctx . Data [ "LatestCommit" ] = latestCommit
2017-05-05 10:15:36 +03:00
ctx . Data [ "LatestCommitVerification" ] = models . ParseCommitWithSignature ( latestCommit )
2016-08-30 02:08:38 -07:00
ctx . Data [ "LatestCommitUser" ] = models . ValidateCommitWithEmail ( latestCommit )
2017-09-14 08:51:32 +02:00
statuses , err := models . GetLatestCommitStatus ( ctx . Repo . Repository , ctx . Repo . Commit . ID . String ( ) , 0 )
if err != nil {
log . Error ( 3 , "GetLatestCommitStatus: %v" , err )
}
ctx . Data [ "LatestCommitStatus" ] = models . CalcCommitStatus ( statuses )
2016-08-30 02:08:38 -07:00
// Check permission to add or upload new file.
2018-11-28 19:26:14 +08:00
if ctx . Repo . CanWrite ( models . UnitTypeCode ) && ctx . Repo . IsViewBranch {
2019-01-23 19:58:38 +01:00
ctx . Data [ "CanAddFile" ] = ! ctx . Repo . Repository . IsArchived
ctx . Data [ "CanUploadFile" ] = setting . Repository . Upload . Enabled && ! ctx . Repo . Repository . IsArchived
2016-08-30 02:08:38 -07:00
}
}
2014-09-30 04:39:53 -04:00
2016-08-30 02:08:38 -07:00
func renderFile ( ctx * context . Context , entry * git . TreeEntry , treeLink , rawLink string ) {
ctx . Data [ "IsViewFile" ] = true
2014-07-26 00:24:27 -04:00
2016-08-30 02:08:38 -07:00
blob := entry . Blob ( )
2017-11-29 02:50:39 +01:00
dataRc , err := blob . DataAsync ( )
2016-08-30 02:08:38 -07:00
if err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "DataAsync" , err )
2016-08-30 02:08:38 -07:00
return
}
2017-11-29 02:50:39 +01:00
defer dataRc . Close ( )
2014-07-26 00:24:27 -04:00
2018-09-10 18:01:49 +01:00
ctx . Data [ "Title" ] = ctx . Data [ "Title" ] . ( string ) + " - " + ctx . Repo . TreePath + " at " + ctx . Repo . BranchName
2016-08-30 02:08:38 -07:00
ctx . Data [ "FileSize" ] = blob . Size ( )
ctx . Data [ "FileName" ] = blob . Name ( )
ctx . Data [ "HighlightClass" ] = highlight . FileNameToHighlightClass ( blob . Name ( ) )
ctx . Data [ "RawFileLink" ] = rawLink + "/" + ctx . Repo . TreePath
buf := make ( [ ] byte , 1024 )
n , _ := dataRc . Read ( buf )
2016-08-31 13:59:23 -07:00
buf = buf [ : n ]
2016-08-30 02:08:38 -07:00
isTextFile := base . IsTextFile ( buf )
ctx . Data [ "IsTextFile" ] = isTextFile
2016-12-26 02:16:37 +01:00
//Check for LFS meta file
if isTextFile && setting . LFS . StartServer {
headString := string ( buf )
if strings . HasPrefix ( headString , models . LFSMetaFileIdentifier ) {
splitLines := strings . Split ( headString , "\n" )
if len ( splitLines ) >= 3 {
oid := strings . TrimPrefix ( splitLines [ 1 ] , models . LFSMetaFileOidPrefix )
size , err := strconv . ParseInt ( strings . TrimPrefix ( splitLines [ 2 ] , "size " ) , 10 , 64 )
if len ( oid ) == 64 && err == nil {
contentStore := & lfs . ContentStore { BasePath : setting . LFS . ContentPath }
meta := & models . LFSMetaObject { Oid : oid }
if contentStore . Exists ( meta ) {
ctx . Data [ "IsTextFile" ] = false
isTextFile = false
ctx . Data [ "IsLFSFile" ] = true
ctx . Data [ "FileSize" ] = size
filenameBase64 := base64 . RawURLEncoding . EncodeToString ( [ ] byte ( blob . Name ( ) ) )
2017-12-08 13:21:37 +01:00
ctx . Data [ "RawFileLink" ] = fmt . Sprintf ( "%s%s.git/info/lfs/objects/%s/%s" , setting . AppURL , ctx . Repo . Repository . FullName ( ) , oid , filenameBase64 )
2016-12-26 02:16:37 +01:00
}
}
}
}
}
2016-08-30 02:08:38 -07:00
// Assume file is not editable first.
if ! isTextFile {
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.cannot_edit_non_text_files" )
}
switch {
case isTextFile :
if blob . Size ( ) >= setting . UI . MaxDisplayFileSize {
ctx . Data [ "IsFileTooLarge" ] = true
break
2014-07-26 00:24:27 -04:00
}
2016-08-30 02:08:38 -07:00
d , _ := ioutil . ReadAll ( dataRc )
2018-09-29 11:33:54 +03:00
buf = templates . ToUTF8WithFallback ( append ( buf , d ... ) )
2014-07-26 00:24:27 -04:00
2017-09-21 13:20:14 +08:00
readmeExist := markup . IsReadmeFile ( blob . Name ( ) )
2016-08-30 02:08:38 -07:00
ctx . Data [ "ReadmeExist" ] = readmeExist
2017-10-16 15:04:34 +08:00
if markup . Type ( blob . Name ( ) ) != "" {
2017-10-17 01:17:22 +02:00
ctx . Data [ "IsMarkup" ] = true
2017-04-21 15:01:08 +08:00
ctx . Data [ "FileContent" ] = string ( markup . Render ( blob . Name ( ) , buf , path . Dir ( treeLink ) , ctx . Repo . Repository . ComposeMetas ( ) ) )
2017-09-22 02:24:19 +08:00
} else if readmeExist {
2017-10-16 15:04:34 +08:00
ctx . Data [ "IsRenderedHTML" ] = true
2018-06-10 11:42:16 -07:00
ctx . Data [ "FileContent" ] = strings . Replace (
gotemplate . HTMLEscapeString ( string ( buf ) ) , "\n" , ` <br> ` , - 1 ,
)
2016-08-30 02:08:38 -07:00
} else {
// Building code view blocks with line number on server side.
var fileContent string
2016-12-06 18:58:31 +01:00
if content , err := templates . ToUTF8WithErr ( buf ) ; err != nil {
2016-08-30 02:08:38 -07:00
if err != nil {
2017-01-29 12:13:57 -08:00
log . Error ( 4 , "ToUTF8WithErr: %v" , err )
2014-07-26 00:24:27 -04:00
}
2016-08-30 02:08:38 -07:00
fileContent = string ( buf )
} else {
fileContent = content
2014-07-26 00:24:27 -04:00
}
2014-09-26 08:55:13 -04:00
2016-08-30 02:08:38 -07:00
var output bytes . Buffer
lines := strings . Split ( fileContent , "\n" )
2018-05-29 06:08:55 +03:00
//Remove blank line at the end of file
if len ( lines ) > 0 && lines [ len ( lines ) - 1 ] == "" {
lines = lines [ : len ( lines ) - 1 ]
}
2016-08-30 02:08:38 -07:00
for index , line := range lines {
2017-06-10 17:20:25 +02:00
line = gotemplate . HTMLEscapeString ( line )
if index != len ( lines ) - 1 {
line += "\n"
}
output . WriteString ( fmt . Sprintf ( ` <li class="L%d" rel="L%d">%s</li> ` , index + 1 , index + 1 , line ) )
2014-09-26 08:55:13 -04:00
}
2016-08-30 02:08:38 -07:00
ctx . Data [ "FileContent" ] = gotemplate . HTML ( output . String ( ) )
output . Reset ( )
for i := 0 ; i < len ( lines ) ; i ++ {
output . WriteString ( fmt . Sprintf ( ` <span id="L%d">%d</span> ` , i + 1 , i + 1 ) )
}
ctx . Data [ "LineNums" ] = gotemplate . HTML ( output . String ( ) )
2014-09-26 08:55:13 -04:00
}
2016-08-30 02:08:38 -07:00
if ctx . Repo . CanEnableEditor ( ) {
ctx . Data [ "CanEditFile" ] = true
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.edit_this_file" )
} else if ! ctx . Repo . IsViewBranch {
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.must_be_on_a_branch" )
2018-11-28 19:26:14 +08:00
} else if ! ctx . Repo . CanWrite ( models . UnitTypeCode ) {
2016-08-30 02:08:38 -07:00
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.fork_before_edit" )
2016-08-28 01:41:44 -07:00
}
2016-08-30 02:08:38 -07:00
case base . IsPDFFile ( buf ) :
ctx . Data [ "IsPDFFile" ] = true
2016-12-20 09:09:11 +01:00
case base . IsVideoFile ( buf ) :
ctx . Data [ "IsVideoFile" ] = true
2018-10-30 03:17:26 +01:00
case base . IsAudioFile ( buf ) :
ctx . Data [ "IsAudioFile" ] = true
2016-08-30 02:08:38 -07:00
case base . IsImageFile ( buf ) :
ctx . Data [ "IsImageFile" ] = true
2014-07-26 00:24:27 -04:00
}
2016-08-30 02:08:38 -07:00
if ctx . Repo . CanEnableEditor ( ) {
ctx . Data [ "CanDeleteFile" ] = true
ctx . Data [ "DeleteFileTooltip" ] = ctx . Tr ( "repo.editor.delete_this_file" )
} else if ! ctx . Repo . IsViewBranch {
ctx . Data [ "DeleteFileTooltip" ] = ctx . Tr ( "repo.editor.must_be_on_a_branch" )
2018-11-28 19:26:14 +08:00
} else if ! ctx . Repo . CanWrite ( models . UnitTypeCode ) {
2016-08-30 02:08:38 -07:00
ctx . Data [ "DeleteFileTooltip" ] = ctx . Tr ( "repo.editor.must_have_write_access" )
}
}
2016-11-21 18:03:29 +08:00
// Home render repository home page
2016-08-30 02:08:38 -07:00
func Home ( ctx * context . Context ) {
2018-11-28 19:26:14 +08:00
if len ( ctx . Repo . Units ) > 0 {
2017-10-01 15:50:56 +02:00
var firstUnit * models . Unit
2018-11-28 19:26:14 +08:00
for _ , repoUnit := range ctx . Repo . Units {
2017-10-01 15:50:56 +02:00
if repoUnit . Type == models . UnitTypeCode {
renderCode ( ctx )
return
}
unit , ok := models . Units [ repoUnit . Type ]
if ok && ( firstUnit == nil || ! firstUnit . IsLessThan ( unit ) ) {
firstUnit = & unit
}
2017-05-18 22:54:24 +08:00
}
2017-10-01 15:50:56 +02:00
if firstUnit != nil {
ctx . Redirect ( fmt . Sprintf ( "%s/%s%s" , setting . AppSubURL , ctx . Repo . Repository . FullName ( ) , firstUnit . URI ) )
2017-05-18 22:54:24 +08:00
return
}
}
2018-01-10 22:34:17 +01:00
ctx . NotFound ( "Home" , fmt . Errorf ( ctx . Tr ( "units.error.no_unit_allowed_repo" ) ) )
2017-05-18 22:54:24 +08:00
}
func renderCode ( ctx * context . Context ) {
2017-03-18 05:59:07 -05:00
ctx . Data [ "PageIsViewCode" ] = true
2019-01-18 00:01:04 +00:00
if ctx . Repo . Repository . IsEmpty {
ctx . HTML ( 200 , tplRepoEMPTY )
2017-03-18 05:59:07 -05:00
return
}
2016-08-30 02:08:38 -07:00
title := ctx . Repo . Repository . Owner . Name + "/" + ctx . Repo . Repository . Name
if len ( ctx . Repo . Repository . Description ) > 0 {
title += ": " + ctx . Repo . Repository . Description
}
ctx . Data [ "Title" ] = title
ctx . Data [ "RequireHighlightJS" ] = true
2017-10-29 19:04:25 -07:00
branchLink := ctx . Repo . RepoLink + "/src/" + ctx . Repo . BranchNameSubURL ( )
2016-08-30 02:08:38 -07:00
treeLink := branchLink
2017-10-29 19:04:25 -07:00
rawLink := ctx . Repo . RepoLink + "/raw/" + ctx . Repo . BranchNameSubURL ( )
2016-08-30 02:08:38 -07:00
if len ( ctx . Repo . TreePath ) > 0 {
treeLink += "/" + ctx . Repo . TreePath
}
2018-04-11 10:51:44 +08:00
// Get Topics of this repo
topics , err := models . FindTopics ( & models . FindTopicOptions {
RepoID : ctx . Repo . Repository . ID ,
} )
if err != nil {
ctx . ServerError ( "models.FindTopics" , err )
return
}
ctx . Data [ "Topics" ] = topics
2016-08-30 02:08:38 -07:00
// Get current entry user currently looking at.
entry , err := ctx . Repo . Commit . GetTreeEntryByPath ( ctx . Repo . TreePath )
if err != nil {
ctx . NotFoundOrServerError ( "Repo.Commit.GetTreeEntryByPath" , git . IsErrNotExist , err )
return
}
if entry . IsDir ( ) {
renderDirectory ( ctx , treeLink )
} else {
renderFile ( ctx , entry , treeLink , rawLink )
}
if ctx . Written ( ) {
return
}
2014-07-26 00:24:27 -04:00
2016-08-30 02:08:38 -07:00
var treeNames [ ] string
paths := make ( [ ] string , 0 , 5 )
if len ( ctx . Repo . TreePath ) > 0 {
treeNames = strings . Split ( ctx . Repo . TreePath , "/" )
for i := range treeNames {
paths = append ( paths , strings . Join ( treeNames [ : i + 1 ] , "/" ) )
2014-07-26 00:24:27 -04:00
}
ctx . Data [ "HasParentPath" ] = true
2016-08-09 12:56:00 -07:00
if len ( paths ) - 2 >= 0 {
ctx . Data [ "ParentPath" ] = "/" + paths [ len ( paths ) - 2 ]
2014-07-26 00:24:27 -04:00
}
}
2016-08-09 12:56:00 -07:00
ctx . Data [ "Paths" ] = paths
2016-08-27 15:25:01 -07:00
ctx . Data [ "TreeLink" ] = treeLink
2016-08-30 02:08:38 -07:00
ctx . Data [ "TreeNames" ] = treeNames
2014-07-26 00:24:27 -04:00
ctx . Data [ "BranchLink" ] = branchLink
2016-11-21 18:03:29 +08:00
ctx . HTML ( 200 , tplRepoHome )
2014-07-26 00:24:27 -04:00
}
2015-11-16 23:28:46 -05:00
2017-02-28 12:56:15 +08:00
// RenderUserCards render a page show users according the input templaet
2016-03-11 11:56:52 -05:00
func RenderUserCards ( ctx * context . Context , total int , getter func ( page int ) ( [ ] * models . User , error ) , tpl base . TplName ) {
2015-11-16 23:28:46 -05:00
page := ctx . QueryInt ( "page" )
if page <= 0 {
page = 1
}
pager := paginater . New ( total , models . ItemsPerPage , page , 5 )
ctx . Data [ "Page" ] = pager
items , err := getter ( pager . Current ( ) )
if err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "getter" , err )
2015-11-16 23:28:46 -05:00
return
}
2015-12-21 04:24:11 -08:00
ctx . Data [ "Cards" ] = items
2015-11-16 23:28:46 -05:00
2015-12-21 04:24:11 -08:00
ctx . HTML ( 200 , tpl )
2015-11-16 23:28:46 -05:00
}
2016-11-21 18:03:29 +08:00
// Watchers render repository's watch users
2016-03-11 11:56:52 -05:00
func Watchers ( ctx * context . Context ) {
2015-11-16 23:28:46 -05:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.watchers" )
2015-12-21 04:24:11 -08:00
ctx . Data [ "CardsTitle" ] = ctx . Tr ( "repo.watchers" )
2015-11-16 23:28:46 -05:00
ctx . Data [ "PageIsWatchers" ] = true
2016-11-21 18:03:29 +08:00
RenderUserCards ( ctx , ctx . Repo . Repository . NumWatches , ctx . Repo . Repository . GetWatchers , tplWatchers )
2015-11-16 23:28:46 -05:00
}
2016-11-21 18:03:29 +08:00
// Stars render repository's starred users
2016-03-11 11:56:52 -05:00
func Stars ( ctx * context . Context ) {
2015-11-16 23:28:46 -05:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.stargazers" )
2015-12-21 04:24:11 -08:00
ctx . Data [ "CardsTitle" ] = ctx . Tr ( "repo.stargazers" )
2015-11-16 23:28:46 -05:00
ctx . Data [ "PageIsStargazers" ] = true
2016-11-21 18:03:29 +08:00
RenderUserCards ( ctx , ctx . Repo . Repository . NumStars , ctx . Repo . Repository . GetStargazers , tplWatchers )
2015-11-16 23:28:46 -05:00
}
2015-11-16 23:33:40 -05:00
2016-11-21 18:03:29 +08:00
// Forks render repository's forked users
2016-03-11 11:56:52 -05:00
func Forks ( ctx * context . Context ) {
2015-11-16 23:33:40 -05:00
ctx . Data [ "Title" ] = ctx . Tr ( "repos.forks" )
forks , err := ctx . Repo . Repository . GetForks ( )
if err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "GetForks" , err )
2015-11-16 23:33:40 -05:00
return
}
for _ , fork := range forks {
if err = fork . GetOwner ( ) ; err != nil {
2018-01-10 22:34:17 +01:00
ctx . ServerError ( "GetOwner" , err )
2015-11-16 23:33:40 -05:00
return
}
}
ctx . Data [ "Forks" ] = forks
2016-11-21 18:03:29 +08:00
ctx . HTML ( 200 , tplForks )
2015-11-16 23:33:40 -05:00
}