Allow unsharing, only show share controls for owner

This commit is contained in:
Maximilian Friedersdorff 2025-10-09 21:04:45 +01:00
parent a54abeaea2
commit 4fb4bec5a8
3 changed files with 60 additions and 15 deletions

View file

@ -7,15 +7,23 @@
<a class="btn btn-primary" href="{{.urlEdit}}">Edit</a> <a class="btn btn-primary" href="{{.urlEdit}}">Edit</a>
<a class="btn btn-danger" href="{{.urlDelete}}">Delete</a> <a class="btn btn-danger" href="{{.urlDelete}}">Delete</a>
</div> </div>
{{ if .isOwner }}
<div class="mt-2"> <div class="mt-2">
<h3>Ownership</h3> <h3>Ownership</h3>
{{if .note.Viewers}} {{if .note.Viewers}}
<p>This note is owned by <em>{{.note.Owner}}</em> and is further visible to</p> <p>This note is owned by <em>{{.note.Owner}}</em> and is further visible to</p>
<ul> <form action="{{.urlUnshare}}" method="POST">
<table class="table vertical-align-middle">
{{range .viewers}} {{range .viewers}}
<li>{{.}}</li> <tr>
<td>{{.}}</td>
<td class="text-end">
<button class="btn btn-outline-warning btn-sm" type="submit" name="viewer" value="{{.}}">Un-Share</button>
</td>
</tr>
{{end}} {{end}}
</ul> </table>
</form>
{{else}} {{else}}
<p>This note is owned by <em>{{.note.Owner}}</em>.</p> <p>This note is owned by <em>{{.note.Owner}}</em>.</p>
{{end}} {{end}}
@ -27,6 +35,7 @@
</div> </div>
<button class="btn btn-primary" type="submit">Share</button> <button class="btn btn-primary" type="submit">Share</button>
</div> </div>
{{end}}
<script> <script>
let checkBoxes = document.querySelectorAll('input[type=checkbox]') let checkBoxes = document.querySelectorAll('input[type=checkbox]')

View file

