mirror of
https://github.com/gitbucket/gitbucket.git
synced 2025-11-03 12:05:59 +01:00
Merge branch 'master' into fork-and-pullreq
Conflicts: src/main/scala/app/CreateRepositoryController.scala src/main/scala/service/WikiService.scala src/main/scala/util/JGitUtil.scala
This commit is contained in:
Binary file not shown.
|
Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 473 B |
@@ -58,7 +58,10 @@ trait AccountControllerBase extends AccountManagementControllerBase with FlashMa
|
|||||||
getAccountByUserName(userName).flatMap(_.image).map { image =>
|
getAccountByUserName(userName).flatMap(_.image).map { image =>
|
||||||
contentType = FileUtil.getMimeType(image)
|
contentType = FileUtil.getMimeType(image)
|
||||||
new java.io.File(getUserUploadDir(userName), image)
|
new java.io.File(getUserUploadDir(userName), image)
|
||||||
} getOrElse NotFound
|
} getOrElse {
|
||||||
|
contentType = "image/png"
|
||||||
|
Thread.currentThread.getContextClassLoader.getResourceAsStream("noimage.png")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get("/:userName/_edit")(oneselfOnly {
|
get("/:userName/_edit")(oneselfOnly {
|
||||||
@@ -76,7 +79,7 @@ trait AccountControllerBase extends AccountManagementControllerBase with FlashMa
|
|||||||
|
|
||||||
updateImage(userName, form.fileId, form.clearImage)
|
updateImage(userName, form.fileId, form.clearImage)
|
||||||
flash += "info" -> "Account information has been updated."
|
flash += "info" -> "Account information has been updated."
|
||||||
redirect("/%s/_edit".format(userName))
|
redirect(s"/${userName}/_edit")
|
||||||
|
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -69,9 +69,13 @@ trait CreateRepositoryControllerBase extends ControllerBase {
|
|||||||
// Create README.md
|
// Create README.md
|
||||||
FileUtils.writeStringToFile(new File(tmpdir, "README.md"),
|
FileUtils.writeStringToFile(new File(tmpdir, "README.md"),
|
||||||
if(form.description.nonEmpty){
|
if(form.description.nonEmpty){
|
||||||
form.name + "\n===============\n\n" + form.description.get
|
form.name + "\n" +
|
||||||
|
"===============\n" +
|
||||||
|
"\n" +
|
||||||
|
form.description.get
|
||||||
} else {
|
} else {
|
||||||
form.name + "\n===============\n"
|
form.name + "\n" +
|
||||||
|
"===============\n"
|
||||||
}, "UTF-8")
|
}, "UTF-8")
|
||||||
|
|
||||||
val git = Git.open(tmpdir)
|
val git = Git.open(tmpdir)
|
||||||
@@ -91,7 +95,7 @@ trait CreateRepositoryControllerBase extends ControllerBase {
|
|||||||
recordCreateRepositoryActivity(loginUserName, form.name, loginUserName)
|
recordCreateRepositoryActivity(loginUserName, form.name, loginUserName)
|
||||||
|
|
||||||
// redirect to the repository
|
// redirect to the repository
|
||||||
redirect("/%s/%s".format(loginUserName, form.name))
|
redirect(s"/${loginUserName}/${form.name}")
|
||||||
})
|
})
|
||||||
|
|
||||||
post("/:owner/:repository/_fork")(referrersOnly { repository =>
|
post("/:owner/:repository/_fork")(referrersOnly { repository =>
|
||||||
|
|||||||
@@ -9,10 +9,13 @@ trait IndexControllerBase extends ControllerBase { self: RepositoryService
|
|||||||
with SystemSettingsService with ActivityService =>
|
with SystemSettingsService with ActivityService =>
|
||||||
|
|
||||||
get("/"){
|
get("/"){
|
||||||
|
val loginAccount = context.loginAccount
|
||||||
|
|
||||||
html.index(getRecentActivities(),
|
html.index(getRecentActivities(),
|
||||||
getAccessibleRepositories(context.loginAccount, baseUrl),
|
getAccessibleRepositories(loginAccount, baseUrl),
|
||||||
loadSystemSettings(),
|
loadSystemSettings(),
|
||||||
context.loginAccount.map{ account => getRepositoryNamesOfUser(account.userName) }.getOrElse(Nil))
|
loginAccount.map{ account => getRepositoryNamesOfUser(account.userName) }.getOrElse(Nil)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -17,10 +17,9 @@ trait IssuesControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
case class IssueCreateForm(title: String, content: Option[String],
|
case class IssueCreateForm(title: String, content: Option[String],
|
||||||
assignedUserName: Option[String], milestoneId: Option[Int], labelNames: Option[String])
|
assignedUserName: Option[String], milestoneId: Option[Int], labelNames: Option[String])
|
||||||
|
|
||||||
case class IssueEditForm(title: String, content: Option[String])
|
case class IssueEditForm(title: String, content: Option[String])
|
||||||
|
|
||||||
case class CommentForm(issueId: Int, content: String)
|
case class CommentForm(issueId: Int, content: String)
|
||||||
|
case class IssueStateForm(issueId: Int, content: Option[String])
|
||||||
|
|
||||||
val issueCreateForm = mapping(
|
val issueCreateForm = mapping(
|
||||||
"title" -> trim(label("Title", text(required))),
|
"title" -> trim(label("Title", text(required))),
|
||||||
@@ -40,6 +39,11 @@ trait IssuesControllerBase extends ControllerBase {
|
|||||||
"content" -> trim(label("Comment", text(required)))
|
"content" -> trim(label("Comment", text(required)))
|
||||||
)(CommentForm.apply)
|
)(CommentForm.apply)
|
||||||
|
|
||||||
|
val issueStateForm = mapping(
|
||||||
|
"issueId" -> label("Issue Id", number()),
|
||||||
|
"content" -> trim(optional(text()))
|
||||||
|
)(IssueStateForm.apply)
|
||||||
|
|
||||||
get("/:owner/:repository/issues")(referrersOnly {
|
get("/:owner/:repository/issues")(referrersOnly {
|
||||||
searchIssues("all", _)
|
searchIssues("all", _)
|
||||||
})
|
})
|
||||||
@@ -124,29 +128,11 @@ trait IssuesControllerBase extends ControllerBase {
|
|||||||
})
|
})
|
||||||
|
|
||||||
post("/:owner/:repository/issue_comments/new", commentForm)(readableUsersOnly { (form, repository) =>
|
post("/:owner/:repository/issue_comments/new", commentForm)(readableUsersOnly { (form, repository) =>
|
||||||
val owner = repository.owner
|
handleComment(form.issueId, Some(form.content), repository)
|
||||||
val name = repository.name
|
})
|
||||||
val userName = context.loginAccount.get.userName
|
|
||||||
|
|
||||||
getIssue(owner, name, form.issueId.toString).map { issue =>
|
post("/:owner/:repository/issue_comments/state", issueStateForm)(readableUsersOnly { (form, repository) =>
|
||||||
val action = if(isEditable(owner, name, issue.openedUserName)){
|
handleComment(form.issueId, form.content, repository)
|
||||||
params.get("action") filter { action =>
|
|
||||||
updateClosed(owner, name, form.issueId, if(action == "close") true else false) > 0
|
|
||||||
}
|
|
||||||
} else None
|
|
||||||
|
|
||||||
val commentId = createComment(owner, name, userName, form.issueId, form.content, action)
|
|
||||||
|
|
||||||
// record activity
|
|
||||||
recordCommentIssueActivity(owner, name, userName, issue.issueId, form.content)
|
|
||||||
action match {
|
|
||||||
case Some("reopen") => recordReopenIssueActivity(owner, name, userName, issue.issueId, issue.title)
|
|
||||||
case Some("close") => recordCloseIssueActivity(owner, name, userName, issue.issueId, issue.title)
|
|
||||||
case _ =>
|
|
||||||
}
|
|
||||||
|
|
||||||
redirect("/%s/%s/issues/%d#comment-%d".format(owner, name, form.issueId, commentId))
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
|
|
||||||
ajaxPost("/:owner/:repository/issue_comments/edit/:id", commentForm)(readableUsersOnly { (form, repository) =>
|
ajaxPost("/:owner/:repository/issue_comments/edit/:id", commentForm)(readableUsersOnly { (form, repository) =>
|
||||||
@@ -172,7 +158,7 @@ trait IssuesControllerBase extends ControllerBase {
|
|||||||
org.json4s.jackson.Serialization.write(
|
org.json4s.jackson.Serialization.write(
|
||||||
Map("title" -> x.title,
|
Map("title" -> x.title,
|
||||||
"content" -> view.Markdown.toHtml(x.content getOrElse "No description given.",
|
"content" -> view.Markdown.toHtml(x.content getOrElse "No description given.",
|
||||||
repository, false, true, true)
|
repository, false, true)
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
} else Unauthorized
|
} else Unauthorized
|
||||||
@@ -189,7 +175,7 @@ trait IssuesControllerBase extends ControllerBase {
|
|||||||
contentType = formats("json")
|
contentType = formats("json")
|
||||||
org.json4s.jackson.Serialization.write(
|
org.json4s.jackson.Serialization.write(
|
||||||
Map("content" -> view.Markdown.toHtml(x.content,
|
Map("content" -> view.Markdown.toHtml(x.content,
|
||||||
repository, false, true, true)
|
repository, false, true)
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
} else Unauthorized
|
} else Unauthorized
|
||||||
@@ -222,9 +208,85 @@ trait IssuesControllerBase extends ControllerBase {
|
|||||||
Ok("updated")
|
Ok("updated")
|
||||||
})
|
})
|
||||||
|
|
||||||
|
post("/:owner/:repository/issues/batchedit/state")(collaboratorsOnly { repository =>
|
||||||
|
val owner = repository.owner
|
||||||
|
val name = repository.name
|
||||||
|
val userName = context.loginAccount.get.userName
|
||||||
|
|
||||||
|
params.get("value") collect {
|
||||||
|
case s if s == "close" => (s.capitalize, Some(s), true)
|
||||||
|
case s if s == "reopen" => (s.capitalize, Some(s), false)
|
||||||
|
} map { case (content, action, closed) =>
|
||||||
|
params("checked").split(',') foreach { issueId =>
|
||||||
|
createComment(owner, name, userName, issueId.toInt, content, action)
|
||||||
|
updateClosed(owner, name, issueId.toInt, closed)
|
||||||
|
}
|
||||||
|
redirect("/%s/%s/issues".format(owner, name))
|
||||||
|
} getOrElse NotFound
|
||||||
|
})
|
||||||
|
|
||||||
|
post("/:owner/:repository/issues/batchedit/label")(collaboratorsOnly { repository =>
|
||||||
|
val owner = repository.owner
|
||||||
|
val name = repository.name
|
||||||
|
|
||||||
|
params.get("value").map(_.toInt) map { labelId =>
|
||||||
|
params("checked").split(',') foreach { issueId =>
|
||||||
|
getIssueLabel(owner, name, issueId.toInt, labelId) getOrElse {
|
||||||
|
registerIssueLabel(owner, name, issueId.toInt, labelId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
redirect("/%s/%s/issues".format(owner, name))
|
||||||
|
} getOrElse NotFound
|
||||||
|
})
|
||||||
|
|
||||||
|
post("/:owner/:repository/issues/batchedit/assign")(collaboratorsOnly { repository =>
|
||||||
|
params("checked").split(',') foreach { issueId =>
|
||||||
|
updateAssignedUserName(repository.owner, repository.name, issueId.toInt,
|
||||||
|
params.get("value") filter (_.trim != ""))
|
||||||
|
}
|
||||||
|
redirect("/%s/%s/issues".format(repository.owner, repository.name))
|
||||||
|
})
|
||||||
|
|
||||||
|
post("/:owner/:repository/issues/batchedit/milestone")(collaboratorsOnly { repository =>
|
||||||
|
params("checked").split(',') foreach { issueId =>
|
||||||
|
updateMilestoneId(repository.owner, repository.name, issueId.toInt,
|
||||||
|
params.get("value") collect { case x if x.trim != "" => x.toInt })
|
||||||
|
}
|
||||||
|
redirect("/%s/%s/issues".format(repository.owner, repository.name))
|
||||||
|
})
|
||||||
|
|
||||||
private def isEditable(owner: String, repository: String, author: String)(implicit context: app.Context): Boolean =
|
private def isEditable(owner: String, repository: String, author: String)(implicit context: app.Context): Boolean =
|
||||||
hasWritePermission(owner, repository, context.loginAccount) || author == context.loginAccount.get.userName
|
hasWritePermission(owner, repository, context.loginAccount) || author == context.loginAccount.get.userName
|
||||||
|
|
||||||
|
private def handleComment(issueId: Int, content: Option[String], repository: RepositoryService.RepositoryInfo) = {
|
||||||
|
val owner = repository.owner
|
||||||
|
val name = repository.name
|
||||||
|
val userName = context.loginAccount.get.userName
|
||||||
|
|
||||||
|
getIssue(owner, name, issueId.toString) map { issue =>
|
||||||
|
val (action, recordActivity) =
|
||||||
|
params.get("action")
|
||||||
|
.filter(_ => isEditable(owner, name, issue.openedUserName))
|
||||||
|
.collect {
|
||||||
|
case s if s == "close" => true -> (Some(s) -> Some(recordCloseIssueActivity _))
|
||||||
|
case s if s == "reopen" => false -> (Some(s) -> Some(recordReopenIssueActivity _))
|
||||||
|
}
|
||||||
|
.map { case (closed, t) =>
|
||||||
|
updateClosed(owner, name, issueId, closed)
|
||||||
|
t
|
||||||
|
}
|
||||||
|
.getOrElse(None -> None)
|
||||||
|
|
||||||
|
val commentId = createComment(owner, name, userName, issueId, content.getOrElse(action.get.capitalize), action)
|
||||||
|
|
||||||
|
// record activity
|
||||||
|
content foreach ( recordCommentIssueActivity(owner, name, userName, issueId, _) )
|
||||||
|
recordActivity foreach ( _ (owner, name, userName, issueId, issue.title) )
|
||||||
|
|
||||||
|
redirect("/%s/%s/issues/%d#comment-%d".format(owner, name, issueId, commentId))
|
||||||
|
} getOrElse NotFound
|
||||||
|
}
|
||||||
|
|
||||||
private def searchIssues(filter: String, repository: RepositoryService.RepositoryInfo) = {
|
private def searchIssues(filter: String, repository: RepositoryService.RepositoryInfo) = {
|
||||||
val owner = repository.owner
|
val owner = repository.owner
|
||||||
val repoName = repository.name
|
val repoName = repository.name
|
||||||
@@ -248,8 +310,9 @@ trait IssuesControllerBase extends ControllerBase {
|
|||||||
issues.html.list(
|
issues.html.list(
|
||||||
searchIssue(owner, repoName, condition, filter, userName, (page - 1) * IssueLimit, IssueLimit),
|
searchIssue(owner, repoName, condition, filter, userName, (page - 1) * IssueLimit, IssueLimit),
|
||||||
page,
|
page,
|
||||||
getLabels(owner, repoName),
|
(getCollaborators(owner, repoName) :+ owner).sorted,
|
||||||
getMilestones(owner, repoName).filter(_.closedDate.isEmpty),
|
getMilestones(owner, repoName).filter(_.closedDate.isEmpty),
|
||||||
|
getLabels(owner, repoName),
|
||||||
countIssue(owner, repoName, condition.copy(state = "open"), filter, userName),
|
countIssue(owner, repoName, condition.copy(state = "open"), filter, userName),
|
||||||
countIssue(owner, repoName, condition.copy(state = "closed"), filter, userName),
|
countIssue(owner, repoName, condition.copy(state = "closed"), filter, userName),
|
||||||
countIssue(owner, repoName, condition, "all", None),
|
countIssue(owner, repoName, condition, "all", None),
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ trait LabelsControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
post("/:owner/:repository/issues/label/new", newForm)(collaboratorsOnly { (form, repository) =>
|
post("/:owner/:repository/issues/label/new", newForm)(collaboratorsOnly { (form, repository) =>
|
||||||
createLabel(repository.owner, repository.name, form.labelName, form.color.substring(1))
|
createLabel(repository.owner, repository.name, form.labelName, form.color.substring(1))
|
||||||
redirect("/%s/%s/issues".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues")
|
||||||
})
|
})
|
||||||
|
|
||||||
ajaxGet("/:owner/:repository/issues/label/edit")(collaboratorsOnly { repository =>
|
ajaxGet("/:owner/:repository/issues/label/edit")(collaboratorsOnly { repository =>
|
||||||
@@ -53,9 +53,9 @@ trait LabelsControllerBase extends ControllerBase {
|
|||||||
private def labelName: Constraint = new Constraint(){
|
private def labelName: Constraint = new Constraint(){
|
||||||
def validate(name: String, value: String): Option[String] =
|
def validate(name: String, value: String): Option[String] =
|
||||||
if(!value.matches("^[^,]+$")){
|
if(!value.matches("^[^,]+$")){
|
||||||
Some("%s contains invalid character.".format(name))
|
Some(s"${name} contains invalid character.")
|
||||||
} else if(value.startsWith("_") || value.startsWith("-")){
|
} else if(value.startsWith("_") || value.startsWith("-")){
|
||||||
Some("%s starts with invalid character.".format(name))
|
Some(s"${name} starts with invalid character.")
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ trait MilestonesControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
post("/:owner/:repository/issues/milestones/new", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
post("/:owner/:repository/issues/milestones/new", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
||||||
createMilestone(repository.owner, repository.name, form.title, form.description, form.dueDate)
|
createMilestone(repository.owner, repository.name, form.title, form.description, form.dueDate)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/edit")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/edit")(collaboratorsOnly { repository =>
|
||||||
@@ -45,28 +45,28 @@ trait MilestonesControllerBase extends ControllerBase {
|
|||||||
post("/:owner/:repository/issues/milestones/:milestoneId/edit", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
post("/:owner/:repository/issues/milestones/:milestoneId/edit", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
updateMilestone(milestone.copy(title = form.title, description = form.description, dueDate = form.dueDate))
|
updateMilestone(milestone.copy(title = form.title, description = form.description, dueDate = form.dueDate))
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/close")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/close")(collaboratorsOnly { repository =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
closeMilestone(milestone)
|
closeMilestone(milestone)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/open")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/open")(collaboratorsOnly { repository =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
openMilestone(milestone)
|
openMilestone(milestone)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/delete")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/delete")(collaboratorsOnly { repository =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
deleteMilestone(repository.owner, repository.name, milestone.milestoneId)
|
deleteMilestone(repository.owner, repository.name, milestone.milestoneId)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
* Redirect to the Options page.
|
* Redirect to the Options page.
|
||||||
*/
|
*/
|
||||||
get("/:owner/:repository/settings")(ownerOnly { repository =>
|
get("/:owner/:repository/settings")(ownerOnly { repository =>
|
||||||
redirect("/%s/%s/settings/options".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/options")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -47,7 +47,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
post("/:owner/:repository/settings/options", optionsForm)(ownerOnly { (form, repository) =>
|
post("/:owner/:repository/settings/options", optionsForm)(ownerOnly { (form, repository) =>
|
||||||
saveRepositoryOptions(repository.owner, repository.name, form.description, form.defaultBranch, form.isPrivate)
|
saveRepositoryOptions(repository.owner, repository.name, form.description, form.defaultBranch, form.isPrivate)
|
||||||
flash += "info" -> "Repository settings has been updated."
|
flash += "info" -> "Repository settings has been updated."
|
||||||
redirect("/%s/%s/settings/options".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/options")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -70,7 +70,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
*/
|
*/
|
||||||
post("/:owner/:repository/settings/collaborators/add", collaboratorForm)(ownerOnly { (form, repository) =>
|
post("/:owner/:repository/settings/collaborators/add", collaboratorForm)(ownerOnly { (form, repository) =>
|
||||||
addCollaborator(repository.owner, repository.name, form.userName)
|
addCollaborator(repository.owner, repository.name, form.userName)
|
||||||
redirect("/%s/%s/settings/collaborators".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/collaborators")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -78,7 +78,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
*/
|
*/
|
||||||
get("/:owner/:repository/settings/collaborators/remove")(ownerOnly { repository =>
|
get("/:owner/:repository/settings/collaborators/remove")(ownerOnly { repository =>
|
||||||
removeCollaborator(repository.owner, repository.name, params("name"))
|
removeCollaborator(repository.owner, repository.name, params("name"))
|
||||||
redirect("/%s/%s/settings/collaborators".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/collaborators")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -98,7 +98,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
FileUtils.deleteDirectory(getWikiRepositoryDir(repository.owner, repository.name))
|
FileUtils.deleteDirectory(getWikiRepositoryDir(repository.owner, repository.name))
|
||||||
FileUtils.deleteDirectory(getTemporaryDir(repository.owner, repository.name))
|
FileUtils.deleteDirectory(getTemporaryDir(repository.owner, repository.name))
|
||||||
|
|
||||||
redirect("/%s".format(repository.owner))
|
redirect(s"/${repository.owner}")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -27,8 +27,7 @@ trait RepositoryViewerControllerBase extends ControllerBase {
|
|||||||
contentType = "text/html"
|
contentType = "text/html"
|
||||||
view.helpers.markdown(params("content"), repository,
|
view.helpers.markdown(params("content"), repository,
|
||||||
params("enableWikiLink").toBoolean,
|
params("enableWikiLink").toBoolean,
|
||||||
params("enableCommitLink").toBoolean,
|
params("enableRefsLink").toBoolean)
|
||||||
params("enableIssueLink").toBoolean)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -58,13 +57,14 @@ trait RepositoryViewerControllerBase extends ControllerBase {
|
|||||||
get("/:owner/:repository/commits/:branch")(referrersOnly { repository =>
|
get("/:owner/:repository/commits/:branch")(referrersOnly { repository =>
|
||||||
val branchName = params("branch")
|
val branchName = params("branch")
|
||||||
val page = params.getOrElse("page", "1").toInt
|
val page = params.getOrElse("page", "1").toInt
|
||||||
|
|
||||||
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
val (logs, hasNext) = JGitUtil.getCommitLog(git, branchName, page, 30)
|
JGitUtil.getCommitLog(git, branchName, page, 30) match {
|
||||||
|
case Right((logs, hasNext)) =>
|
||||||
repo.html.commits(Nil, branchName, repository, logs.splitWith{ (commit1, commit2) =>
|
repo.html.commits(Nil, branchName, repository, logs.splitWith{ (commit1, commit2) =>
|
||||||
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
||||||
}, page, hasNext)
|
}, page, hasNext)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -77,12 +77,14 @@ trait RepositoryViewerControllerBase extends ControllerBase {
|
|||||||
val page = params.getOrElse("page", "1").toInt
|
val page = params.getOrElse("page", "1").toInt
|
||||||
|
|
||||||
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
val (logs, hasNext) = JGitUtil.getCommitLog(git, branchName, page, 30, path)
|
JGitUtil.getCommitLog(git, branchName, page, 30, path) match {
|
||||||
|
case Right((logs, hasNext)) =>
|
||||||
repo.html.commits(path.split("/").toList, branchName, repository,
|
repo.html.commits(path.split("/").toList, branchName, repository,
|
||||||
logs.splitWith{ (commit1, commit2) =>
|
logs.splitWith{ (commit1, commit2) =>
|
||||||
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
||||||
}, page, hasNext)
|
}, page, hasNext)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -214,27 +216,23 @@ trait RepositoryViewerControllerBase extends ControllerBase {
|
|||||||
repo.html.guide(repository)
|
repo.html.guide(repository)
|
||||||
} else {
|
} else {
|
||||||
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
|
val revisions = Seq(if(revstr.isEmpty) repository.repository.defaultBranch else revstr, repository.branchList.head)
|
||||||
// get specified commit
|
// get specified commit
|
||||||
val (revCommit, revision) = try {
|
revisions.map { rev => (git.getRepository.resolve(rev), rev)}.find(_._1 != null).map { case (objectId, revision) =>
|
||||||
val revision = if(revstr.isEmpty) repository.repository.defaultBranch else revstr
|
val revCommit = JGitUtil.getRevCommitFromId(git, objectId)
|
||||||
(JGitUtil.getRevCommitFromId(git, git.getRepository.resolve(revision)), revision)
|
|
||||||
} catch {
|
|
||||||
case e: NullPointerException => {
|
|
||||||
val revision = repository.branchList.head
|
|
||||||
(JGitUtil.getRevCommitFromId(git, git.getRepository.resolve(revision)), revision)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// get files
|
|
||||||
val files = JGitUtil.getFileList(git, revision, path)
|
|
||||||
// process README.md
|
|
||||||
val readme = files.find(_.name == "README.md").map { file =>
|
|
||||||
new String(JGitUtil.getContent(Git.open(getRepositoryDir(repository.owner, repository.name)), file.id, true).get, "UTF-8")
|
|
||||||
}
|
|
||||||
|
|
||||||
repo.html.files(revision, repository,
|
// get files
|
||||||
if(path == ".") Nil else path.split("/").toList, // current path
|
val files = JGitUtil.getFileList(git, revision, path)
|
||||||
new JGitUtil.CommitInfo(revCommit), // latest commit
|
// process README.md
|
||||||
files, readme)
|
val readme = files.find(_.name == "README.md").map { file =>
|
||||||
|
new String(JGitUtil.getContent(Git.open(getRepositoryDir(repository.owner, repository.name)), file.id, true).get, "UTF-8")
|
||||||
|
}
|
||||||
|
|
||||||
|
repo.html.files(revision, repository,
|
||||||
|
if(path == ".") Nil else path.split("/").toList, // current path
|
||||||
|
new JGitUtil.CommitInfo(revCommit), // latest commit
|
||||||
|
files, readme)
|
||||||
|
} getOrElse NotFound
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
package app
|
package app
|
||||||
|
|
||||||
import service._
|
import service._
|
||||||
import util.{CollaboratorsAuthenticator, ReferrerAuthenticator, JGitUtil}
|
import util.{CollaboratorsAuthenticator, ReferrerAuthenticator, JGitUtil, StringUtil}
|
||||||
import util.Directory._
|
import util.Directory._
|
||||||
import jp.sf.amateras.scalatra.forms._
|
import jp.sf.amateras.scalatra.forms._
|
||||||
|
|
||||||
@@ -16,14 +16,14 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
case class WikiPageEditForm(pageName: String, content: String, message: Option[String], currentPageName: String)
|
case class WikiPageEditForm(pageName: String, content: String, message: Option[String], currentPageName: String)
|
||||||
|
|
||||||
val newForm = mapping(
|
val newForm = mapping(
|
||||||
"pageName" -> trim(label("Page name" , text(required, maxlength(40), identifier, unique))),
|
"pageName" -> trim(label("Page name" , text(required, maxlength(40), pagename, unique))),
|
||||||
"content" -> trim(label("Content" , text(required))),
|
"content" -> trim(label("Content" , text(required))),
|
||||||
"message" -> trim(label("Message" , optional(text()))),
|
"message" -> trim(label("Message" , optional(text()))),
|
||||||
"currentPageName" -> trim(label("Current page name" , text()))
|
"currentPageName" -> trim(label("Current page name" , text()))
|
||||||
)(WikiPageEditForm.apply)
|
)(WikiPageEditForm.apply)
|
||||||
|
|
||||||
val editForm = mapping(
|
val editForm = mapping(
|
||||||
"pageName" -> trim(label("Page name" , text(required, maxlength(40), identifier))),
|
"pageName" -> trim(label("Page name" , text(required, maxlength(40), pagename))),
|
||||||
"content" -> trim(label("Content" , text(required))),
|
"content" -> trim(label("Content" , text(required))),
|
||||||
"message" -> trim(label("Message" , optional(text()))),
|
"message" -> trim(label("Message" , optional(text()))),
|
||||||
"currentPageName" -> trim(label("Current page name" , text(required)))
|
"currentPageName" -> trim(label("Current page name" , text(required)))
|
||||||
@@ -32,27 +32,30 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
get("/:owner/:repository/wiki")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki")(referrersOnly { repository =>
|
||||||
getWikiPage(repository.owner, repository.name, "Home").map { page =>
|
getWikiPage(repository.owner, repository.name, "Home").map { page =>
|
||||||
wiki.html.page("Home", page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
wiki.html.page("Home", page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
||||||
} getOrElse redirect("/%s/%s/wiki/Home/_edit".format(repository.owner, repository.name))
|
} getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/Home/_edit")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/:page")(referrersOnly { repository =>
|
||||||
val pageName = params("page")
|
val pageName = StringUtil.urlDecode(params("page"))
|
||||||
|
|
||||||
getWikiPage(repository.owner, repository.name, pageName).map { page =>
|
getWikiPage(repository.owner, repository.name, pageName).map { page =>
|
||||||
wiki.html.page(pageName, page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
wiki.html.page(pageName, page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
||||||
} getOrElse redirect("/%s/%s/wiki/%s/_edit".format(repository.owner, repository.name, pageName)) // TODO URLEncode
|
} getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/${pageName}/_edit") // TODO URLEncode
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page/_history")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/:page/_history")(referrersOnly { repository =>
|
||||||
val pageName = params("page")
|
val pageName = StringUtil.urlDecode(params("page"))
|
||||||
|
|
||||||
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
wiki.html.history(Some(pageName), JGitUtil.getCommitLog(git, "master", path = pageName + ".md")._1, repository)
|
JGitUtil.getCommitLog(git, "master", path = pageName + ".md") match {
|
||||||
|
case Right((logs, hasNext)) => wiki.html.history(Some(pageName), logs, repository)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page/_compare/:commitId")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/:page/_compare/:commitId")(referrersOnly { repository =>
|
||||||
val pageName = params("page")
|
val pageName = StringUtil.urlDecode(params("page"))
|
||||||
val commitId = params("commitId").split("\\.\\.\\.")
|
val commitId = params("commitId").split("\\.\\.\\.")
|
||||||
|
|
||||||
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
@@ -69,7 +72,7 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page/_edit")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/wiki/:page/_edit")(collaboratorsOnly { repository =>
|
||||||
val pageName = params("page")
|
val pageName = StringUtil.urlDecode(params("page"))
|
||||||
wiki.html.edit(pageName, getWikiPage(repository.owner, repository.name, pageName), repository)
|
wiki.html.edit(pageName, getWikiPage(repository.owner, repository.name, pageName), repository)
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -82,7 +85,7 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
updateLastActivityDate(repository.owner, repository.name)
|
updateLastActivityDate(repository.owner, repository.name)
|
||||||
recordEditWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
recordEditWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
||||||
|
|
||||||
redirect("/%s/%s/wiki/%s".format(repository.owner, repository.name, form.pageName))
|
redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(form.pageName)}")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/_new")(collaboratorsOnly {
|
get("/:owner/:repository/wiki/_new")(collaboratorsOnly {
|
||||||
@@ -98,16 +101,16 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
updateLastActivityDate(repository.owner, repository.name)
|
updateLastActivityDate(repository.owner, repository.name)
|
||||||
recordCreateWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
recordCreateWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
||||||
|
|
||||||
redirect("/%s/%s/wiki/%s".format(repository.owner, repository.name, form.pageName))
|
redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(form.pageName)}")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page/_delete")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/wiki/:page/_delete")(collaboratorsOnly { repository =>
|
||||||
val pageName = params("page")
|
val pageName = StringUtil.urlDecode(params("page"))
|
||||||
|
|
||||||
deleteWikiPage(repository.owner, repository.name, pageName, context.loginAccount.get.userName, "Delete %s".format(pageName))
|
deleteWikiPage(repository.owner, repository.name, pageName, context.loginAccount.get.userName, s"Delete ${pageName}")
|
||||||
updateLastActivityDate(repository.owner, repository.name)
|
updateLastActivityDate(repository.owner, repository.name)
|
||||||
|
|
||||||
redirect("/%s/%s/wiki".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/wiki")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/_pages")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/_pages")(referrersOnly { repository =>
|
||||||
@@ -117,7 +120,10 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
get("/:owner/:repository/wiki/_history")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/_history")(referrersOnly { repository =>
|
||||||
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
wiki.html.history(None, JGitUtil.getCommitLog(git, "master")._1, repository)
|
JGitUtil.getCommitLog(git, "master") match {
|
||||||
|
case Right((logs, hasNext)) => wiki.html.history(None, logs, repository)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -133,4 +139,16 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
getWikiPageList(params("owner"), params("repository")).find(_ == value).map(_ => "Page already exists.")
|
getWikiPageList(params("owner"), params("repository")).find(_ == value).map(_ => "Page already exists.")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private def pagename: Constraint = new Constraint(){
|
||||||
|
def validate(name: String, value: String): Option[String] =
|
||||||
|
if(value.exists("\\/:*?\"<>|".contains(_))){
|
||||||
|
Some(s"${name} contains invalid character.")
|
||||||
|
} else if(value.startsWith("_") || value.startsWith("-")){
|
||||||
|
Some(s"${name} starts with invalid character.")
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -55,7 +55,7 @@ trait ActivityService {
|
|||||||
def recordReopenIssueActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, title: String): Unit =
|
def recordReopenIssueActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, title: String): Unit =
|
||||||
Activities.autoInc insert(userName, repositoryName, activityUserName,
|
Activities.autoInc insert(userName, repositoryName, activityUserName,
|
||||||
"reopen_issue",
|
"reopen_issue",
|
||||||
s"[user:${activityUserName}] closed reopened [issue:${userName}/${repositoryName}#${issueId}]",
|
s"[user:${activityUserName}] reopened issue [issue:${userName}/${repositoryName}#${issueId}]",
|
||||||
Some(title),
|
Some(title),
|
||||||
currentDate)
|
currentDate)
|
||||||
|
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import scala.slick.jdbc.{StaticQuery => Q}
|
|||||||
import Q.interpolation
|
import Q.interpolation
|
||||||
|
|
||||||
import model._
|
import model._
|
||||||
|
import util.StringUtil._
|
||||||
import util.Implicits._
|
import util.Implicits._
|
||||||
|
|
||||||
trait IssuesService {
|
trait IssuesService {
|
||||||
@@ -35,6 +36,9 @@ trait IssuesService {
|
|||||||
.map ( _._2 )
|
.map ( _._2 )
|
||||||
.list
|
.list
|
||||||
|
|
||||||
|
def getIssueLabel(owner: String, repository: String, issueId: Int, labelId: Int) =
|
||||||
|
Query(IssueLabels) filter (_.byPrimaryKey(owner, repository, issueId, labelId)) firstOption
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the count of the search result against issues.
|
* Returns the count of the search result against issues.
|
||||||
*
|
*
|
||||||
@@ -233,7 +237,6 @@ trait IssuesService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
object IssuesService {
|
object IssuesService {
|
||||||
import java.net.URLEncoder
|
|
||||||
import javax.servlet.http.HttpServletRequest
|
import javax.servlet.http.HttpServletRequest
|
||||||
|
|
||||||
val IssueLimit = 30
|
val IssueLimit = 30
|
||||||
@@ -245,8 +248,6 @@ object IssuesService {
|
|||||||
sort: String = "created",
|
sort: String = "created",
|
||||||
direction: String = "desc"){
|
direction: String = "desc"){
|
||||||
|
|
||||||
import IssueSearchCondition._
|
|
||||||
|
|
||||||
def toURL: String =
|
def toURL: String =
|
||||||
"?" + List(
|
"?" + List(
|
||||||
if(labels.isEmpty) None else Some("labels=" + urlEncode(labels.mkString(" "))),
|
if(labels.isEmpty) None else Some("labels=" + urlEncode(labels.mkString(" "))),
|
||||||
@@ -262,8 +263,6 @@ object IssuesService {
|
|||||||
|
|
||||||
object IssueSearchCondition {
|
object IssueSearchCondition {
|
||||||
|
|
||||||
private def urlEncode(value: String): String = URLEncoder.encode(value, "UTF-8")
|
|
||||||
|
|
||||||
private def param(request: HttpServletRequest, name: String, allow: Seq[String] = Nil): Option[String] = {
|
private def param(request: HttpServletRequest, name: String, allow: Seq[String] = Nil): Option[String] = {
|
||||||
val value = request.getParameter(name)
|
val value = request.getParameter(name)
|
||||||
if(value == null || value.isEmpty || (allow.nonEmpty && !allow.contains(value))) None else Some(value)
|
if(value == null || value.isEmpty || (allow.nonEmpty && !allow.contains(value))) None else Some(value)
|
||||||
|
|||||||
26
src/main/scala/service/RequestCache.scala
Normal file
26
src/main/scala/service/RequestCache.scala
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import model._
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This service is used for a view helper mainly.
|
||||||
|
*
|
||||||
|
* It may be called many times in one request, so each method stores
|
||||||
|
* its result into the cache which available during a request.
|
||||||
|
*/
|
||||||
|
trait RequestCache {
|
||||||
|
|
||||||
|
def getIssue(userName: String, repositoryName: String, issueId: String)(implicit context: app.Context): Option[Issue] = {
|
||||||
|
context.cache(s"issue.${userName}/${repositoryName}#${issueId}"){
|
||||||
|
new IssuesService {}.getIssue(userName, repositoryName, issueId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def getAccountByUserName(userName: String)(implicit context: app.Context): Option[Account] = {
|
||||||
|
context.cache(s"account.${userName}"){
|
||||||
|
new AccountService {}.getAccountByUserName(userName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
@@ -25,7 +25,7 @@ object AutoUpdate {
|
|||||||
* If corresponding SQL file does not exist, this method do nothing.
|
* If corresponding SQL file does not exist, this method do nothing.
|
||||||
*/
|
*/
|
||||||
def update(conn: Connection): Unit = {
|
def update(conn: Connection): Unit = {
|
||||||
val sqlPath = "update/%d_%d.sql".format(majorVersion, minorVersion)
|
val sqlPath = s"update/${majorVersion}_${minorVersion}.sql"
|
||||||
val in = Thread.currentThread.getContextClassLoader.getResourceAsStream(sqlPath)
|
val in = Thread.currentThread.getContextClassLoader.getResourceAsStream(sqlPath)
|
||||||
if(in != null){
|
if(in != null){
|
||||||
val sql = IOUtils.toString(in, "UTF-8")
|
val sql = IOUtils.toString(in, "UTF-8")
|
||||||
@@ -42,14 +42,30 @@ object AutoUpdate {
|
|||||||
/**
|
/**
|
||||||
* MAJOR.MINOR
|
* MAJOR.MINOR
|
||||||
*/
|
*/
|
||||||
val versionString = "%d.%d".format(majorVersion, minorVersion)
|
val versionString = s"${majorVersion}.${minorVersion}"
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The history of versions. A head of this sequence is the current BitBucket version.
|
* The history of versions. A head of this sequence is the current BitBucket version.
|
||||||
*/
|
*/
|
||||||
val versions = Seq(
|
val versions = Seq(
|
||||||
Version(1, 3),
|
new Version(1, 3){
|
||||||
|
override def update(conn: Connection): Unit = {
|
||||||
|
super.update(conn)
|
||||||
|
// Fix wiki repository configuration
|
||||||
|
val rs = conn.createStatement.executeQuery("SELECT USER_NAME, REPOSITORY_NAME FROM REPOSITORY")
|
||||||
|
while(rs.next){
|
||||||
|
val wikidir = Directory.getWikiRepositoryDir(rs.getString("USER_NAME"), rs.getString("REPOSITORY_NAME"))
|
||||||
|
val repository = org.eclipse.jgit.api.Git.open(wikidir).getRepository
|
||||||
|
val config = repository.getConfig
|
||||||
|
if(!config.getBoolean("http", "receivepack", false)){
|
||||||
|
config.setBoolean("http", null, "receivepack", true)
|
||||||
|
config.save
|
||||||
|
}
|
||||||
|
repository.close
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
Version(1, 2),
|
Version(1, 2),
|
||||||
Version(1, 1),
|
Version(1, 1),
|
||||||
Version(1, 0)
|
Version(1, 0)
|
||||||
|
|||||||
@@ -21,7 +21,6 @@ class TransactionFilter extends Filter {
|
|||||||
// assets don't need transaction
|
// assets don't need transaction
|
||||||
chain.doFilter(req, res)
|
chain.doFilter(req, res)
|
||||||
} else {
|
} else {
|
||||||
// TODO begin transaction!
|
|
||||||
val context = req.getServletContext
|
val context = req.getServletContext
|
||||||
Database.forURL(context.getInitParameter("db.url"),
|
Database.forURL(context.getInitParameter("db.url"),
|
||||||
context.getInitParameter("db.user"),
|
context.getInitParameter("db.user"),
|
||||||
|
|||||||
@@ -13,13 +13,13 @@ object Directory {
|
|||||||
|
|
||||||
val GitBucketConf = new File(GitBucketHome, "gitbucket.conf")
|
val GitBucketConf = new File(GitBucketHome, "gitbucket.conf")
|
||||||
|
|
||||||
val RepositoryHome = "%s/repositories".format(GitBucketHome)
|
val RepositoryHome = s"${GitBucketHome}/repositories"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Repository names of the specified user.
|
* Repository names of the specified user.
|
||||||
*/
|
*/
|
||||||
def getRepositories(owner: String): List[String] = {
|
def getRepositories(owner: String): List[String] = {
|
||||||
val dir = new File("%s/%s".format(RepositoryHome, owner))
|
val dir = new File(s"${RepositoryHome}/${owner}")
|
||||||
if(dir.exists){
|
if(dir.exists){
|
||||||
dir.listFiles.filter { file =>
|
dir.listFiles.filter { file =>
|
||||||
file.isDirectory && !file.getName.endsWith(".wiki.git")
|
file.isDirectory && !file.getName.endsWith(".wiki.git")
|
||||||
@@ -33,24 +33,24 @@ object Directory {
|
|||||||
* Substance directory of the repository.
|
* Substance directory of the repository.
|
||||||
*/
|
*/
|
||||||
def getRepositoryDir(owner: String, repository: String): File =
|
def getRepositoryDir(owner: String, repository: String): File =
|
||||||
new File("%s/%s/%s.git".format(RepositoryHome, owner, repository))
|
new File(s"${RepositoryHome}/${owner}/${repository}.git")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Directory for uploaded files by the specified user.
|
* Directory for uploaded files by the specified user.
|
||||||
*/
|
*/
|
||||||
def getUserUploadDir(userName: String): File = new File("%s/data/%s/files".format(GitBucketHome, userName))
|
def getUserUploadDir(userName: String): File = new File(s"${GitBucketHome}/data/${userName}/files")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Root of temporary directories for the specified repository.
|
* Root of temporary directories for the specified repository.
|
||||||
*/
|
*/
|
||||||
def getTemporaryDir(owner: String, repository: String): File =
|
def getTemporaryDir(owner: String, repository: String): File =
|
||||||
new File("%s/tmp/%s/%s".format(GitBucketHome, owner, repository))
|
new File(s"${GitBucketHome}/tmp/${owner}/${repository}")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Temporary directory which is used to create an archive to download repository contents.
|
* Temporary directory which is used to create an archive to download repository contents.
|
||||||
*/
|
*/
|
||||||
def getDownloadWorkDir(owner: String, repository: String, sessionId: String): File =
|
def getDownloadWorkDir(owner: String, repository: String, sessionId: String): File =
|
||||||
new File(getTemporaryDir(owner, repository), "download/%s".format(sessionId))
|
new File(getTemporaryDir(owner, repository), s"download/${sessionId}")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Temporary directory which is used in the repository creation.
|
* Temporary directory which is used in the repository creation.
|
||||||
@@ -65,7 +65,7 @@ object Directory {
|
|||||||
* Substance directory of the wiki repository.
|
* Substance directory of the wiki repository.
|
||||||
*/
|
*/
|
||||||
def getWikiRepositoryDir(owner: String, repository: String): File =
|
def getWikiRepositoryDir(owner: String, repository: String): File =
|
||||||
new File("%s/%s/%s.wiki.git".format(Directory.RepositoryHome, owner, repository))
|
new File(s"${RepositoryHome}/${owner}/${repository}.wiki.git")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Wiki working directory which is cloned from the wiki repository.
|
* Wiki working directory which is cloned from the wiki repository.
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ object FileUploadUtil {
|
|||||||
new SimpleDateFormat("yyyyMMddHHmmSSsss").format(new java.util.Date(System.currentTimeMillis))
|
new SimpleDateFormat("yyyyMMddHHmmSSsss").format(new java.util.Date(System.currentTimeMillis))
|
||||||
|
|
||||||
def TemporaryDir(implicit session: HttpSession): java.io.File =
|
def TemporaryDir(implicit session: HttpSession): java.io.File =
|
||||||
new java.io.File(GitBucketHome, "tmp/_upload/%s".format(session.getId))
|
new java.io.File(GitBucketHome, s"tmp/_upload/${session.getId}")
|
||||||
|
|
||||||
def getTemporaryFile(fileId: String)(implicit session: HttpSession): java.io.File =
|
def getTemporaryFile(fileId: String)(implicit session: HttpSession): java.io.File =
|
||||||
new java.io.File(TemporaryDir, fileId)
|
new java.io.File(TemporaryDir, fileId)
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
package util
|
package util
|
||||||
|
|
||||||
import twirl.api.Html
|
|
||||||
import scala.slick.driver.H2Driver.simple._
|
import scala.slick.driver.H2Driver.simple._
|
||||||
|
import scala.util.matching.Regex
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provides some usable implicit conversions.
|
* Provides some usable implicit conversions.
|
||||||
@@ -30,4 +30,23 @@ object Implicits {
|
|||||||
def &&(c2: => Column[Boolean], guard: => Boolean): Column[Boolean] = if(guard) c1 && c2 else c1
|
def &&(c2: => Column[Boolean], guard: => Boolean): Column[Boolean] = if(guard) c1 && c2 else c1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
implicit class RichString(value: String){
|
||||||
|
def replaceBy(regex: Regex)(replace: Regex.MatchData => Option[String]): String = {
|
||||||
|
val sb = new StringBuilder()
|
||||||
|
var i = 0
|
||||||
|
regex.findAllIn(value).matchData.foreach { m =>
|
||||||
|
sb.append(value.substring(i, m.start))
|
||||||
|
i = m.end
|
||||||
|
replace(m) match {
|
||||||
|
case Some(s) => sb.append(s)
|
||||||
|
case None => sb.append(m.matched)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(i < value.length){
|
||||||
|
sb.append(value.substring(i))
|
||||||
|
}
|
||||||
|
sb.toString
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -153,7 +153,7 @@ object JGitUtil {
|
|||||||
}
|
}
|
||||||
|
|
||||||
RepositoryInfo(
|
RepositoryInfo(
|
||||||
owner, repository, baseUrl + "/git/%s/%s.git".format(owner, repository),
|
owner, repository, s"${baseUrl}/git/${owner}/${repository}.git",
|
||||||
// commit count
|
// commit count
|
||||||
commitCount,
|
commitCount,
|
||||||
// branches
|
// branches
|
||||||
@@ -169,7 +169,7 @@ object JGitUtil {
|
|||||||
} catch {
|
} catch {
|
||||||
// not initialized
|
// not initialized
|
||||||
case e: NoHeadException => RepositoryInfo(
|
case e: NoHeadException => RepositoryInfo(
|
||||||
owner, repository, baseUrl + "/git/%s/%s.git".format(owner, repository), 0, Nil, Nil)
|
owner, repository, s"${baseUrl}/git/${owner}/${repository}.git", 0, Nil, Nil)
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -253,7 +253,7 @@ object JGitUtil {
|
|||||||
* @param path filters by this path. default is no filter.
|
* @param path filters by this path. default is no filter.
|
||||||
* @return a tuple of the commit list and whether has next
|
* @return a tuple of the commit list and whether has next
|
||||||
*/
|
*/
|
||||||
def getCommitLog(git: Git, revision: String, page: Int = 1, limit: Int = 0, path: String = ""): (List[CommitInfo], Boolean) = {
|
def getCommitLog(git: Git, revision: String, page: Int = 1, limit: Int = 0, path: String = ""): Either[String, (List[CommitInfo], Boolean)] = {
|
||||||
val fixedPage = if(page <= 0) 1 else page
|
val fixedPage = if(page <= 0) 1 else page
|
||||||
|
|
||||||
@scala.annotation.tailrec
|
@scala.annotation.tailrec
|
||||||
@@ -267,20 +267,25 @@ object JGitUtil {
|
|||||||
}
|
}
|
||||||
|
|
||||||
val revWalk = new RevWalk(git.getRepository)
|
val revWalk = new RevWalk(git.getRepository)
|
||||||
revWalk.markStart(revWalk.parseCommit(git.getRepository.resolve(revision)))
|
val objectId = git.getRepository.resolve(revision)
|
||||||
if(path.nonEmpty){
|
if(objectId == null){
|
||||||
revWalk.setRevFilter(new RevFilter(){
|
Left(s"${revision} can't be resolved.")
|
||||||
def include(walk: RevWalk, commit: RevCommit): Boolean = {
|
} else {
|
||||||
getDiffs(git, commit.getName, false).find(_.newPath == path).nonEmpty
|
revWalk.markStart(revWalk.parseCommit(objectId))
|
||||||
}
|
if(path.nonEmpty){
|
||||||
override def clone(): RevFilter = this
|
revWalk.setRevFilter(new RevFilter(){
|
||||||
})
|
def include(walk: RevWalk, commit: RevCommit): Boolean = {
|
||||||
|
getDiffs(git, commit.getName, false).find(_.newPath == path).nonEmpty
|
||||||
|
}
|
||||||
|
override def clone(): RevFilter = this
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
val commits = getCommitLog(revWalk.iterator, 0, Nil)
|
||||||
|
revWalk.release
|
||||||
|
|
||||||
|
Right(commits)
|
||||||
}
|
}
|
||||||
|
|
||||||
val commits = getCommitLog(revWalk.iterator, 0, Nil)
|
|
||||||
revWalk.release
|
|
||||||
|
|
||||||
commits
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package util
|
package util
|
||||||
|
|
||||||
|
import java.net.{URLDecoder, URLEncoder}
|
||||||
|
|
||||||
object StringUtil {
|
object StringUtil {
|
||||||
|
|
||||||
def sha1(value: String): String = {
|
def sha1(value: String): String = {
|
||||||
@@ -14,4 +16,8 @@ object StringUtil {
|
|||||||
md.digest.map(b => "%02x".format(b)).mkString
|
md.digest.map(b => "%02x".format(b)).mkString
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def urlEncode(value: String): String = URLEncoder.encode(value, "UTF-8")
|
||||||
|
|
||||||
|
def urlDecode(value: String): String = URLDecoder.decode(value, "UTF-8")
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,9 +11,9 @@ trait Validations {
|
|||||||
def identifier: Constraint = new Constraint(){
|
def identifier: Constraint = new Constraint(){
|
||||||
def validate(name: String, value: String): Option[String] =
|
def validate(name: String, value: String): Option[String] =
|
||||||
if(!value.matches("^[a-zA-Z0-9\\-_]+$")){
|
if(!value.matches("^[a-zA-Z0-9\\-_]+$")){
|
||||||
Some("%s contains invalid character.".format(name))
|
Some(s"${name} contains invalid character.")
|
||||||
} else if(value.startsWith("_") || value.startsWith("-")){
|
} else if(value.startsWith("_") || value.startsWith("-")){
|
||||||
Some("%s starts with invalid character.".format(name))
|
Some(s"${name} starts with invalid character.")
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|||||||
26
src/main/scala/view/AvatarImageProvider.scala
Normal file
26
src/main/scala/view/AvatarImageProvider.scala
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
package view
|
||||||
|
|
||||||
|
import service.RequestCache
|
||||||
|
import twirl.api.Html
|
||||||
|
import util.StringUtil
|
||||||
|
|
||||||
|
trait AvatarImageProvider { self: RequestCache =>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns <img> which displays the avatar icon.
|
||||||
|
* Looks up Gravatar if avatar icon has not been configured in user settings.
|
||||||
|
*/
|
||||||
|
protected def getAvatarImageHtml(userName: String, size: Int, tooltip: Boolean = false)(implicit context: app.Context): Html = {
|
||||||
|
val src = getAccountByUserName(userName).collect { case account if(account.image.isEmpty) =>
|
||||||
|
s"""http://www.gravatar.com/avatar/${StringUtil.md5(account.mailAddress)}?s=${size}"""
|
||||||
|
} getOrElse {
|
||||||
|
s"""${context.path}/${userName}/_avatar"""
|
||||||
|
}
|
||||||
|
if(tooltip){
|
||||||
|
Html(s"""<img src=${src} class="avatar" style="width: ${size}px; height: ${size}px;" data-toggle="tooltip" title=${userName}/>""")
|
||||||
|
} else {
|
||||||
|
Html(s"""<img src=${src} class="avatar" style="width: ${size}px; height: ${size}px;" />""")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
33
src/main/scala/view/LinkConverter.scala
Normal file
33
src/main/scala/view/LinkConverter.scala
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
package view
|
||||||
|
|
||||||
|
import service.RequestCache
|
||||||
|
import util.Implicits.RichString
|
||||||
|
|
||||||
|
trait LinkConverter { self: RequestCache =>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts issue id, username and commit id to link.
|
||||||
|
*/
|
||||||
|
protected def convertRefsLinks(value: String, repository: service.RepositoryService.RepositoryInfo,
|
||||||
|
issueIdPrefix: String = "#")(implicit context: app.Context): String = {
|
||||||
|
value
|
||||||
|
// escape HTML tags
|
||||||
|
.replaceAll("&", "&").replaceAll("<", "<").replaceAll(">", ">").replaceAll("\"", """)
|
||||||
|
// convert issue id to link
|
||||||
|
.replaceBy(("(^|\\W)" + issueIdPrefix + "(\\d+)(\\W|$)").r){ m =>
|
||||||
|
if(getIssue(repository.owner, repository.name, m.group(2)).isDefined){
|
||||||
|
Some(s"""${m.group(1)}<a href="${context.path}/${repository.owner}/${repository.name}/issues/${m.group(2)}">#${m.group(2)}</a>${m.group(3)}""")
|
||||||
|
} else {
|
||||||
|
Some(s"""${m.group(1)}#${m.group(2)}${m.group(3)}""")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// convert @username to link
|
||||||
|
.replaceBy("(^|\\W)@([a-zA-Z0-9\\-_]+)(\\W|$)".r){ m =>
|
||||||
|
getAccountByUserName(m.group(2)).map { _ =>
|
||||||
|
s"""${m.group(1)}<a href="${context.path}/${m.group(2)}">@${m.group(2)}</a>${m.group(3)}"""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// convert commit id to link
|
||||||
|
.replaceAll("(^|\\W)([a-f0-9]{40})(\\W|$)", s"""$$1<a href="${context.path}/${repository.owner}/${repository.name}/commit/$$2">$$2</a>$$3""")
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
package view
|
package view
|
||||||
|
|
||||||
|
import util.StringUtil
|
||||||
import org.parboiled.common.StringUtils
|
import org.parboiled.common.StringUtils
|
||||||
import org.pegdown._
|
import org.pegdown._
|
||||||
import org.pegdown.ast._
|
import org.pegdown.ast._
|
||||||
import org.pegdown.LinkRenderer.Rendering
|
import org.pegdown.LinkRenderer.Rendering
|
||||||
import scala.collection.JavaConverters._
|
import scala.collection.JavaConverters._
|
||||||
|
import service.RequestCache
|
||||||
|
|
||||||
object Markdown {
|
object Markdown {
|
||||||
|
|
||||||
@@ -12,12 +14,17 @@ object Markdown {
|
|||||||
* Converts Markdown of Wiki pages to HTML.
|
* Converts Markdown of Wiki pages to HTML.
|
||||||
*/
|
*/
|
||||||
def toHtml(markdown: String, repository: service.RepositoryService.RepositoryInfo,
|
def toHtml(markdown: String, repository: service.RepositoryService.RepositoryInfo,
|
||||||
enableWikiLink: Boolean, enableCommitLink: Boolean, enableIssueLink: Boolean)(implicit context: app.Context): String = {
|
enableWikiLink: Boolean, enableRefsLink: Boolean)(implicit context: app.Context): String = {
|
||||||
|
// escape issue id
|
||||||
|
val source = if(enableRefsLink){
|
||||||
|
markdown.replaceAll("(^|\\W)#([0-9]+)(\\W|$)", "$1issue:$2$3")
|
||||||
|
} else markdown
|
||||||
|
|
||||||
val rootNode = new PegDownProcessor(
|
val rootNode = new PegDownProcessor(
|
||||||
Extensions.AUTOLINKS | Extensions.WIKILINKS | Extensions.FENCED_CODE_BLOCKS | Extensions.TABLES
|
Extensions.AUTOLINKS | Extensions.WIKILINKS | Extensions.FENCED_CODE_BLOCKS | Extensions.TABLES
|
||||||
).parseMarkdown(markdown.toCharArray)
|
).parseMarkdown(source.toCharArray)
|
||||||
|
|
||||||
new GitBucketHtmlSerializer(markdown, context, repository, enableWikiLink, enableCommitLink, enableIssueLink).toHtml(rootNode)
|
new GitBucketHtmlSerializer(markdown, repository, enableWikiLink, enableRefsLink).toHtml(rootNode)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -33,11 +40,10 @@ class GitBucketLinkRender(context: app.Context, repository: service.RepositorySe
|
|||||||
} else {
|
} else {
|
||||||
(text, text)
|
(text, text)
|
||||||
}
|
}
|
||||||
val url = repository.url.replaceFirst("/git/", "/").replaceFirst("\\.git$", "") +
|
val url = repository.url.replaceFirst("/git/", "/").replaceFirst("\\.git$", "") + "/wiki/" + StringUtil.urlEncode(page)
|
||||||
"/wiki/" + java.net.URLEncoder.encode(page.replace(' ', '-'), "UTF-8")
|
|
||||||
new Rendering(url, label)
|
new Rendering(url, label)
|
||||||
} catch {
|
} catch {
|
||||||
case e: java.io.UnsupportedEncodingException => throw new IllegalStateException();
|
case e: java.io.UnsupportedEncodingException => throw new IllegalStateException
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
super.render(node)
|
super.render(node)
|
||||||
@@ -64,15 +70,13 @@ class GitBucketVerbatimSerializer extends VerbatimSerializer {
|
|||||||
|
|
||||||
class GitBucketHtmlSerializer(
|
class GitBucketHtmlSerializer(
|
||||||
markdown: String,
|
markdown: String,
|
||||||
context: app.Context,
|
|
||||||
repository: service.RepositoryService.RepositoryInfo,
|
repository: service.RepositoryService.RepositoryInfo,
|
||||||
enableWikiLink: Boolean,
|
enableWikiLink: Boolean,
|
||||||
enableCommitLink: Boolean,
|
enableRefsLink: Boolean
|
||||||
enableIssueLink: Boolean
|
)(implicit val context: app.Context) extends ToHtmlSerializer(
|
||||||
) extends ToHtmlSerializer(
|
|
||||||
new GitBucketLinkRender(context, repository, enableWikiLink),
|
new GitBucketLinkRender(context, repository, enableWikiLink),
|
||||||
Map[String, VerbatimSerializer](VerbatimSerializer.DEFAULT -> new GitBucketVerbatimSerializer).asJava
|
Map[String, VerbatimSerializer](VerbatimSerializer.DEFAULT -> new GitBucketVerbatimSerializer).asJava
|
||||||
) {
|
) with LinkConverter with RequestCache {
|
||||||
|
|
||||||
override protected def printImageTag(imageNode: SuperNode, url: String): Unit =
|
override protected def printImageTag(imageNode: SuperNode, url: String): Unit =
|
||||||
printer.print("<img src=\"").print(fixUrl(url)).print("\" alt=\"").printEncoded(printChildrenToString(imageNode)).print("\"/>")
|
printer.print("<img src=\"").print(fixUrl(url)).print("\" alt=\"").printEncoded(printChildrenToString(imageNode)).print("\"/>")
|
||||||
@@ -99,10 +103,8 @@ class GitBucketHtmlSerializer(
|
|||||||
}
|
}
|
||||||
|
|
||||||
override def visit(node: TextNode) {
|
override def visit(node: TextNode) {
|
||||||
// convert commit id to link.
|
// convert commit id and username to link.
|
||||||
val text = if(enableCommitLink) node.getText.replaceAll("(^|\\W)([0-9a-f]{40})(\\W|$)",
|
val text = if(enableRefsLink) convertRefsLinks(node.getText, repository, "issue:") else node.getText
|
||||||
"<a href=\"%s/%s/%s/commit/$2\">$2</a>".format(context.path, repository.owner, repository.name))
|
|
||||||
else node.getText
|
|
||||||
|
|
||||||
if (abbreviations.isEmpty) {
|
if (abbreviations.isEmpty) {
|
||||||
printer.print(text)
|
printer.print(text)
|
||||||
@@ -111,15 +113,4 @@ class GitBucketHtmlSerializer(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override def visit(node: HeaderNode) {
|
|
||||||
val text = markdown.substring(node.getStartIndex, node.getEndIndex - 1).trim
|
|
||||||
if(enableIssueLink && text.matches("#[\\d]+")){
|
|
||||||
// convert issue id to link
|
|
||||||
val issueId = text.substring(1).toInt
|
|
||||||
printer.print("<a href=\"%s/%s/%s/issues/%d\">#%d</a>".format(context.path, repository.owner, repository.name, issueId, issueId))
|
|
||||||
} else {
|
|
||||||
printTag(node, "h" + node.getLevel)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -3,12 +3,12 @@ import java.util.Date
|
|||||||
import java.text.SimpleDateFormat
|
import java.text.SimpleDateFormat
|
||||||
import twirl.api.Html
|
import twirl.api.Html
|
||||||
import util.StringUtil
|
import util.StringUtil
|
||||||
import service.AccountService
|
import service.RequestCache
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provides helper methods for Twirl templates.
|
* Provides helper methods for Twirl templates.
|
||||||
*/
|
*/
|
||||||
object helpers {
|
object helpers extends AvatarImageProvider with LinkConverter with RequestCache {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Format java.util.Date to "yyyy-MM-dd HH:mm:ss".
|
* Format java.util.Date to "yyyy-MM-dd HH:mm:ss".
|
||||||
@@ -31,13 +31,24 @@ object helpers {
|
|||||||
* Converts Markdown of Wiki pages to HTML.
|
* Converts Markdown of Wiki pages to HTML.
|
||||||
*/
|
*/
|
||||||
def markdown(value: String, repository: service.RepositoryService.RepositoryInfo,
|
def markdown(value: String, repository: service.RepositoryService.RepositoryInfo,
|
||||||
enableWikiLink: Boolean, enableCommitLink: Boolean, enableIssueLink: Boolean)(implicit context: app.Context): Html = {
|
enableWikiLink: Boolean, enableRefsLink: Boolean)(implicit context: app.Context): Html = {
|
||||||
Html(Markdown.toHtml(value, repository, enableWikiLink, enableCommitLink, enableIssueLink))
|
Html(Markdown.toHtml(value, repository, enableWikiLink, enableRefsLink))
|
||||||
}
|
}
|
||||||
|
|
||||||
def activityMessage(message: String)(implicit context: app.Context): Html = {
|
/**
|
||||||
val a = s"a $message aa $$1 a"
|
* Returns <img> which displays the avatar icon.
|
||||||
|
* Looks up Gravatar if avatar icon has not been configured in user settings.
|
||||||
|
*/
|
||||||
|
def avatar(userName: String, size: Int, tooltip: Boolean = false)(implicit context: app.Context): Html =
|
||||||
|
getAvatarImageHtml(userName, size, tooltip)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts commit id, issue id and username to the link.
|
||||||
|
*/
|
||||||
|
def link(value: String, repository: service.RepositoryService.RepositoryInfo)(implicit context: app.Context): Html =
|
||||||
|
Html(convertRefsLinks(value, repository))
|
||||||
|
|
||||||
|
def activityMessage(message: String)(implicit context: app.Context): Html =
|
||||||
Html(message
|
Html(message
|
||||||
.replaceAll("\\[issue:([^\\s]+?)/([^\\s]+?)#((\\d+))\\]" , s"""<a href="${context.path}/$$1/$$2/issues/$$3">$$1/$$2#$$3</a>""")
|
.replaceAll("\\[issue:([^\\s]+?)/([^\\s]+?)#((\\d+))\\]" , s"""<a href="${context.path}/$$1/$$2/issues/$$3">$$1/$$2#$$3</a>""")
|
||||||
.replaceAll("\\[repo:([^\\s]+?)/([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2\">$$1/$$2</a>""")
|
.replaceAll("\\[repo:([^\\s]+?)/([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2\">$$1/$$2</a>""")
|
||||||
@@ -45,56 +56,29 @@ object helpers {
|
|||||||
.replaceAll("\\[tag:([^\\s]+?)/([^\\s]+?)#([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2/tree/$$3">$$3</a>""")
|
.replaceAll("\\[tag:([^\\s]+?)/([^\\s]+?)#([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2/tree/$$3">$$3</a>""")
|
||||||
.replaceAll("\\[user:([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1">$$1</a>""")
|
.replaceAll("\\[user:([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1">$$1</a>""")
|
||||||
)
|
)
|
||||||
}
|
|
||||||
|
def urlEncode(value: String): String = StringUtil.urlEncode(value)
|
||||||
|
|
||||||
|
def urlEncode(value: Option[String]): String = value.map(urlEncode).getOrElse("")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generates the url to the repository.
|
* Generates the url to the repository.
|
||||||
*/
|
*/
|
||||||
def url(repository: service.RepositoryService.RepositoryInfo)(implicit context: app.Context): String =
|
def url(repository: service.RepositoryService.RepositoryInfo)(implicit context: app.Context): String =
|
||||||
"%s/%s/%s".format(context.path, repository.owner, repository.name)
|
s"${context.path}/${repository.owner}/${repository.name}"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generates the url to the account page.
|
* Generates the url to the account page.
|
||||||
*/
|
*/
|
||||||
def url(userName: String)(implicit context: app.Context): String = "%s/%s".format(context.path, userName)
|
def url(userName: String)(implicit context: app.Context): String =
|
||||||
|
s"${context.path}/${userName}"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the url to the root of assets.
|
* Returns the url to the root of assets.
|
||||||
*/
|
*/
|
||||||
def assets(implicit context: app.Context): String = "%s/assets".format(context.path)
|
def assets(implicit context: app.Context): String =
|
||||||
|
s"${context.path}/assets"
|
||||||
|
|
||||||
/**
|
|
||||||
* Converts issue id and commit id to link.
|
|
||||||
*/
|
|
||||||
def link(value: String, repository: service.RepositoryService.RepositoryInfo)(implicit context: app.Context): Html =
|
|
||||||
Html(value
|
|
||||||
// escape HTML tags
|
|
||||||
.replaceAll("&", "&").replaceAll("<", "<").replaceAll(">", ">").replaceAll("\"", """)
|
|
||||||
// convert issue id to link
|
|
||||||
.replaceAll("(^|\\W)#(\\d+)(\\W|$)", "$1<a href=\"%s/%s/%s/issues/$2\">#$2</a>$3".format(context.path, repository.owner, repository.name))
|
|
||||||
// convert commit id to link
|
|
||||||
.replaceAll("(^|\\W)([a-f0-9]{40})(\\W|$)", "$1<a href=\"%s/%s/%s/commit/$2\">$2</a>$3").format(context.path, repository.owner, repository.name))
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns <img> which displays the avatar icon.
|
|
||||||
* Looks up Gravatar if avatar icon has not been configured in user settings.
|
|
||||||
*/
|
|
||||||
def avatar(userName: String, size: Int, tooltip: Boolean = false)(implicit context: app.Context): Html = {
|
|
||||||
val account = context.cache(s"account.${userName}"){
|
|
||||||
new AccountService {}.getAccountByUserName(userName)
|
|
||||||
}
|
|
||||||
val src = account.collect { case account if(account.image.isEmpty) =>
|
|
||||||
s"""http://www.gravatar.com/avatar/${StringUtil.md5(account.mailAddress)}?s=${size}"""
|
|
||||||
} getOrElse {
|
|
||||||
s"""${context.path}/${userName}/_avatar"""
|
|
||||||
}
|
|
||||||
if(tooltip){
|
|
||||||
Html(s"""<img src=${src} class="avatar" style="width: ${size}px; height: ${size}px;" data-toggle="tooltip" title=${userName}/>""")
|
|
||||||
} else {
|
|
||||||
Html(s"""<img src=${src} class="avatar" style="width: ${size}px; height: ${size}px;" />""")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Implicit conversion to add mkHtml() to Seq[Html].
|
* Implicit conversion to add mkHtml() to Seq[Html].
|
||||||
|
|||||||
@@ -14,10 +14,10 @@
|
|||||||
@activity.additionalInfo.map { additionalInfo =>
|
@activity.additionalInfo.map { additionalInfo =>
|
||||||
@(activity.activityType match {
|
@(activity.activityType match {
|
||||||
case "create_wiki" => {
|
case "create_wiki" => {
|
||||||
<div class="small activity-message">Created <a href={"%s/%s/%s/wiki/%s".format(path, activity.userName, activity.repositoryName, additionalInfo)}>{additionalInfo}</a>.</div>
|
<div class="small activity-message">Created <a href={s"${path}/${activity.userName}/${activity.repositoryName}/wiki/${additionalInfo}"}>{additionalInfo}</a>.</div>
|
||||||
}
|
}
|
||||||
case "edit_wiki" => {
|
case "edit_wiki" => {
|
||||||
<div class="small activity-message">Edited <a href={"%s/%s/%s/wiki/%s".format(path, activity.userName, activity.repositoryName, additionalInfo)}>{additionalInfo}</a>.</div>
|
<div class="small activity-message">Edited <a href={s"${path}/${activity.userName}/${activity.repositoryName}/wiki/${additionalInfo}"}>{additionalInfo}</a>.</div>
|
||||||
}
|
}
|
||||||
case "push" => {
|
case "push" => {
|
||||||
<div class="small activity-message">
|
<div class="small activity-message">
|
||||||
@@ -26,7 +26,7 @@
|
|||||||
<div>...</div>
|
<div>...</div>
|
||||||
} else {
|
} else {
|
||||||
<div>
|
<div>
|
||||||
<a href={"%s/%s/%s/commit/%s".format(path, activity.userName, activity.repositoryName, commit.substring(0, 40))} class="monospace">{commit.substring(0, 7)}</a>
|
<a href={s"${path}/${activity.userName}/${activity.repositoryName}/commit/${commit.substring(0, 40)}"} class="monospace">{commit.substring(0, 7)}</a>
|
||||||
<span>{commit.substring(41)}</span>
|
<span>{commit.substring(41)}</span>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
@(repository: service.RepositoryService.RepositoryInfo, content: String, enableWikiLink: Boolean,
|
@(repository: service.RepositoryService.RepositoryInfo, content: String, enableWikiLink: Boolean, enableRefsLink: Boolean,
|
||||||
enableCommitLink: Boolean, enableIssueLink: Boolean, style: String = "", placeholder: String = "Leave a comment")(implicit context: app.Context)
|
style: String = "", placeholder: String = "Leave a comment")(implicit context: app.Context)
|
||||||
@import context._
|
@import context._
|
||||||
@import view.helpers._
|
@import view.helpers._
|
||||||
<div class="tabbable">
|
<div class="tabbable">
|
||||||
@@ -30,10 +30,9 @@ $(function(){
|
|||||||
$('#preview').click(function(){
|
$('#preview').click(function(){
|
||||||
$('#preview-area').html('<img src="@assets/common/images/indicator.gif"> Previewing...');
|
$('#preview-area').html('<img src="@assets/common/images/indicator.gif"> Previewing...');
|
||||||
$.post('@url(repository)/_preview', {
|
$.post('@url(repository)/_preview', {
|
||||||
content : $('#content').val(),
|
content : $('#content').val(),
|
||||||
enableWikiLink : @enableWikiLink,
|
enableWikiLink : @enableWikiLink,
|
||||||
enableCommitLink : @enableCommitLink,
|
enableRefsLink : @enableRefsLink
|
||||||
enableIssueLink : @enableIssueLink
|
|
||||||
}, function(data){
|
}, function(data){
|
||||||
$('#preview-area').html(data);
|
$('#preview-area').html(data);
|
||||||
prettyPrint();
|
prettyPrint();
|
||||||
|
|||||||
@@ -43,7 +43,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<hr>
|
<hr>
|
||||||
@helper.html.preview(repository, "", false, true, true, "width: 600px; height: 200px;")
|
@helper.html.preview(repository, "", false, true, "width: 600px; height: 200px;")
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="pull-right">
|
<div class="pull-right">
|
||||||
|
|||||||
@@ -64,7 +64,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="issue-content" id="issueContent">
|
<div class="issue-content" id="issueContent">
|
||||||
@markdown(issue.content getOrElse "No description given.", repository, false, true, true)
|
@markdown(issue.content getOrElse "No description given.", repository, false, true)
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -82,7 +82,7 @@
|
|||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="box-content"class="issue-content" id="commentContent-@comment.commentId">
|
<div class="box-content"class="issue-content" id="commentContent-@comment.commentId">
|
||||||
@markdown(comment.content, repository, false, true, true)
|
@markdown(comment.content, repository, false, true)
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@comment.action.map { action =>
|
@comment.action.map { action =>
|
||||||
@@ -98,18 +98,18 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
@if(loginAccount.isDefined){
|
@if(loginAccount.isDefined){
|
||||||
<form action="@url(repository)/issue_comments/new" method="POST" validate="true">
|
<form method="POST" validate="true">
|
||||||
<div class="issue-avatar-image">@avatar(loginAccount.get.userName, 48)</div>
|
<div class="issue-avatar-image">@avatar(loginAccount.get.userName, 48)</div>
|
||||||
<div class="box issue-comment-box">
|
<div class="box issue-comment-box">
|
||||||
<div class="box-content">
|
<div class="box-content">
|
||||||
@helper.html.preview(repository, "", false, true, true, "width: 680px; height: 100px;")
|
@helper.html.preview(repository, "", false, true, "width: 680px; height: 100px;")
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="pull-right">
|
<div class="pull-right">
|
||||||
<input type="hidden" name="issueId" value="@issue.issueId"/>
|
<input type="hidden" name="issueId" value="@issue.issueId"/>
|
||||||
<input type="submit" class="btn btn-success" value="Comment"/>
|
<input type="submit" class="btn btn-success" formaction="@url(repository)/issue_comments/new" value="Comment"/>
|
||||||
@if(hasWritePermission || issue.openedUserName == loginAccount.get.userName){
|
@if(hasWritePermission || issue.openedUserName == loginAccount.get.userName){
|
||||||
<input type="submit" class="btn" value="@{if(issue.closed) "Reopen" else "Close"}" id="action"/>
|
<input type="submit" class="btn" formaction="@url(repository)/issue_comments/state" value="@{if(issue.closed) "Reopen" else "Close"}" id="action"/>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
@(issues: List[(model.Issue, List[model.Label], Int)],
|
@(issues: List[(model.Issue, List[model.Label], Int)],
|
||||||
page: Int,
|
page: Int,
|
||||||
labels: List[model.Label],
|
collaborators: List[String],
|
||||||
milestones: List[model.Milestone],
|
milestones: List[model.Milestone],
|
||||||
|
labels: List[model.Label],
|
||||||
openCount: Int,
|
openCount: Int,
|
||||||
closedCount: Int,
|
closedCount: Int,
|
||||||
allCount: Int,
|
allCount: Int,
|
||||||
@@ -178,15 +179,16 @@
|
|||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
} else {
|
} else {
|
||||||
|
@if(hasWritePermission){
|
||||||
<tr>
|
<tr>
|
||||||
<td style="background-color: #eee;">
|
<td style="background-color: #eee;">
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
<button class="btn btn-mini"><strong>@if(condition.state == "open"){ Close } else { Reopen }</strong></button>
|
<button class="btn btn-mini" id="state"><strong>@{if(condition.state == "open") "Close" else "Reopen"}</strong></button>
|
||||||
</div>
|
</div>
|
||||||
@helper.html.dropdown("Label") {
|
@helper.html.dropdown("Label") {
|
||||||
@labels.map { label =>
|
@labels.map { label =>
|
||||||
<li>
|
<li>
|
||||||
<a href="javascript:void(0);" class="toggle-label" data-label-id="@label.labelId">
|
<a href="javascript:void(0);" class="toggle-label" data-id="@label.labelId">
|
||||||
<i class="icon-white"></i>
|
<i class="icon-white"></i>
|
||||||
<span class="label" style="background-color: #@label.color;"> </span>
|
<span class="label" style="background-color: #@label.color;"> </span>
|
||||||
@label.labelName
|
@label.labelName
|
||||||
@@ -195,24 +197,30 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
@helper.html.dropdown("Assignee") {
|
@helper.html.dropdown("Assignee") {
|
||||||
<li><a href="javascript:void(0);" class="assign" data-name=""><i class="icon-remove-circle"></i> Clear assignee</a></li>
|
<li><a href="javascript:void(0);" class="toggle-assign" data-name=""><i class="icon-remove-circle"></i> Clear assignee</a></li>
|
||||||
<li class="divider"></li>
|
<li class="divider"></li>
|
||||||
|
@collaborators.map { collaborator =>
|
||||||
|
<li><a href="javascript:void(0);" class="toggle-assign" data-name="@collaborator"><i class="icon-white"></i>@avatar(collaborator, 20) @collaborator</a></li>
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@helper.html.dropdown("Milestone") {
|
@helper.html.dropdown("Milestone") {
|
||||||
<li><a href="javascript:void(0);" class="milestone" data-id=""><i class="icon-remove-circle"></i> Clear this milestone</a></li>
|
<li><a href="javascript:void(0);" class="toggle-milestone" data-id=""><i class="icon-remove-circle"></i> Clear this milestone</a></li>
|
||||||
<li class="divider"></li>
|
<li class="divider"></li>
|
||||||
@milestones.map { milestone =>
|
@milestones.map { milestone =>
|
||||||
<li><a href="javascript:void(0);" class="milestone" data-id="@milestone.milestoneId"><i class="icon-white"></i> @milestone.title</a></li>
|
<li><a href="javascript:void(0);" class="toggle-milestone" data-id="@milestone.milestoneId"><i class="icon-white"></i> @milestone.title</a></li>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@issues.map { case (issue, labels, commentCount) =>
|
@issues.map { case (issue, labels, commentCount) =>
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
<td>
|
||||||
|
@if(hasWritePermission){
|
||||||
<label class="checkbox" style="cursor: default;">
|
<label class="checkbox" style="cursor: default;">
|
||||||
<input type="checkbox" value="@issue.issueId"/>
|
<input type="checkbox" value="@issue.issueId"/>
|
||||||
|
}
|
||||||
<a href="@url(repository)/issues/@issue.issueId" class="issue-title">@issue.title</a>
|
<a href="@url(repository)/issues/@issue.issueId" class="issue-title">@issue.title</a>
|
||||||
@labels.map { label =>
|
@labels.map { label =>
|
||||||
<span class="label-color small" style="background-color: #@label.color; color: #@label.fontColor; padding-left: 4px; padding-right: 4px">@label.labelName</span>
|
<span class="label-color small" style="background-color: #@label.color; color: #@label.fontColor; padding-left: 4px; padding-right: 4px">@label.labelName</span>
|
||||||
@@ -257,7 +265,28 @@ $(function(){
|
|||||||
$('.table-issues input[type=checkbox]').change(function(){
|
$('.table-issues input[type=checkbox]').change(function(){
|
||||||
$('.table-issues button').prop('disabled',
|
$('.table-issues button').prop('disabled',
|
||||||
!$('.table-issues input[type=checkbox]').filter(':checked').length);
|
!$('.table-issues input[type=checkbox]').filter(':checked').length);
|
||||||
}).change();
|
}).filter(':first').change();
|
||||||
|
|
||||||
|
var submitBatchEdit = function(action, value) {
|
||||||
|
var checked = $('.table-issues input[type=checkbox]').filter(':checked').map(function(){ return this.value; }).get().join();
|
||||||
|
$('<form>').attr({action : action, method : 'POST'})
|
||||||
|
.append($('<input type="hidden">').attr('name', 'value').val(value))
|
||||||
|
.append($('<input type="hidden">').attr('name', 'checked').val(checked))
|
||||||
|
.submit();
|
||||||
|
};
|
||||||
|
|
||||||
|
$('#state').click(function(){
|
||||||
|
submitBatchEdit('@url(repository)/issues/batchedit/state', $(this).text().toLowerCase());
|
||||||
|
});
|
||||||
|
$('a.toggle-label').click(function(){
|
||||||
|
submitBatchEdit('@url(repository)/issues/batchedit/label', $(this).data('id'));
|
||||||
|
});
|
||||||
|
$('a.toggle-assign').click(function(){
|
||||||
|
submitBatchEdit('@url(repository)/issues/batchedit/assign', $(this).data('name'));
|
||||||
|
});
|
||||||
|
$('a.toggle-milestone').click(function(){
|
||||||
|
submitBatchEdit('@url(repository)/issues/batchedit/milestone', $(this).data('id'));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -81,7 +81,7 @@
|
|||||||
</div>
|
</div>
|
||||||
@if(milestone.description.isDefined){
|
@if(milestone.description.isDefined){
|
||||||
<div class="milestone-description">
|
<div class="milestone-description">
|
||||||
@markdown(milestone.description.get, repository, false, false, false)
|
@markdown(milestone.description.get, repository, false, false)
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
</td>
|
</td>
|
||||||
|
|||||||
@@ -23,6 +23,7 @@
|
|||||||
<tr>
|
<tr>
|
||||||
<th style="font-weight: normal;">
|
<th style="font-weight: normal;">
|
||||||
<div class="pull-left">
|
<div class="pull-left">
|
||||||
|
@avatar(latestCommit.committer, 20)
|
||||||
<a href="@url(latestCommit.committer)" class="username strong">@latestCommit.committer</a>
|
<a href="@url(latestCommit.committer)" class="username strong">@latestCommit.committer</a>
|
||||||
<span class="muted">@datetime(latestCommit.time)</span>
|
<span class="muted">@datetime(latestCommit.time)</span>
|
||||||
<a href="@url(repository)/commit/@latestCommit.id" class="commit-message">@link(latestCommit.summary, repository)</a>
|
<a href="@url(repository)/commit/@latestCommit.id" class="commit-message">@link(latestCommit.summary, repository)</a>
|
||||||
|
|||||||
@@ -39,17 +39,19 @@
|
|||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div class="commit-avatar-image">@avatar(commit.committer, 40)</div>
|
<div class="commit-avatar-image">@avatar(commit.committer, 40)</div>
|
||||||
<a href="@url(repository)/commit/@commit.id" class="commit-message" style="font-weight: bold;">@link(commit.summary, repository)</a>
|
<div class="commit-message-box">
|
||||||
@if(commit.description.isDefined){
|
<a href="@url(repository)/commit/@commit.id" class="commit-message" style="font-weight: bold;">@link(commit.summary, repository)</a>
|
||||||
<a href="javascript:void(0)" onclick="$('#description-@commit.id').toggle();" class="omit">...</a>
|
@if(commit.description.isDefined){
|
||||||
}
|
<a href="javascript:void(0)" onclick="$('#description-@commit.id').toggle();" class="omit">...</a>
|
||||||
<br>
|
}
|
||||||
@if(commit.description.isDefined){
|
<br>
|
||||||
<pre id="description-@commit.id" style="display: none;" class="commit-description">@link(commit.description.get, repository)</pre>
|
@if(commit.description.isDefined){
|
||||||
}
|
<pre id="description-@commit.id" style="display: none;" class="commit-description">@link(commit.description.get, repository)</pre>
|
||||||
<div class="small">
|
}
|
||||||
<a href="@url(commit.committer)" class="username">@commit.committer</a>
|
<div class="small">
|
||||||
<span class="muted">@datetime(commit.time)</span>
|
<a href="@url(commit.committer)" class="username">@commit.committer</a>
|
||||||
|
<span class="muted">@datetime(commit.time)</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
|
|||||||
@@ -27,8 +27,6 @@
|
|||||||
<a href="@url(repository)/commit/@latestCommit.id" class="commit-message">@link(latestCommit.summary, repository)</a>
|
<a href="@url(repository)/commit/@latestCommit.id" class="commit-message">@link(latestCommit.summary, repository)</a>
|
||||||
@if(latestCommit.description.isDefined){
|
@if(latestCommit.description.isDefined){
|
||||||
<a href="javascript:void(0)" onclick="$('#description-@latestCommit.id').toggle();" class="omit">...</a>
|
<a href="javascript:void(0)" onclick="$('#description-@latestCommit.id').toggle();" class="omit">...</a>
|
||||||
}
|
|
||||||
@if(latestCommit.description.isDefined){
|
|
||||||
<pre id="description-@latestCommit.id" class="commit-description" style="display: none;">@link(latestCommit.description.get, repository)</pre>
|
<pre id="description-@latestCommit.id" class="commit-description" style="display: none;">@link(latestCommit.description.get, repository)</pre>
|
||||||
}
|
}
|
||||||
</th>
|
</th>
|
||||||
@@ -79,7 +77,7 @@
|
|||||||
@readme.map { content =>
|
@readme.map { content =>
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<div class="box-header">README.md</div>
|
<div class="box-header">README.md</div>
|
||||||
<div class="box-content">@markdown(content, repository, false, false, false)</div>
|
<div class="box-content">@markdown(content, repository, false, false)</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -14,8 +14,8 @@
|
|||||||
<li class="pull-right">
|
<li class="pull-right">
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
@if(pageName.isDefined){
|
@if(pageName.isDefined){
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName">View Page</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)">View Page</a>
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName/_history">Back to Page History</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)/_history">Back to Page History</a>
|
||||||
} else {
|
} else {
|
||||||
<a class="btn" href="@url(repository)/wiki/_history">Back to Wiki History</a>
|
<a class="btn" href="@url(repository)/wiki/_history">Back to Wiki History</a>
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,9 +13,9 @@
|
|||||||
<li class="pull-right">
|
<li class="pull-right">
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
@if(pageName != ""){
|
@if(pageName != ""){
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName">View Page</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)">View Page</a>
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName/_delete" id="delete">Delete Page</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)/_delete" id="delete">Delete Page</a>
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName/_history">Page History</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)/_history">Page History</a>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</li>
|
</li>
|
||||||
@@ -23,7 +23,7 @@
|
|||||||
<form action="@url(repository)/wiki/@if(pageName == ""){_new} else {_edit}" method="POST" validate="true">
|
<form action="@url(repository)/wiki/@if(pageName == ""){_new} else {_edit}" method="POST" validate="true">
|
||||||
<span id="error-pageName" class="error"></span>
|
<span id="error-pageName" class="error"></span>
|
||||||
<input type="text" name="pageName" value="@pageName" style="width: 900px; font-weight: bold;" placeholder="Input a page name."/>
|
<input type="text" name="pageName" value="@pageName" style="width: 900px; font-weight: bold;" placeholder="Input a page name."/>
|
||||||
@helper.html.preview(repository, page.map(_.content).getOrElse(""), true, false, false, "width: 900px; height: 400px;", "")
|
@helper.html.preview(repository, page.map(_.content).getOrElse(""), true, false, "width: 900px; height: 400px;", "")
|
||||||
<input type="text" name="message" value="" style="width: 900px;" placeholder="Write a small message here explaining this change. (Optional)"/>
|
<input type="text" name="message" value="" style="width: 900px;" placeholder="Write a small message here explaining this change. (Optional)"/>
|
||||||
<input type="hidden" name="currentPageName" value="@pageName"/>
|
<input type="hidden" name="currentPageName" value="@pageName"/>
|
||||||
<input type="submit" value="Save" class="btn btn-success">
|
<input type="submit" value="Save" class="btn btn-success">
|
||||||
|
|||||||
@@ -23,9 +23,9 @@
|
|||||||
<a class="btn" href="@url(repository)/wiki/_new">New Page</a>
|
<a class="btn" href="@url(repository)/wiki/_new">New Page</a>
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName">View Page</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)">View Page</a>
|
||||||
@if(loginAccount.isDefined){
|
@if(loginAccount.isDefined){
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName/_edit">Edit Page</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)/_edit">Edit Page</a>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
@@ -58,7 +58,7 @@
|
|||||||
location.href = '@url(repository)/wiki/_compare/' +
|
location.href = '@url(repository)/wiki/_compare/' +
|
||||||
$(e.get(1)).attr('value') + '...' + $(e.get(0)).attr('value');
|
$(e.get(1)).attr('value') + '...' + $(e.get(0)).attr('value');
|
||||||
} else {
|
} else {
|
||||||
location.href = '@url(repository)/wiki/@pageName.get/_compare/' +
|
location.href = '@url(repository)/wiki/@urlEncode(pageName.get)/_compare/' +
|
||||||
$(e.get(1)).attr('value') + '...' + $(e.get(0)).attr('value');
|
$(e.get(1)).attr('value') + '...' + $(e.get(0)).attr('value');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,19 +15,16 @@
|
|||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
@if(hasWritePermission){
|
@if(hasWritePermission){
|
||||||
<a class="btn" href="@url(repository)/wiki/_new">New Page</a>
|
<a class="btn" href="@url(repository)/wiki/_new">New Page</a>
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName/_edit">Edit Page</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)/_edit">Edit Page</a>
|
||||||
}
|
}
|
||||||
<a class="btn" href="@url(repository)/wiki/@pageName/_history">Page History</a>
|
<a class="btn" href="@url(repository)/wiki/@urlEncode(pageName)/_history">Page History</a>
|
||||||
</div>
|
</div>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
<div class="markdown-body">
|
<div class="markdown-body">
|
||||||
@markdown(page.content, repository, true, false, false)
|
@markdown(page.content, repository, true, false)
|
||||||
</div>
|
</div>
|
||||||
<div class="small">
|
<div class="small">
|
||||||
<span class="muted">Last edited by @page.committer at @datetime(page.time)</span>
|
<span class="muted">Last edited by @page.committer at @datetime(page.time)</span>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
<script>
|
|
||||||
$(function(){ prettyPrint(); });
|
|
||||||
</script>
|
|
||||||
|
|||||||
@@ -18,7 +18,7 @@
|
|||||||
</ul>
|
</ul>
|
||||||
<ul>
|
<ul>
|
||||||
@pages.map { page =>
|
@pages.map { page =>
|
||||||
<li><a href="@url(repository)/wiki/@page">@page</a></li>
|
<li><a href="@url(repository)/wiki/@urlEncode(page)">@page</a></li>
|
||||||
}
|
}
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
|
|||||||
@@ -345,6 +345,10 @@ div.commit-avatar-image {
|
|||||||
margin-right: 4px;
|
margin-right: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
div.commit-message-box {
|
||||||
|
margin-left: 42px;
|
||||||
|
}
|
||||||
|
|
||||||
pre.commit-description {
|
pre.commit-description {
|
||||||
font-weight: normal;
|
font-weight: normal;
|
||||||
border: none;
|
border: none;
|
||||||
|
|||||||
@@ -2,6 +2,12 @@ $(function(){
|
|||||||
$.each($('form[validate=true]'), function(i, form){
|
$.each($('form[validate=true]'), function(i, form){
|
||||||
$(form).submit(validate);
|
$(form).submit(validate);
|
||||||
});
|
});
|
||||||
|
$.each($('input[formaction]'), function(i, input){
|
||||||
|
$(input).click(function(){
|
||||||
|
var form = $(input).parents('form')
|
||||||
|
$(form).attr('action', $(input).attr('formaction'))
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
function validate(e){
|
function validate(e){
|
||||||
@@ -19,6 +25,7 @@ function validate(e){
|
|||||||
form.data('validated', true);
|
form.data('validated', true);
|
||||||
form.submit();
|
form.submit();
|
||||||
} else {
|
} else {
|
||||||
|
form.data('validated', false);
|
||||||
displayErrors(data);
|
displayErrors(data);
|
||||||
}
|
}
|
||||||
}, 'json');
|
}, 'json');
|
||||||
|
|||||||
Reference in New Issue
Block a user