* Use relative URLs * Notifications - Mark as read/unread * Feature of pinning a notification * On view issue, do not mark as read a pinned notificationtags/v1.2.0-rc1
@@ -589,7 +589,10 @@ func runWeb(ctx *cli.Context) error { | |||
}) | |||
// ***** END: Repository ***** | |||
m.Get("/notifications", reqSignIn, user.Notifications) | |||
m.Group("/notifications", func() { | |||
m.Get("", user.Notifications) | |||
m.Post("/status", user.NotificationStatusPost) | |||
}, reqSignIn) | |||
m.Group("/api", func() { | |||
apiv1.RegisterRoutes(m) | |||
@@ -448,7 +448,7 @@ func (issue *Issue) ReadBy(userID int64) error { | |||
return err | |||
} | |||
if err := setNotificationStatusRead(x, userID, issue.ID); err != nil { | |||
if err := setNotificationStatusReadIfUnread(x, userID, issue.ID); err != nil { | |||
return err | |||
} | |||
@@ -5,6 +5,7 @@ | |||
package models | |||
import ( | |||
"fmt" | |||
"time" | |||
) | |||
@@ -20,6 +21,8 @@ const ( | |||
NotificationStatusUnread NotificationStatus = iota + 1 | |||
// NotificationStatusRead represents a read notification | |||
NotificationStatusRead | |||
// NotificationStatusPinned represents a pinned notification | |||
NotificationStatusPinned | |||
) | |||
const ( | |||
@@ -182,13 +185,19 @@ func getIssueNotification(e Engine, userID, issueID int64) (*Notification, error | |||
} | |||
// NotificationsForUser returns notifications for a given user and status | |||
func NotificationsForUser(user *User, status NotificationStatus, page, perPage int) ([]*Notification, error) { | |||
return notificationsForUser(x, user, status, page, perPage) | |||
func NotificationsForUser(user *User, statuses []NotificationStatus, page, perPage int) ([]*Notification, error) { | |||
return notificationsForUser(x, user, statuses, page, perPage) | |||
} | |||
func notificationsForUser(e Engine, user *User, status NotificationStatus, page, perPage int) (notifications []*Notification, err error) { | |||
func notificationsForUser(e Engine, user *User, statuses []NotificationStatus, page, perPage int) (notifications []*Notification, err error) { | |||
// FIXME: Xorm does not support aliases types (like NotificationStatus) on In() method | |||
s := make([]uint8, len(statuses)) | |||
for i, status := range statuses { | |||
s[i] = uint8(status) | |||
} | |||
sess := e. | |||
Where("user_id = ?", user.ID). | |||
And("status = ?", status). | |||
In("status", s). | |||
OrderBy("updated_unix DESC") | |||
if page > 0 && perPage > 0 { | |||
@@ -241,15 +250,53 @@ func getNotificationCount(e Engine, user *User, status NotificationStatus) (coun | |||
return | |||
} | |||
func setNotificationStatusRead(e Engine, userID, issueID int64) error { | |||
func setNotificationStatusReadIfUnread(e Engine, userID, issueID int64) error { | |||
notification, err := getIssueNotification(e, userID, issueID) | |||
// ignore if not exists | |||
if err != nil { | |||
return nil | |||
} | |||
if notification.Status != NotificationStatusUnread { | |||
return nil | |||
} | |||
notification.Status = NotificationStatusRead | |||
_, err = e.Id(notification.ID).Update(notification) | |||
return err | |||
} | |||
// SetNotificationStatus change the notification status | |||
func SetNotificationStatus(notificationID int64, user *User, status NotificationStatus) error { | |||
notification, err := getNotificationByID(notificationID) | |||
if err != nil { | |||
return err | |||
} | |||
if notification.UserID != user.ID { | |||
return fmt.Errorf("Can't change notification of another user: %d, %d", notification.UserID, user.ID) | |||
} | |||
notification.Status = status | |||
_, err = x.Id(notificationID).Update(notification) | |||
return err | |||
} | |||
func getNotificationByID(notificationID int64) (*Notification, error) { | |||
notification := new(Notification) | |||
ok, err := x. | |||
Where("id = ?", notificationID). | |||
Get(notification) | |||
if err != nil { | |||
return nil, err | |||
} | |||
if !ok { | |||
return nil, fmt.Errorf("Notification %d does not exists", notificationID) | |||
} | |||
return notification, nil | |||
} |
@@ -2712,6 +2712,12 @@ footer .ui.language .menu { | |||
float: left; | |||
margin-left: 7px; | |||
} | |||
.user.notification .buttons-panel button { | |||
padding: 3px; | |||
} | |||
.user.notification .buttons-panel form { | |||
display: inline-block; | |||
} | |||
.user.notification .octicon-issue-opened, | |||
.user.notification .octicon-git-pull-request { | |||
color: #21ba45; | |||
@@ -2722,6 +2728,9 @@ footer .ui.language .menu { | |||
.user.notification .octicon-git-merge { | |||
color: #a333c8; | |||
} | |||
.user.notification .octicon-pin { | |||
color: #2185d0; | |||
} | |||
.dashboard { | |||
padding-top: 15px; | |||
padding-bottom: 80px; | |||
@@ -85,6 +85,16 @@ | |||
margin-left: 7px; | |||
} | |||
.buttons-panel { | |||
button { | |||
padding: 3px; | |||
} | |||
form { | |||
display: inline-block; | |||
} | |||
} | |||
.octicon-issue-opened, .octicon-git-pull-request { | |||
color: #21ba45; | |||
} | |||
@@ -94,5 +104,8 @@ | |||
.octicon-git-merge { | |||
color: #a333c8; | |||
} | |||
.octicon-pin { | |||
color: #2185d0; | |||
} | |||
} | |||
} |
@@ -1,7 +1,9 @@ | |||
package user | |||
import ( | |||
"errors" | |||
"fmt" | |||
"strconv" | |||
"strings" | |||
"github.com/Unknwon/paginater" | |||
@@ -9,6 +11,7 @@ import ( | |||
"code.gitea.io/gitea/models" | |||
"code.gitea.io/gitea/modules/base" | |||
"code.gitea.io/gitea/modules/context" | |||
"code.gitea.io/gitea/modules/setting" | |||
) | |||
const ( | |||
@@ -56,7 +59,8 @@ func Notifications(c *context.Context) { | |||
status = models.NotificationStatusUnread | |||
} | |||
notifications, err := models.NotificationsForUser(c.User, status, page, perPage) | |||
statuses := []models.NotificationStatus{status, models.NotificationStatusPinned} | |||
notifications, err := models.NotificationsForUser(c.User, statuses, page, perPage) | |||
if err != nil { | |||
c.Handle(500, "ErrNotificationsForUser", err) | |||
return | |||
@@ -79,3 +83,32 @@ func Notifications(c *context.Context) { | |||
c.Data["Page"] = paginater.New(int(total), perPage, page, 5) | |||
c.HTML(200, tplNotification) | |||
} | |||
// NotificationStatusPost is a route for changing the status of a notification | |||
func NotificationStatusPost(c *context.Context) { | |||
var ( | |||
notificationID, _ = strconv.ParseInt(c.Req.PostFormValue("notification_id"), 10, 64) | |||
statusStr = c.Req.PostFormValue("status") | |||
status models.NotificationStatus | |||
) | |||
switch statusStr { | |||
case "read": | |||
status = models.NotificationStatusRead | |||
case "unread": | |||
status = models.NotificationStatusUnread | |||
case "pinned": | |||
status = models.NotificationStatusPinned | |||
default: | |||
c.Handle(500, "InvalidNotificationStatus", errors.New("Invalid notification status")) | |||
return | |||
} | |||
if err := models.SetNotificationStatus(notificationID, c.User, status); err != nil { | |||
c.Handle(500, "SetNotificationStatus", err) | |||
return | |||
} | |||
url := fmt.Sprintf("%s/notifications", setting.AppSubURL) | |||
c.Redirect(url, 303) | |||
} |
@@ -82,7 +82,7 @@ | |||
{{if .IsSigned}} | |||
<div class="right menu"> | |||
<a href="/notifications" class="ui head link jump item poping up" data-content='{{.i18n.Tr "notifications"}}' data-variation="tiny inverted"> | |||
<a href="{{$.AppSubUrl}}/notifications" class="ui head link jump item poping up" data-content='{{.i18n.Tr "notifications"}}' data-variation="tiny inverted"> | |||
<span class="text"> | |||
<i class="octicon octicon-inbox"><span class="sr-only">{{.i18n.Tr "notifications"}}</span></i> | |||
@@ -5,7 +5,7 @@ | |||
<h1 class="ui header">{{.i18n.Tr "notification.notifications"}}</h1> | |||
<div class="ui top attached tabular menu"> | |||
<a href="/notifications?q=unread"> | |||
<a href="{{$.AppSubUrl}}/notifications?q=unread"> | |||
<div class="{{if eq .Status 1}}active{{end}} item"> | |||
{{.i18n.Tr "notification.unread"}} | |||
{{if eq .Status 1}} | |||
@@ -13,7 +13,7 @@ | |||
{{end}} | |||
</div> | |||
</a> | |||
<a href="/notifications?q=read"> | |||
<a href="{{$.AppSubUrl}}/notifications?q=read"> | |||
<div class="{{if eq .Status 2}}active{{end}} item"> | |||
{{.i18n.Tr "notification.read"}} | |||
{{if eq .Status 2}} | |||
@@ -30,34 +30,66 @@ | |||
{{.i18n.Tr "notification.no_read"}} | |||
{{end}} | |||
{{else}} | |||
<div class="ui relaxed divided list"> | |||
<div class="ui relaxed divided selection list"> | |||
{{range $notification := .Notifications}} | |||
{{$issue := $notification.GetIssue}} | |||
{{$repo := $notification.GetRepo}} | |||
{{$repoOwner := $repo.MustOwner}} | |||
<div class="item"> | |||
<a href="{{$.AppSubUrl}}/{{$repoOwner.Name}}/{{$repo.Name}}/issues/{{$issue.Index}}"> | |||
{{if and $issue.IsPull}} | |||
{{if $issue.IsClosed}} | |||
<i class="octicon octicon-git-merge"></i> | |||
{{else}} | |||
<i class="octicon octicon-git-pull-request"></i> | |||
{{end}} | |||
<a class="item" href="{{$.AppSubUrl}}/{{$repoOwner.Name}}/{{$repo.Name}}/issues/{{$issue.Index}}"> | |||
<div class="buttons-panel right floated content"> | |||
{{if ne $notification.Status 3}} | |||
<form action="{{$.AppSubUrl}}/notifications/status" method="POST"> | |||
{{$.CsrfTokenHtml}} | |||
<input type="hidden" name="notification_id" value="{{$notification.ID}}" /> | |||
<input type="hidden" name="status" value="pinned" /> | |||
<button class="ui button" title="Pin notification"> | |||
<i class="octicon octicon-pin"></i> | |||
</button> | |||
</form> | |||
{{end}} | |||
{{if or (eq $notification.Status 1) (eq $notification.Status 3)}} | |||
<form action="{{$.AppSubUrl}}/notifications/status" method="POST"> | |||
{{$.CsrfTokenHtml}} | |||
<input type="hidden" name="notification_id" value="{{$notification.ID}}" /> | |||
<input type="hidden" name="status" value="read" /> | |||
<button class="ui button" title="Mark as read"> | |||
<i class="octicon octicon-check"></i> | |||
</button> | |||
</form> | |||
{{else if eq $notification.Status 2}} | |||
<form action="{{$.AppSubUrl}}/notifications/status" method="POST"> | |||
{{$.CsrfTokenHtml}} | |||
<input type="hidden" name="notification_id" value="{{$notification.ID}}" /> | |||
<input type="hidden" name="status" value="unread" /> | |||
<button class="ui button" title="Mark as unread"> | |||
<i class="octicon octicon-bell"></i> | |||
</button> | |||
</form> | |||
{{end}} | |||
</div> | |||
{{if eq $notification.Status 3}} | |||
<i class="blue octicon octicon-pin"></i> | |||
{{else if $issue.IsPull}} | |||
{{if $issue.IsClosed}} | |||
<i class="octicon octicon-git-merge"></i> | |||
{{else}} | |||
<i class="octicon octicon-git-pull-request"></i> | |||
{{end}} | |||
{{else}} | |||
{{if $issue.IsClosed}} | |||
<i class="octicon octicon-issue-closed"></i> | |||
{{else}} | |||
{{if $issue.IsClosed}} | |||
<i class="octicon octicon-issue-closed"></i> | |||
{{else}} | |||
<i class="octicon octicon-issue-opened"></i> | |||
{{end}} | |||
<i class="octicon octicon-issue-opened"></i> | |||
{{end}} | |||
{{end}} | |||
<div class="content"> | |||
<div class="header">{{$repoOwner.Name}}/{{$repo.Name}}</div> | |||
<div class="description">#{{$issue.Index}} - {{$issue.Title}}</div> | |||
</div> | |||
</a> | |||
</div> | |||
<div class="content"> | |||
<div class="header">{{$repoOwner.Name}}/{{$repo.Name}}</div> | |||
<div class="description">#{{$issue.Index}} - {{$issue.Title}}</div> | |||
</div> | |||
</a> | |||
{{end}} | |||
</div> | |||
{{end}} | |||