@ -40,8 +40,10 @@ type Note struct {
LastModified time.Time LastModified time.Time
} }
type noteSet map[*Note]bool
type NoteStore struct { type NoteStore struct {
notes map[string][]*Note notes map[string]noteSet
} }
var ( var (
@ -51,7 +53,7 @@ var (
func Init() error { func Init() error {
Notes = NoteStore{ Notes = NoteStore{
notes: make(map[string][]*Note), notes: make(map[string]noteSet),
} }
md = goldmark.New( md = goldmark.New(
@ -95,14 +97,14 @@ func Init() error {
return nil return nil
} }
func (ns *NoteStore) Get(owner string) []*Note { func (ns *NoteStore) Get(owner string) noteSet {
return ns.notes[owner] return ns.notes[owner]
} }
func (ns *NoteStore) GetOne(owner string, uid string) (*Note, bool) { func (ns *NoteStore) GetOne(owner string, uid string) (*Note, bool) {
notes := ns.Get(owner) notes := ns.Get(owner)
for _, note := range notes { for note, _ := range notes {
if note.Uid == uid { if note.Uid == uid {
log.Printf("Found single note during GetOne %s", note.Title) log.Printf("Found single note during GetOne %s", note.Title)
return note, true return note, true
@ -112,7 +114,14 @@ func (ns *NoteStore) GetOne(owner string, uid string) (*Note, bool) {
} }
func (ns *NoteStore) Add(note *Note, user string) { func (ns *NoteStore) Add(note *Note, user string) {
ns.notes[user] = append(ns.notes[user], note) if ns.notes[user] == nil {
ns.notes[user] = make(noteSet)
}
ns.notes[user][note] = true
}
func (ns *NoteStore) Del(note *Note, user string) {
delete(ns.notes[user], note)
} }
func fmtPath(path string) string { func fmtPath(path string) string {
@ -268,6 +277,10 @@ func (n *Note) AddViewer(viewer string) {
n.Viewers[viewer] = struct{}{} n.Viewers[viewer] = struct{}{}
} }
func (n *Note) DelViewer(viewer string) {
delete(n.Viewers, viewer)
}
func DeleteNote(uid string) error { func DeleteNote(uid string) error {
filename := filepath.Join(conf.Conf.NotesDir, fmtPath(uid)) filename := filepath.Join(conf.Conf.NotesDir, fmtPath(uid))
return os.Remove(filename) return os.Remove(filename)

View file

@ -28,6 +28,7 @@ func GetRoutes(prefix string) *http.ServeMux {
"delete": {Path: "/{note}/delete/", Protocol: "GET", Handler: delete}, "delete": {Path: "/{note}/delete/", Protocol: "GET", Handler: delete},
"edit": {Path: "/{note}/edit/", Protocol: "GET", Handler: edit}, "edit": {Path: "/{note}/edit/", Protocol: "GET", Handler: edit},
"share": {Path: "/{note}/share/", Protocol: "POST", Handler: share}, "share": {Path: "/{note}/share/", Protocol: "POST", Handler: share},
"unshare": {Path: "/{note}/unshare/", Protocol: "POST", Handler: unshare},
"save": {Path: "/{note}/edit/save/", Protocol: "POST", Handler: save}, "save": {Path: "/{note}/edit/save/", Protocol: "POST", Handler: save},
"togglebox": {Path: "/{note}/togglebox/", Protocol: "POST", Handler: togglebox}, "togglebox": {Path: "/{note}/togglebox/", Protocol: "POST", Handler: togglebox},
}, },
@ -47,12 +48,17 @@ func view(w http.ResponseWriter, r *http.Request) {
} }
viewers := note.ViewersAsList() viewers := note.ViewersAsList()
urlEdit := myurls.Reverse("edit", urls.Repl{"note": uid}) context := templ.Ctx{
urlNew := myurls.Reverse("new", urls.Repl{}) "note": note,
urlDelete := myurls.Reverse("delete", urls.Repl{"note": uid}) "urlEdit": myurls.Reverse("edit", urls.Repl{"note": uid}),
urlShare := myurls.Reverse("share", urls.Repl{"note": uid}) "urlDelete": myurls.Reverse("delete", urls.Repl{"note": uid}),
"urlNew": myurls.Reverse("new", urls.Repl{}),
"urlShare": myurls.Reverse("share", urls.Repl{"note": uid}),
"urlUnshare": myurls.Reverse("unshare", urls.Repl{"note": uid}),
"viewers": viewers,
"isOwner": user == note.Owner,
}
context := templ.Ctx{"note": note, "urlEdit": urlEdit, "urlDelete": urlDelete, "urlNew": urlNew, "urlShare": urlShare, "viewers": viewers}
note.Render() note.Render()
err := templ.RenderTemplate(w, r, "view.tmpl.html", context) err := templ.RenderTemplate(w, r, "view.tmpl.html", context)
if err != nil { if err != nil {
@ -135,7 +141,7 @@ func share(w http.ResponseWriter, r *http.Request) {
uid := r.PathValue("note") uid := r.PathValue("note")
note, ok := notes.Notes.GetOne(user, uid) note, ok := notes.Notes.GetOne(user, uid)
if !ok { if !ok || note.Owner != user {
http.NotFound(w, r) http.NotFound(w, r)
} }
@ -147,6 +153,23 @@ func share(w http.ResponseWriter, r *http.Request) {
http.Redirect(w, r, myurls.Reverse("view", urls.Repl{"note": note.Uid}), http.StatusFound) http.Redirect(w, r, myurls.Reverse("view", urls.Repl{"note": note.Uid}), http.StatusFound)
} }
func unshare(w http.ResponseWriter, r *http.Request) {
user := r.Context().Value(middleware.ContextKey("user")).(string)
uid := r.PathValue("note")
note, ok := notes.Notes.GetOne(user, uid)
if !ok || note.Owner != user {
http.NotFound(w, r)
}
viewer := r.FormValue("viewer")
note.DelViewer(viewer)
note.Save()
notes.Notes.Del(note, viewer)
http.Redirect(w, r, myurls.Reverse("view", urls.Repl{"note": note.Uid}), http.StatusFound)
}
func togglebox(w http.ResponseWriter, r *http.Request) { func togglebox(w http.ResponseWriter, r *http.Request) {
user := r.Context().Value(middleware.ContextKey("user")).(string) user := r.Context().Value(middleware.ContextKey("user")).(string)
@ -182,7 +205,7 @@ func list(w http.ResponseWriter, r *http.Request) {
log.Printf("Notes: %+v", notes.Notes) log.Printf("Notes: %+v", notes.Notes)
log.Printf("Notes for %s: %+v", user, ns) log.Printf("Notes for %s: %+v", user, ns)
for _, note := range ns { for note := range ns {
titlesAndUrls = append( titlesAndUrls = append(
titlesAndUrls, titlesAndUrls,
titleAndURL{Title: note.Title, URL: myurls.Reverse("view", urls.Repl{"note": note.Uid})}, titleAndURL{Title: note.Title, URL: myurls.Reverse("view", urls.Repl{"note": note.Uid})},