[Notice] Announcing the End of Demo Server [Read me]
File name
Commit message
Commit date
File name
Commit message
Commit date
File name
Commit message
Commit date
File name
Commit message
Commit date
@**
* Yobi, Project Hosting SW
*
* Copyright 2012 NAVER Corp.
* http://yobi.io
*
* @author Ahn Hyeok Jun
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
**@
@(title:String, issue:Issue, issueForm: play.data.Form[Issue], commentForm: play.data.Form[Comment],project:Project)
@import org.apache.commons.lang.StringUtils
@import helper._
@import scala.collection.mutable.Map
@import models.enumeration.ResourceType
@import models.enumeration.Operation
@import models.Milestone
@import java.net._
@import java.text.SimpleDateFormat
@import java.security.MessageDigest
@import utils.JodaDateUtil
@import utils.TemplateHelper._
@import utils.AccessControl._
@import utils.Config
@import play.libs.Json.toJson
@import utils.Markdown
@getTitle(issue:Issue) = @{ "#" + issue.getNumber + " " + issue.title }
@isFirstState(state:State) = {@if(issue.state.state == Issue.availableStates.get(0).state()){dirty}}
@urlToIssues = @{
requestHeader.headers.get("Referer") match {
case Some(u) => urlToList(u, routes.IssueApp.issues(project.owner, project.name, "open").toString())
case _ => routes.IssueApp.issues(project.owner, project.name, "open").toString()
}
}
@urlToVote = @{
if(issue.isVotedBy(UserApp.currentUser)){
routes.VoteApp.unvote(project.owner, project.name, issue.getNumber).toString
} else {
routes.VoteApp.vote(project.owner, project.name, issue.getNumber).toString
}
}
@getVoteButtonTitle = {
@if(issue.isVotedBy(UserApp.currentUser)){
@Messages("issue.unvote.description")
} else {
@Messages("issue.vote.description")
}
}
@titleForOGTag = @{getTitle(issue) + " |:| " + issue.body.substring(0, Math.min(issue.body.length, 200))}
@projectLayout(titleForOGTag, project, utils.MenuType.ISSUE){
@projectMenu(project, utils.MenuType.ISSUE, "main-menu-only")
<div class="page-wrap-outer">
<div class="project-page-wrap board-view">
@** Post Info **@
<div class="board-header issue">
<div class="pull-right mr10 mt10 hide-in-mobile">
<div class="date" title="@JodaDateUtil.getDateString(issue.createdDate)">
@agoOrDateString(issue.createdDate)
</div>
<span class="badge badge-issue-@issue.state.state.toLowerCase">@Messages("issue.state." + issue.state.state)</span>
</div>
<div class="title">
<strong class="board-id">#@issue.getNumber</strong> @issue.title
<div class="pull-right hide show-in-mobile" style="font-size: 0.7em">
<span class="date" title="@JodaDateUtil.getDateString(issue.createdDate)">
@agoOrDateString(issue.createdDate)
</span>
<span class="badge badge-small badge-issue-@issue.state.state.toLowerCase">@Messages("issue.state." + issue.state.state)</span>
</div>
</div>
</div>
@** Content body **@
<!--board-body-->
<div class="board-body row-fluid">
<div class="span9 span-left-pane">
<div class="author-info">
<a href="@userInfo(issue.authorLoginId)" class="usf-group">
<span class="avatar-wrap smaller">
<img src="@User.findByLoginId(issue.authorLoginId).avatarUrl" width="20" height="20">
</span>
@if(issue.authorLoginId != null){
<strong class="name">@issue.authorName</strong>
<span class="loginid"> <strong>@{"@"}</strong>@issue.authorLoginId</span>
} else {
<strong class="name">@Messages("issue.noAuthor")</strong>
}
</a>
</div>
@if(StringUtils.isEmpty(issue.body)){
<div class="content empty-content"></div>
} else {
<div class="content markdown-wrap">@Html(Markdown.render(issue.body, issue.project))</div>
}
<div class="attachments" id="attachments" data-attachments="@toJson(AttachmentApp.getFileList(ResourceType.ISSUE_POST.toString(), issue.id.toString()))"></div>
<div class="board-actrow right-txt">
<div class="pull-left">
<div>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.WATCH)) {
<button id="watch-button" type="button" class="ybtn @if(issue.getWatchers(false).contains(UserApp.currentUser())) {ybtn-watching}"
data-toggle="button" data-watching="@if(issue.getWatchers(false).contains(UserApp.currentUser())){true}else{false}">
@if(issue.getWatchers(false).contains(UserApp.currentUser())) {
@Messages("issue.unwatch")
} else {
@Messages("issue.watch")
}
</button>
}
<div id="vote" class="vote-wrap">
@if(isResourceCreatable(UserApp.currentUser, issue.asResource(), ResourceType.ISSUE_COMMENT)) {
<a href="@urlToVote" class="ybtn @if(issue.isVotedBy(UserApp.currentUser)){ybtn-watching}" title="@getVoteButtonTitle"
data-request-method="post" data-toggle="tooltip">
<span class="heart"><i class="yobicon-hearts"></i></span>
<span class="desc">@Messages("issue.vote")</span>
@if(issue.voters.size > 0) {
<strong class="count">@issue.voters.size</strong>
}
</a>
} else {
<span class="ybtn ybtn-disabled" style="color:#777;" data-toggle="tooltip" title="@Messages("user.login.alert")" data-login="required">
<span class="heart"><i class="yobicon-hearts"></i></span>
<span class="desc">@Messages("issue.vote")</span>
@if(issue.voters.size > 0) {
<strong class="count">@issue.voters.size</strong>
}
</span>
}
@if(issue.voters.size > 0){
@partial_voters(issue, 3)
}
</div>
</div>
</div>
@if(issue.canBeDeleted) {
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.DELETE)) {
<a href="#deleteConfirm" data-toggle="modal" class="ybtn ybtn-danger">@Messages("button.delete")</a>
}
} else {
<button class="ybtn ybtn-disabled" data-toggle="tooltip" data-placement="top" title="@Messages("issue.can.not.be.deleted")">@Messages("button.delete")</button>
}
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
<a href="@routes.IssueApp.editIssueForm(project.owner, project.name, issue.getNumber)" class="ybtn">@Messages("button.edit")</a>
}
</div>
@** Comment **@
<div id="comments" class="board-comment-wrap">
<div id="timeline">
<div class="timeline-list">
@partial_comments(project, issue)
</div>
</div>
@common.commentForm(issue.asResource(), ResourceType.ISSUE_COMMENT, routes.IssueApp.newComment(project.owner, project.name, issue.getNumber).toString())
</div>
@** // Comment **@
</div>
<div class="span3 span-right-pane mb20">
<div class="issue-info">
<form id="issueUpdateForm" action="@routes.IssueApp.massUpdate(project.owner, project.name)" method="post">
<input type="hidden" name="issues[0].id" value="@issue.id" />
@**<!-- assignee -->**@
<dl>
<dt>@Messages("issue.assignee")</dt>
<dd>
@defining(issue.assigneeName != null) { isAssigned =>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
@partial_assignee(project, issue)
} else {
@if(isAssigned){
<a href="@userInfo(issue.assignee.user.loginId)" class="usf-group">
<span class="avatar-wrap smaller">
<img src="@User.findByLoginId(issue.assignee.user.loginId).avatarUrl" width="20" height="20">
</span>
<strong class="name">@issue.assigneeName</strong>
<span class="loginid"> <strong>@{"@"}</strong>@issue.assignee.user.loginId</span>
</a>
} else {
<div>
@Messages("issue.noAssignee")
</div>
}
}
}
</dd>
</dl>
@**<!-- // -->**@
@**<!-- milestones -->**@
@if(project.menuSetting.milestone) {
<dl>
<dt>@Messages("milestone")</dt>
<dd>
@if(Milestone.findByProjectId(project.id).isEmpty()){
<a href="@routes.MilestoneApp.newMilestoneForm(project.owner, project.name)"
class="ybtn ybtn-small ybtn-fullsize" target="_blank">
@Messages("milestone.menu.new")
</a>
} else {
@defining(issue.milestone != null) { hasMilestone =>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
<select id="milestone" name="milestone.id"
data-toggle="select2" data-format="milestone" data-container-css-class="fullsize">
<option value="@Milestone.NULL_MILESTONE_ID" @if(!hasMilestone){ selected }>
@Messages("issue.noMilestone")
</option>
<optgroup label="@Messages("milestone.state.open")">
@for(milestone <- Milestone.findOpenMilestones(project.id)){
<option value="@milestone.id" data-state="@milestone.state"
@if(hasMilestone && issue.milestone.id == milestone.id){
selected
}>
@milestone.title
</option>
}
</optgroup>
<optgroup label="@Messages("milestone.state.closed")">
@for(milestone <- Milestone.findClosedMilestones(project.id)){
<option value="@milestone.id" data-state="@milestone.state"
@if(hasMilestone && issue.milestone.id == milestone.id){
selected
}>
@milestone.title
</option>
}
</optgroup>
</select>
} else {
@if(hasMilestone){
<a href="@routes.MilestoneApp.milestone(project.owner, project.name, issue.milestone.id)">
@issue.milestone.title
</a>
} else {
@Messages("issue.noMilestone")
}
}
}
}
</dd>
</dl>
}
@**<!-- // -->**@
<dl>
<dt>
@Messages("issue.dueDate")
<span class="duedate-status @if(issue.isOverDueDate) {overdue}">
@if(issue.dueDate != null) {
@if(issue.isOverDueDate) {
(@Messages("issue.dueDate.overdue"))
} else {
(@issue.until)
}
}
</span>
</dt>
<dd>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
<div class="search search-bar">
<input type="text" name="dueDate" value="@issue.getDueDateString" class="textbox full" autocomplete="off" data-toggle="calendar">
<button type="button" class="search-btn btn-calendar"><i class="yobicon-calendar2"></i></button>
</div>
} else {
@if(issue.dueDate != null) {
@issue.getDueDateString
} else {
@Messages("issue.noDuedate")
}
}
</dd>
</dl>
@**<!-- labels -->**@
@if(!IssueLabel.findByProject(project).isEmpty){
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)){
@partial_select_label(IssueLabel.findByProject(project), issue.getLabelIds)
} else {
@partial_show_selected_label(issue.labels.toList, routes.IssueApp.issues(project.owner, project.name, issue.state.state(), "html", 1).toString)
}
}
@**<!-- // -->**@
</form>
</div>
</div>
</div>
<div class="board-footer">
<a href="@urlToIssues" class="ybtn pull-left" data-button="historyBack">@Messages("button.list")</a>
@help.keymap("issueDetail", project)
</div>
</div>
<script type="text/x-jquery-tmpl" id="tplAttachedFile"><!--
--><li class="attached-file" data-name="${fileName}" data-href="${fileHref}" data-mime="${mimeType}" data-size="${fileSize}">
<strong>${fileName}(${fileSizeReadable})${notice}</strong><!--
--><a class="attached-delete"><i class="ico btn-delete"></i></a></li>
</script>
@** Confirm to delete post **@
<div id="deleteConfirm" class="modal hide fade">
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal">×</button>
<h3>@Messages("issue.delete")</h3>
</div>
<div class="modal-body">
<p>@Messages("post.delete.confirm")</p>
</div>
<div class="modal-footer">
<button type="button" class="ybtn ybtn-danger" data-request-method="delete" data-request-uri="@routes.IssueApp.deleteIssue(project.owner, project.name, issue.getNumber)">@Messages("button.yes")</button>
<button type="button" class="ybtn" data-dismiss="modal">@Messages("button.no")</button>
</div>
</div>
</div>
@common.markdown(project)
@common.commentDeleteModal()
@common.select2()
@common.calendar()
<link rel="stylesheet" type="text/css" media="screen" href="@routes.IssueLabelApp.labelStyles(project.owner, project.name)">
<link rel="stylesheet" type="text/css" media="screen" href="@routes.Assets.at("javascripts/lib/atjs/jquery.atwho.css")">
<script type="text/javascript" src="@routes.Assets.at("javascripts/lib/atjs/jquery.caret.min.js")"></script>
<script type="text/javascript" src="@routes.Assets.at("javascripts/lib/atjs/jquery.atwho.js")"></script>
<script type="text/javascript">
$(function(){
// yobi.issue.View
$yobi.loadModule("issue.View", {
"issueId" : "@issue.id",
"nextState": "@issue.nextState().toString.toLowerCase",
"urls" : {
"watch" : "@routes.WatchApp.watch(issue.asResource.asParameter)",
"unwatch" : "@routes.WatchApp.unwatch(issue.asResource.asParameter)",
"timeline" : "@routes.IssueApp.timeline(project.owner, project.name, issue.getNumber)",
"nextState" : "@routes.IssueApp.nextState(project.owner, project.name, issue.getNumber)",
"massUpdate": "@routes.IssueApp.massUpdate(project.owner, project.name)"
}
});
// yobi.ShortcutKey
yobi.ShortcutKey.setKeymapLink({
"L": "@urlToIssues"
@if(project.menuSetting.issue) {
,"N": "@routes.IssueApp.newIssueForm(project.owner, project.name)"
}
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
,"E": "@routes.IssueApp.editIssueForm(project.owner, project.name, issue.getNumber)"
}
});
// yobi.Mention
yobi.Mention({
"target": 'textarea[id^=editor-]',
"url" : "@Html(routes.ProjectApp.mentionList(project.owner, project.name, issue.getNumber, issue.asResource().getType.resource()).toString())"
});
// detect comment which contains mention at me
$(".comment-body:contains('@UserApp.currentUser().loginId')").closest(".comment").addClass("mentioned");
});
</script>
}