Rename comments --> notes

This commit is contained in:
shamoon 2023-03-17 16:36:08 -07:00
parent 89c639f850
commit bf8ae22f3f
45 changed files with 540 additions and 528 deletions

View File

@ -80,7 +80,7 @@ django_checks() {
search_index() { search_index() {
local -r index_version=3 local -r index_version=4
local -r index_version_file=${DATA_DIR}/.index_version local -r index_version_file=${DATA_DIR}/.index_version
if [[ (! -f "${index_version_file}") || $(<"${index_version_file}") != "$index_version" ]]; then if [[ (! -f "${index_version_file}") || $(<"${index_version_file}") != "$index_version" ]]; then

View File

@ -17,28 +17,28 @@ describe('document-detail', () => {
req.reply({ result: 'OK' }) req.reply({ result: 'OK' })
}).as('saveDoc') }).as('saveDoc')
cy.fixture('documents/1/comments.json').then((commentsJson) => { cy.fixture('documents/1/notes.json').then((notesJson) => {
cy.intercept( cy.intercept(
'GET', 'GET',
'http://localhost:8000/api/documents/1/comments/', 'http://localhost:8000/api/documents/1/notes/',
(req) => { (req) => {
req.reply(commentsJson.filter((c) => c.id != 10)) // 3 req.reply(notesJson.filter((c) => c.id != 10)) // 3
} }
) )
cy.intercept( cy.intercept(
'DELETE', 'DELETE',
'http://localhost:8000/api/documents/1/comments/?id=9', 'http://localhost:8000/api/documents/1/notes/?id=9',
(req) => { (req) => {
req.reply(commentsJson.filter((c) => c.id != 9 && c.id != 10)) // 2 req.reply(notesJson.filter((c) => c.id != 9 && c.id != 10)) // 2
} }
) )
cy.intercept( cy.intercept(
'POST', 'POST',
'http://localhost:8000/api/documents/1/comments/', 'http://localhost:8000/api/documents/1/notes/',
(req) => { (req) => {
req.reply(commentsJson) // 4 req.reply(notesJson) // 4
} }
) )
}) })
@ -75,48 +75,40 @@ describe('document-detail', () => {
cy.get('pdf-viewer').should('be.visible') cy.get('pdf-viewer').should('be.visible')
}) })
it('should show a list of comments', () => { it('should show a list of notes', () => {
cy.wait(1000) cy.wait(1000).get('a').contains('Notes').click({ force: true }).wait(1000)
.get('a') cy.get('app-document-notes').find('.card').its('length').should('eq', 3)
.contains('Comments')
.click({ force: true })
.wait(1000)
cy.get('app-document-comments').find('.card').its('length').should('eq', 3)
}) })
it('should support comment deletion', () => { it('should support note deletion', () => {
cy.wait(1000).get('a').contains('Comments').click().wait(1000) cy.wait(1000).get('a').contains('Notes').click().wait(1000)
cy.get('app-document-comments') cy.get('app-document-notes')
.find('.card') .find('.card')
.first() .first()
.find('button') .find('button')
.click({ force: true }) .click({ force: true })
.wait(500) .wait(500)
cy.get('app-document-comments').find('.card').its('length').should('eq', 2) cy.get('app-document-notes').find('.card').its('length').should('eq', 2)
}) })
it('should support comment insertion', () => { it('should support note insertion', () => {
cy.wait(1000).get('a').contains('Comments').click().wait(1000) cy.wait(1000).get('a').contains('Notes').click().wait(1000)
cy.get('app-document-comments') cy.get('app-document-notes')
.find('form textarea') .find('form textarea')
.type('Testing new comment') .type('Testing new note')
.wait(500) .wait(500)
cy.get('app-document-comments').find('form button').click().wait(1500) cy.get('app-document-notes').find('form button').click().wait(1500)
cy.get('app-document-comments').find('.card').its('length').should('eq', 4) cy.get('app-document-notes').find('.card').its('length').should('eq', 4)
}) })
it('should support navigation to comments tab by url', () => { it('should support navigation to notes tab by url', () => {
cy.visit('/documents/1/comments') cy.visit('/documents/1/notes')
cy.get('app-document-comments').should('exist') cy.get('app-document-notes').should('exist')
}) })
it('should dynamically update comment counts', () => { it('should dynamically update note counts', () => {
cy.visit('/documents/1/comments') cy.visit('/documents/1/notes')
cy.get('app-document-comments').within(() => cy.contains('Delete').click()) cy.get('app-document-notes').within(() => cy.contains('Delete').click())
cy.get('ul.nav') cy.get('ul.nav').find('li').contains('Notes').find('.badge').contains('2')
.find('li')
.contains('Comments')
.find('.badge')
.contains('2')
}) })
}) })

View File

@ -1,46 +0,0 @@
[
{
"id": 10,
"comment": "Testing new comment",
"created": "2022-08-08T04:24:55.176008Z",
"user": {
"id": 1,
"username": "user2",
"first_name": "",
"last_name": ""
}
},
{
"id": 9,
"comment": "Testing one more time",
"created": "2022-02-18T04:24:55.176008Z",
"user": {
"id": 2,
"username": "user1",
"first_name": "",
"last_name": ""
}
},
{
"id": 8,
"comment": "Another comment",
"created": "2021-11-08T04:24:47.925042Z",
"user": {
"id": 2,
"username": "user33",
"first_name": "",
"last_name": ""
}
},
{
"id": 7,
"comment": "Cupcake ipsum dolor sit amet cheesecake candy cookie tiramisu. Donut chocolate chupa chups macaroon brownie halvah pie cheesecake gummies. Sweet chocolate bar candy donut gummi bears bear claw liquorice bonbon shortbread.\n\nDonut chocolate bar candy wafer wafer tiramisu. Gummies chocolate cake muffin toffee carrot cake macaroon. Toffee toffee jelly beans danish lollipop cake.",
"created": "2021-02-08T02:37:49.724132Z",
"user": {
"id": 3,
"username": "admin",
"first_name": "",
"last_name": ""
}
}
]

View File

@ -0,0 +1,26 @@
[
{
"id": 10,
"note": "Testing new note",
"created": "2022-08-08T04:24:55.176008Z",
"user": 3
},
{
"id": 9,
"note": "Testing one more time",
"created": "2022-02-18T04:24:55.176008Z",
"user": 15
},
{
"id": 8,
"note": "Another note",
"created": "2021-11-08T04:24:47.925042Z",
"user": 3
},
{
"id": 7,
"note": "Cupcake ipsum dolor sit amet cheesecake candy cookie tiramisu. Donut chocolate chupa chups macaroon brownie halvah pie cheesecake gummies. Sweet chocolate bar candy donut gummi bears bear claw liquorice bonbon shortbread.\n\nDonut chocolate bar candy wafer wafer tiramisu. Gummies chocolate cake muffin toffee carrot cake macaroon. Toffee toffee jelly beans danish lollipop cake.",
"created": "2021-02-08T02:37:49.724132Z",
"user": 3
}
]

View File

@ -22,39 +22,24 @@
"archived_file_name": "2022-03-22 no latin title.pdf", "archived_file_name": "2022-03-22 no latin title.pdf",
"owner": null, "owner": null,
"permissions": [], "permissions": [],
"comments": [ "notes": [
{ {
"id": 30, "id": 9,
"comment": "One more time", "note": "Testing one more time",
"created": "2023-03-17T22:02:14.357575Z", "created": "2022-02-18T04:24:55.176008Z",
"user": { "user": 15
"id": 2,
"username": "username",
"first_name": "",
"last_name": ""
}
}, },
{ {
"id": 6, "id": 8,
"comment": "Lets keep going", "note": "Another note",
"created": "2023-03-16T06:57:32.014027Z", "created": "2021-11-08T04:24:47.925042Z",
"user": { "user": 3
"id": 2,
"username": "username",
"first_name": "",
"last_name": ""
}
}, },
{ {
"id": 5, "id": 7,
"comment": "And just one more", "note": "Cupcake ipsum dolor sit amet cheesecake candy cookie tiramisu. Donut chocolate chupa chups macaroon brownie halvah pie cheesecake gummies. Sweet chocolate bar candy donut gummi bears bear claw liquorice bonbon shortbread.\n\nDonut chocolate bar candy wafer wafer tiramisu. Gummies chocolate cake muffin toffee carrot cake macaroon. Toffee toffee jelly beans danish lollipop cake.",
"created": "2023-03-16T06:57:27.022729Z", "created": "2021-02-08T02:37:49.724132Z",
"user": { "user": 3
"id": 2,
"username": "username",
"first_name": "",
"last_name": ""
}
} }
] ]
}, },
@ -75,7 +60,7 @@
"archived_file_name": "2022-03-23 llorem ipsum dolor sit amet.pdf", "archived_file_name": "2022-03-23 llorem ipsum dolor sit amet.pdf",
"owner": null, "owner": null,
"permissions": [], "permissions": [],
"comments": [] "notes": []
}, },
{ {
"id": 3, "id": 3,
@ -96,7 +81,7 @@
"archived_file_name": "2022-03-24 dolor.pdf", "archived_file_name": "2022-03-24 dolor.pdf",
"owner": null, "owner": null,
"permissions": [], "permissions": [],
"comments": [] "notes": []
}, },
{ {
"id": 4, "id": 4,
@ -117,7 +102,7 @@
"archived_file_name": "2022-06-01 sit amet.pdf", "archived_file_name": "2022-06-01 sit amet.pdf",
"owner": null, "owner": null,
"permissions": [], "permissions": [],
"comments": [] "notes": []
} }
] ]
} }

View File

@ -11,10 +11,10 @@
"change_user", "change_user",
"delete_user", "delete_user",
"view_user", "view_user",
"add_comment", "add_note",
"change_comment", "change_note",
"delete_comment", "delete_note",
"view_comment" "view_note"
] ]
}, },
{ {
@ -73,10 +73,10 @@
"change_task", "change_task",
"delete_task", "delete_task",
"view_task", "view_task",
"add_comment", "add_note",
"change_comment", "change_note",
"delete_comment", "delete_note",
"view_comment", "view_note",
"add_correspondent", "add_correspondent",
"change_correspondent", "change_correspondent",
"delete_correspondent", "delete_correspondent",

View File

@ -94,10 +94,10 @@
"change_task", "change_task",
"delete_task", "delete_task",
"view_task", "view_task",
"add_comment", "add_note",
"change_comment", "change_note",
"delete_comment", "delete_note",
"view_comment", "view_note",
"add_correspondent", "add_correspondent",
"change_correspondent", "change_correspondent",
"delete_correspondent", "delete_correspondent",

View File

@ -74,7 +74,7 @@
"change_task", "change_task",
"delete_task", "delete_task",
"view_task", "view_task",
"add_comment", "add_note",
"add_frontendsettings", "add_frontendsettings",
"change_frontendsettings", "change_frontendsettings",
"delete_frontendsettings", "delete_frontendsettings",

View File

@ -30,7 +30,7 @@
"django_celery_results.delete_taskresult", "django_celery_results.delete_taskresult",
"paperless_mail.add_mailaccount", "paperless_mail.add_mailaccount",
"auth.change_group", "auth.change_group",
"documents.add_comment", "documents.add_note",
"paperless_mail.delete_mailaccount", "paperless_mail.delete_mailaccount",
"authtoken.delete_tokenproxy", "authtoken.delete_tokenproxy",
"guardian.delete_groupobjectpermission", "guardian.delete_groupobjectpermission",
@ -44,7 +44,7 @@
"documents.add_documenttype", "documents.add_documenttype",
"django_q.change_success", "django_q.change_success",
"documents.delete_tag", "documents.delete_tag",
"documents.change_comment", "documents.change_note",
"django_q.delete_task", "django_q.delete_task",
"documents.add_savedviewfilterrule", "documents.add_savedviewfilterrule",
"django_q.view_task", "django_q.view_task",
@ -59,7 +59,7 @@
"documents.add_savedview", "documents.add_savedview",
"auth.delete_user", "auth.delete_user",
"documents.view_log", "documents.view_log",
"documents.view_comment", "documents.view_note",
"guardian.change_groupobjectpermission", "guardian.change_groupobjectpermission",
"sessions.delete_session", "sessions.delete_session",
"django_q.change_failure", "django_q.change_failure",
@ -139,7 +139,7 @@
"django_celery_results.view_taskresult", "django_celery_results.view_taskresult",
"contenttypes.add_contenttype", "contenttypes.add_contenttype",
"django_q.delete_success", "django_q.delete_success",
"documents.delete_comment", "documents.delete_note",
"django_q.add_failure", "django_q.add_failure",
"guardian.add_userobjectpermission", "guardian.add_userobjectpermission",
"sessions.view_session", "sessions.view_session",
@ -216,10 +216,10 @@
"change_task", "change_task",
"delete_task", "delete_task",
"view_task", "view_task",
"add_comment", "add_note",
"change_comment", "change_note",
"delete_comment", "delete_note",
"view_comment", "view_note",
"add_frontendsettings", "add_frontendsettings",
"change_frontendsettings", "change_frontendsettings",
"delete_frontendsettings", "delete_frontendsettings",
@ -256,7 +256,7 @@
"django_celery_results.delete_taskresult", "django_celery_results.delete_taskresult",
"authtoken.change_token", "authtoken.change_token",
"auth.change_group", "auth.change_group",
"documents.add_comment", "documents.add_note",
"authtoken.delete_tokenproxy", "authtoken.delete_tokenproxy",
"documents.view_documenttype", "documents.view_documenttype",
"contenttypes.delete_contenttype", "contenttypes.delete_contenttype",
@ -285,7 +285,7 @@
"django_q.change_task", "django_q.change_task",
"sessions.add_session", "sessions.add_session",
"documents.change_taskattributes", "documents.change_taskattributes",
"documents.change_comment", "documents.change_note",
"django_q.delete_task", "django_q.delete_task",
"django_q.delete_ormq", "django_q.delete_ormq",
"auth.change_permission", "auth.change_permission",
@ -311,7 +311,7 @@
"documents.view_document", "documents.view_document",
"documents.add_savedview", "documents.add_savedview",
"django_q.view_failure", "django_q.view_failure",
"documents.view_comment", "documents.view_note",
"documents.view_log", "documents.view_log",
"documents.add_log", "documents.add_log",
"documents.change_savedview", "documents.change_savedview",
@ -324,7 +324,7 @@
"django_celery_results.view_taskresult", "django_celery_results.view_taskresult",
"contenttypes.add_contenttype", "contenttypes.add_contenttype",
"django_q.delete_success", "django_q.delete_success",
"documents.delete_comment", "documents.delete_note",
"django_q.add_failure", "django_q.add_failure",
"sessions.view_session", "sessions.view_session",
"contenttypes.view_contenttype", "contenttypes.view_contenttype",
@ -373,7 +373,7 @@
"django_celery_results.delete_taskresult", "django_celery_results.delete_taskresult",
"authtoken.change_token", "authtoken.change_token",
"auth.change_group", "auth.change_group",
"documents.add_comment", "documents.add_note",
"authtoken.delete_tokenproxy", "authtoken.delete_tokenproxy",
"documents.view_documenttype", "documents.view_documenttype",
"contenttypes.delete_contenttype", "contenttypes.delete_contenttype",
@ -402,7 +402,7 @@
"django_q.change_task", "django_q.change_task",
"sessions.add_session", "sessions.add_session",
"documents.change_taskattributes", "documents.change_taskattributes",
"documents.change_comment", "documents.change_note",
"django_q.delete_task", "django_q.delete_task",
"django_q.delete_ormq", "django_q.delete_ormq",
"auth.change_permission", "auth.change_permission",
@ -429,7 +429,7 @@
"documents.view_document", "documents.view_document",
"documents.add_savedview", "documents.add_savedview",
"django_q.view_failure", "django_q.view_failure",
"documents.view_comment", "documents.view_note",
"documents.view_log", "documents.view_log",
"auth.delete_user", "auth.delete_user",
"documents.add_log", "documents.add_log",
@ -443,7 +443,7 @@
"django_celery_results.view_taskresult", "django_celery_results.view_taskresult",
"contenttypes.add_contenttype", "contenttypes.add_contenttype",
"django_q.delete_success", "django_q.delete_success",
"documents.delete_comment", "documents.delete_note",
"django_q.add_failure", "django_q.add_failure",
"sessions.view_session", "sessions.view_session",
"contenttypes.view_contenttype", "contenttypes.view_contenttype",

View File

@ -492,7 +492,7 @@
</context-group> </context-group>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-list/document-list.component.ts</context> <context context-type="sourcefile">src/app/components/document-list/document-list.component.ts</context>
<context context-type="linenumber">94</context> <context context-type="linenumber">97</context>
</context-group> </context-group>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/manage/management-list/management-list.component.html</context> <context context-type="sourcefile">src/app/components/manage/management-list/management-list.component.html</context>
@ -1826,7 +1826,7 @@
<source>Not assigned</source> <source>Not assigned</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts</context> <context context-type="sourcefile">src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts</context>
<context context-type="linenumber">321</context> <context context-type="linenumber">335</context>
</context-group> </context-group>
<note priority="1" from="description">Filter drop down element to filter for documents with no correspondent/type/tag assigned</note> <note priority="1" from="description">Filter drop down element to filter for documents with no correspondent/type/tag assigned</note>
</trans-unit> </trans-unit>
@ -2328,52 +2328,6 @@
<context context-type="linenumber">1</context> <context context-type="linenumber">1</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="2122666445936087317" datatype="html">
<source>Enter comment</source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-comments/document-comments.component.html</context>
<context context-type="linenumber">4</context>
</context-group>
</trans-unit>
<trans-unit id="4025397324401332794" datatype="html">
<source> Please enter a comment. </source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-comments/document-comments.component.html</context>
<context context-type="linenumber">5,7</context>
</context-group>
</trans-unit>
<trans-unit id="2337485514607640701" datatype="html">
<source>Add comment</source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-comments/document-comments.component.html</context>
<context context-type="linenumber">11</context>
</context-group>
</trans-unit>
<trans-unit id="7978668497183230348" datatype="html">
<source>Delete comment</source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-comments/document-comments.component.html</context>
<context context-type="linenumber">21</context>
</context-group>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-comments/document-comments.component.html</context>
<context context-type="linenumber">25</context>
</context-group>
</trans-unit>
<trans-unit id="5438997040668245251" datatype="html">
<source>Error saving comment: <x id="PH" equiv-text="e.toString()"/></source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-comments/document-comments.component.ts</context>
<context context-type="linenumber">59</context>
</context-group>
</trans-unit>
<trans-unit id="7593210124183303626" datatype="html">
<source>Error deleting comment: <x id="PH" equiv-text="e.toString()"/></source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-comments/document-comments.component.ts</context>
<context context-type="linenumber">75</context>
</context-group>
</trans-unit>
<trans-unit id="1407560924967345762" datatype="html"> <trans-unit id="1407560924967345762" datatype="html">
<source>Page</source> <source>Page</source>
<context-group purpose="location"> <context-group purpose="location">
@ -2664,8 +2618,8 @@
<context context-type="linenumber">215</context> <context context-type="linenumber">215</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="8403302283555274795" datatype="html"> <trans-unit id="8460995830263484763" datatype="html">
<source>Comments <x id="START_TAG_SPAN" ctype="x-span" equiv-text="&lt;span *ngIf=&quot;document?.comments.length&quot; class=&quot;badge text-bg-secondary ms-1&quot;&gt;"/><x id="INTERPOLATION" equiv-text=".length}}"/><x id="CLOSE_TAG_SPAN" ctype="x-span" equiv-text="&lt;/a&gt;"/></source> <source>Notes <x id="START_TAG_SPAN" ctype="x-span" equiv-text="&lt;span *ngIf=&quot;document?.notes.length&quot; class=&quot;badge text-bg-secondary ms-1&quot;&gt;"/><x id="INTERPOLATION" equiv-text="ngth}}"/><x id="CLOSE_TAG_SPAN" ctype="x-span" equiv-text="&lt;/a&gt;"/></source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context>
<context context-type="linenumber">175,176</context> <context context-type="linenumber">175,176</context>
@ -3173,15 +3127,15 @@
<context context-type="linenumber">191</context> <context context-type="linenumber">191</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="3758078190163790058" datatype="html"> <trans-unit id="106713086593101376" datatype="html">
<source>View comments</source> <source>View notes</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-list/document-card-large/document-card-large.component.html</context> <context context-type="sourcefile">src/app/components/document-list/document-card-large/document-card-large.component.html</context>
<context context-type="linenumber">70</context> <context context-type="linenumber">70</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="6326693689225506833" datatype="html"> <trans-unit id="8778002102373462277" datatype="html">
<source><x id="INTERPOLATION" equiv-text="omments.length}}"/> Comments</source> <source><x id="INTERPOLATION" equiv-text="otes.length}}"/> Notes</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-list/document-card-large/document-card-large.component.html</context> <context context-type="sourcefile">src/app/components/document-list/document-card-large/document-card-large.component.html</context>
<context context-type="linenumber">74</context> <context context-type="linenumber">74</context>
@ -3373,8 +3327,8 @@
<context context-type="linenumber">18</context> <context context-type="linenumber">18</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="3807699453257291879" datatype="html"> <trans-unit id="8104421162933956065" datatype="html">
<source>Comments</source> <source>Notes</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-list/document-list.component.html</context> <context context-type="sourcefile">src/app/components/document-list/document-list.component.html</context>
<context context-type="linenumber">147</context> <context context-type="linenumber">147</context>
@ -3383,6 +3337,10 @@
<context context-type="sourcefile">src/app/components/manage/settings/settings.component.html</context> <context context-type="sourcefile">src/app/components/manage/settings/settings.component.html</context>
<context context-type="linenumber">159</context> <context context-type="linenumber">159</context>
</context-group> </context-group>
<context-group purpose="location">
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
<context context-type="linenumber">25</context>
</context-group>
</trans-unit> </trans-unit>
<trans-unit id="231679111972850796" datatype="html"> <trans-unit id="231679111972850796" datatype="html">
<source>Added</source> <source>Added</source>
@ -3410,14 +3368,14 @@
<source>View &quot;<x id="PH" equiv-text="this.list.activeSavedViewTitle"/>&quot; saved successfully.</source> <source>View &quot;<x id="PH" equiv-text="this.list.activeSavedViewTitle"/>&quot; saved successfully.</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-list/document-list.component.ts</context> <context context-type="sourcefile">src/app/components/document-list/document-list.component.ts</context>
<context context-type="linenumber">202</context> <context context-type="linenumber">205</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="6837554170707123455" datatype="html"> <trans-unit id="6837554170707123455" datatype="html">
<source>View &quot;<x id="PH" equiv-text="savedView.name"/>&quot; created successfully.</source> <source>View &quot;<x id="PH" equiv-text="savedView.name"/>&quot; created successfully.</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-list/document-list.component.ts</context> <context context-type="sourcefile">src/app/components/document-list/document-list.component.ts</context>
<context context-type="linenumber">243</context> <context context-type="linenumber">246</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="6849725902312323996" datatype="html"> <trans-unit id="6849725902312323996" datatype="html">
@ -3582,6 +3540,52 @@
<context context-type="linenumber">13</context> <context context-type="linenumber">13</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="1044349881182559852" datatype="html">
<source>Enter note</source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-notes/document-notes.component.html</context>
<context context-type="linenumber">4</context>
</context-group>
</trans-unit>
<trans-unit id="7770536883443596194" datatype="html">
<source> Please enter a note. </source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-notes/document-notes.component.html</context>
<context context-type="linenumber">5,7</context>
</context-group>
</trans-unit>
<trans-unit id="8433732438274024544" datatype="html">
<source>Add note</source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-notes/document-notes.component.html</context>
<context context-type="linenumber">11</context>
</context-group>
</trans-unit>
<trans-unit id="8428006099054244235" datatype="html">
<source>Delete note</source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-notes/document-notes.component.html</context>
<context context-type="linenumber">21</context>
</context-group>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-notes/document-notes.component.html</context>
<context context-type="linenumber">25</context>
</context-group>
</trans-unit>
<trans-unit id="207390237682956115" datatype="html">
<source>Error saving note: <x id="PH" equiv-text="e.toString()"/></source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-notes/document-notes.component.ts</context>
<context context-type="linenumber">65</context>
</context-group>
</trans-unit>
<trans-unit id="5682285129543775369" datatype="html">
<source>Error deleting note: <x id="PH" equiv-text="e.toString()"/></source>
<context-group purpose="location">
<context context-type="sourcefile">src/app/components/document-notes/document-notes.component.ts</context>
<context context-type="linenumber">81</context>
</context-group>
</trans-unit>
<trans-unit id="6316128875819022658" datatype="html"> <trans-unit id="6316128875819022658" datatype="html">
<source>correspondent</source> <source>correspondent</source>
<context-group purpose="location"> <context-group purpose="location">
@ -4050,8 +4054,8 @@
<context context-type="linenumber">155</context> <context context-type="linenumber">155</context>
</context-group> </context-group>
</trans-unit> </trans-unit>
<trans-unit id="4666858503087488647" datatype="html"> <trans-unit id="293524471897878391" datatype="html">
<source>Enable comments</source> <source>Enable notes</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/components/manage/settings/settings.component.html</context> <context context-type="sourcefile">src/app/components/manage/settings/settings.component.html</context>
<context context-type="linenumber">163</context> <context context-type="linenumber">163</context>
@ -4931,7 +4935,7 @@
<source>Search score</source> <source>Search score</source>
<context-group purpose="location"> <context-group purpose="location">
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context> <context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
<context context-type="linenumber">31</context> <context context-type="linenumber">32</context>
</context-group> </context-group>
<note priority="1" from="description">Score is a value returned by the full text search engine and specifies how well a result matches the given query</note> <note priority="1" from="description">Score is a value returned by the full text search engine and specifies how well a result matches the given query</note>
</trans-unit> </trans-unit>

View File

@ -70,7 +70,7 @@ import { ApiVersionInterceptor } from './interceptors/api-version.interceptor'
import { ColorSliderModule } from 'ngx-color/slider' import { ColorSliderModule } from 'ngx-color/slider'
import { ColorComponent } from './components/common/input/color/color.component' import { ColorComponent } from './components/common/input/color/color.component'
import { DocumentAsnComponent } from './components/document-asn/document-asn.component' import { DocumentAsnComponent } from './components/document-asn/document-asn.component'
import { DocumentCommentsComponent } from './components/document-comments/document-comments.component' import { DocumentNotesComponent } from './components/document-notes/document-notes.component'
import { PermissionsGuard } from './guards/permissions.guard' import { PermissionsGuard } from './guards/permissions.guard'
import { DirtyDocGuard } from './guards/dirty-doc.guard' import { DirtyDocGuard } from './guards/dirty-doc.guard'
import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard' import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard'
@ -196,7 +196,7 @@ function initializeApp(settings: SettingsService) {
DateComponent, DateComponent,
ColorComponent, ColorComponent,
DocumentAsnComponent, DocumentAsnComponent,
DocumentCommentsComponent, DocumentNotesComponent,
TasksComponent, TasksComponent,
UserEditDialogComponent, UserEditDialogComponent,
GroupEditDialogComponent, GroupEditDialogComponent,

View File

@ -1,99 +0,0 @@
import { Component, Input, Output, EventEmitter } from '@angular/core'
import { DocumentCommentsService } from 'src/app/services/rest/document-comments.service'
import { PaperlessDocumentComment } from 'src/app/data/paperless-document-comment'
import { FormControl, FormGroup } from '@angular/forms'
import { first } from 'rxjs/operators'
import { ToastService } from 'src/app/services/toast.service'
import { ComponentWithPermissions } from '../with-permissions/with-permissions.component'
@Component({
selector: 'app-document-comments',
templateUrl: './document-comments.component.html',
styleUrls: ['./document-comments.component.scss'],
})
export class DocumentCommentsComponent extends ComponentWithPermissions {
commentForm: FormGroup = new FormGroup({
newComment: new FormControl(''),
})
networkActive = false
newCommentError: boolean = false
@Input()
documentId: number
@Input()
comments: PaperlessDocumentComment[] = []
@Output()
updated: EventEmitter<PaperlessDocumentComment[]> = new EventEmitter()
constructor(
private commentsService: DocumentCommentsService,
private toastService: ToastService
) {
super()
}
addComment() {
const comment: string = this.commentForm
.get('newComment')
.value.toString()
.trim()
if (comment.length == 0) {
this.newCommentError = true
return
}
this.newCommentError = false
this.networkActive = true
this.commentsService.addComment(this.documentId, comment).subscribe({
next: (result) => {
this.comments = result
this.commentForm.get('newComment').reset()
this.networkActive = false
this.updated.emit(this.comments)
},
error: (e) => {
this.networkActive = false
this.toastService.showError(
$localize`Error saving comment: ${e.toString()}`
)
},
})
}
deleteComment(commentId: number) {
this.commentsService.deleteComment(this.documentId, commentId).subscribe({
next: (result) => {
this.comments = result
this.networkActive = false
this.updated.emit(this.comments)
},
error: (e) => {
this.networkActive = false
this.toastService.showError(
$localize`Error deleting comment: ${e.toString()}`
)
},
})
}
displayName(comment: PaperlessDocumentComment): string {
if (!comment.user) return ''
let nameComponents = []
if (comment.user.first_name) nameComponents.unshift(comment.user.first_name)
if (comment.user.last_name) nameComponents.unshift(comment.user.last_name)
if (comment.user.username) {
if (nameComponents.length > 0)
nameComponents.push(`(${comment.user.username})`)
else nameComponents.push(comment.user.username)
}
return nameComponents.join(' ')
}
commentFormKeydown(event: KeyboardEvent) {
if ((event.metaKey || event.ctrlKey) && event.key === 'Enter') {
this.addComment()
}
}
}

View File

@ -171,10 +171,10 @@
</ng-template> </ng-template>
</li> </li>
<li [ngbNavItem]="DocumentDetailNavIDs.Comments" *ngIf="commentsEnabled"> <li [ngbNavItem]="DocumentDetailNavIDs.Notes" *ngIf="notesEnabled">
<a ngbNavLink i18n>Comments <span *ngIf="document?.comments.length" class="badge text-bg-secondary ms-1">{{document.comments.length}}</span></a> <a ngbNavLink i18n>Notes <span *ngIf="document?.notes.length" class="badge text-bg-secondary ms-1">{{document.notes.length}}</span></a>
<ng-template ngbNavContent> <ng-template ngbNavContent>
<app-document-comments [documentId]="documentId" [comments]="document?.comments" (updated)="commentsUpdated($event)"></app-document-comments> <app-document-notes [documentId]="documentId" [notes]="document?.notes" (updated)="notesUpdated($event)"></app-document-notes>
</ng-template> </ng-template>
</li> </li>

View File

@ -42,14 +42,14 @@ import {
} from 'src/app/services/permissions.service' } from 'src/app/services/permissions.service'
import { PaperlessUser } from 'src/app/data/paperless-user' import { PaperlessUser } from 'src/app/data/paperless-user'
import { UserService } from 'src/app/services/rest/user.service' import { UserService } from 'src/app/services/rest/user.service'
import { PaperlessDocumentComment } from 'src/app/data/paperless-document-comment' import { PaperlessDocumentNote } from 'src/app/data/paperless-document-note'
enum DocumentDetailNavIDs { enum DocumentDetailNavIDs {
Details = 1, Details = 1,
Content = 2, Content = 2,
Metadata = 3, Metadata = 3,
Preview = 4, Preview = 4,
Comments = 5, Notes = 5,
Permissions = 6, Permissions = 6,
} }
@ -658,9 +658,9 @@ export class DocumentDetailComponent
} }
} }
get commentsEnabled(): boolean { get notesEnabled(): boolean {
return ( return (
this.settings.get(SETTINGS_KEYS.COMMENTS_ENABLED) && this.settings.get(SETTINGS_KEYS.NOTES_ENABLED) &&
this.permissionsService.currentUserCan( this.permissionsService.currentUserCan(
PermissionAction.View, PermissionAction.View,
PermissionType.Document PermissionType.Document
@ -668,8 +668,8 @@ export class DocumentDetailComponent
) )
} }
commentsUpdated(comments: PaperlessDocumentComment[]) { notesUpdated(notes: PaperlessDocumentNote[]) {
this.document.comments = comments this.document.notes = notes
this.openDocumentService.refreshDocument(this.documentId) this.openDocumentService.refreshDocument(this.documentId)
} }

View File

@ -26,7 +26,7 @@
</div> </div>
<p class="card-text"> <p class="card-text">
<span *ngIf="document.__search_hit__ && document.__search_hit__.highlights" [innerHtml]="document.__search_hit__.highlights"></span> <span *ngIf="document.__search_hit__ && document.__search_hit__.highlights" [innerHtml]="document.__search_hit__.highlights"></span>
<span *ngFor="let highlight of searchCommentHighlights" class="d-block"> <span *ngFor="let highlight of searchNoteHighlights" class="d-block">
<svg width="1em" height="1em" fill="currentColor" class="me-2"> <svg width="1em" height="1em" fill="currentColor" class="me-2">
<use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/> <use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/>
</svg> </svg>
@ -67,11 +67,11 @@
<div class="list-group list-group-horizontal border-0 card-info ms-md-auto mt-2 mt-md-0"> <div class="list-group list-group-horizontal border-0 card-info ms-md-auto mt-2 mt-md-0">
<button routerLink="/documents/{{document.id}}/comments" *ngIf="document.comments.length" class="list-group-item btn btn-sm bg-light text-dark p-1 border-0 me-2" title="View comments" i18n-title> <button *ngIf="notesEnabled && document.notes.length" routerLink="/documents/{{document.id}}/notes" class="list-group-item btn btn-sm bg-light text-dark p-1 border-0 me-2" title="View notes" i18n-title>
<svg class="metadata-icon me-2 text-muted" fill="currentColor"> <svg class="metadata-icon me-2 text-muted" fill="currentColor">
<use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/> <use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/>
</svg> </svg>
<small i18n>{{document.comments.length}} Comments</small> <small i18n>{{document.notes.length}} Notes</small>
</button> </button>
<button *ngIf="document.document_type" type="button" class="list-group-item btn btn-sm bg-light text-dark p-1 border-0 me-2" title="Filter by document type" i18n-title <button *ngIf="document.document_type" type="button" class="list-group-item btn btn-sm bg-light text-dark p-1 border-0 me-2" title="Filter by document type" i18n-title
(click)="clickDocumentType.emit(document.document_type);$event.stopPropagation()"> (click)="clickDocumentType.emit(document.document_type);$event.stopPropagation()">

View File

@ -73,16 +73,14 @@ export class DocumentCardLargeComponent extends ComponentWithPermissions {
} }
} }
get searchCommentHighlights() { get searchNoteHighlights() {
let highlights = [] let highlights = []
if ( if (
this.document['__search_hit__'] && this.document['__search_hit__'] &&
this.document['__search_hit__'].comment_highlights this.document['__search_hit__'].note_highlights
) { ) {
// only show comments with a match // only show notes with a match
highlights = ( highlights = (this.document['__search_hit__'].note_highlights as string)
this.document['__search_hit__'].comment_highlights as string
)
.split(',') .split(',')
.filter((higlight) => higlight.includes('<span')) .filter((higlight) => higlight.includes('<span'))
} }
@ -136,4 +134,8 @@ export class DocumentCardLargeComponent extends ComponentWithPermissions {
(this.document.content.length > 500 ? '...' : '') (this.document.content.length > 500 ? '...' : '')
) )
} }
get notesEnabled(): boolean {
return this.settingsService.get(SETTINGS_KEYS.NOTES_ENABLED)
}
} }

View File

@ -13,20 +13,20 @@
<div class="tags d-flex flex-column text-end position-absolute me-1 fs-6"> <div class="tags d-flex flex-column text-end position-absolute me-1 fs-6">
<app-tag *ngFor="let t of getTagsLimited$() | async" [tag]="t" (click)="clickTag.emit(t.id);$event.stopPropagation()" [clickable]="true" linkTitle="Toggle tag filter" i18n-linkTitle></app-tag> <app-tag *ngFor="let t of getTagsLimited$() | async" [tag]="t" (click)="clickTag.emit(t.id);$event.stopPropagation()" [clickable]="true" linkTitle="Toggle tag filter" i18n-linkTitle></app-tag>
<div *ngIf="moreTags"> <div *ngIf="moreTags">
<span class="badge badge-secondary">+ {{moreTags}}</span> <span class="badge text-dark">+ {{moreTags}}</span>
</div> </div>
</div> </div>
</div> </div>
<a routerLink="/documents/{{document.id}}/comments" *ngIf="document.comments.length" class="document-card-comments py-2 px-1"> <a *ngIf="notesEnabled && document.notes.length" routerLink="/documents/{{document.id}}/notes" class="document-card-notes py-2 px-1">
<span class="badge rounded-pill bg-light border text-primary"> <span class="badge rounded-pill bg-light border text-primary">
<svg class="metadata-icon ms-1 me-1" fill="currentColor"> <svg class="metadata-icon ms-1 me-1" fill="currentColor">
<use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/> <use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/>
</svg> </svg>
{{document.comments.length}}</span> {{document.notes.length}}</span>
</a> </a>
<div class="card-body p-2"> <div class="card-body bg-light p-2">
<p class="card-text"> <p class="card-text">
<ng-container *ngIf="document.correspondent"> <ng-container *ngIf="document.correspondent">
<a title="Toggle correspondent filter" i18n-title (click)="clickCorrespondent.emit(document.correspondent);$event.stopPropagation()" class="fw-bold btn-link">{{(document.correspondent$ | async)?.name}}</a>: <a title="Toggle correspondent filter" i18n-title (click)="clickCorrespondent.emit(document.correspondent);$event.stopPropagation()" class="fw-bold btn-link">{{(document.correspondent$ | async)?.name}}</a>:

View File

@ -34,10 +34,10 @@
display: block; display: block;
} }
.document-card-comments { .document-card-notes {
position: absolute; position: absolute;
right: 0; right: 0;
bottom: 170px; top: 142px;
} }
.card-selected { .card-selected {

View File

@ -74,7 +74,7 @@ export class DocumentCardSmallComponent extends ComponentWithPermissions {
} }
getTagsLimited$() { getTagsLimited$() {
const limit = this.document.comments.length > 0 ? 6 : 7 const limit = this.document.notes.length > 0 ? 6 : 7
return this.document.tags$.pipe( return this.document.tags$.pipe(
map((tags) => { map((tags) => {
if (tags.length > limit) { if (tags.length > limit) {
@ -111,4 +111,8 @@ export class DocumentCardSmallComponent extends ComponentWithPermissions {
mouseLeaveCard() { mouseLeaveCard() {
this.popover.close() this.popover.close()
} }
get notesEnabled(): boolean {
return this.settingsService.get(SETTINGS_KEYS.NOTES_ENABLED)
}
} }

View File

@ -139,12 +139,12 @@
[currentSortReverse]="list.sortReverse" [currentSortReverse]="list.sortReverse"
(sort)="onSort($event)" (sort)="onSort($event)"
i18n>Title</th> i18n>Title</th>
<th class="d-none d-xl-table-cell" <th *ngIf="notesEnabled" class="d-none d-xl-table-cell"
appSortable="num_comments" appSortable="num_notes"
[currentSortField]="list.sortField" [currentSortField]="list.sortField"
[currentSortReverse]="list.sortReverse" [currentSortReverse]="list.sortReverse"
(sort)="onSort($event)" (sort)="onSort($event)"
i18n>Comments</th> i18n>Notes</th>
<th class="d-none d-xl-table-cell" <th class="d-none d-xl-table-cell"
appSortable="document_type__name" appSortable="document_type__name"
[currentSortField]="list.sortField" [currentSortField]="list.sortField"
@ -190,13 +190,13 @@
<a routerLink="/documents/{{d.id}}" title="Edit document" i18n-title style="overflow-wrap: anywhere;">{{d.title | documentTitle}}</a> <a routerLink="/documents/{{d.id}}" title="Edit document" i18n-title style="overflow-wrap: anywhere;">{{d.title | documentTitle}}</a>
<app-tag [tag]="t" *ngFor="let t of d.tags$ | async" class="ms-1" clickable="true" linkTitle="Filter by tag" i18n-linkTitle (click)="clickTag(t.id);$event.stopPropagation()"></app-tag> <app-tag [tag]="t" *ngFor="let t of d.tags$ | async" class="ms-1" clickable="true" linkTitle="Filter by tag" i18n-linkTitle (click)="clickTag(t.id);$event.stopPropagation()"></app-tag>
</td> </td>
<td class="d-none d-xl-table-cell"> <td *ngIf="notesEnabled" class="d-none d-xl-table-cell">
<a routerLink="/documents/{{d.id}}/comments" *ngIf="d.comments.length" class="btn btn-sm p-0"> <a *ngIf="d.notes.length" routerLink="/documents/{{d.id}}/notes" class="btn btn-sm p-0">
<span class="badge rounded-pill bg-light border text-primary"> <span class="badge rounded-pill bg-light border text-primary">
<svg class="metadata-icon ms-1 me-1" fill="currentColor"> <svg class="metadata-icon ms-1 me-1" fill="currentColor">
<use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/> <use xlink:href="assets/bootstrap-icons.svg#chat-left-text"/>
</svg> </svg>
{{d.comments.length}}</span> {{d.notes.length}}</span>
</a> </a>
</td> </td>
<td class="d-none d-xl-table-cell"> <td class="d-none d-xl-table-cell">

View File

@ -17,6 +17,7 @@ import {
import { FILTER_FULLTEXT_MORELIKE } from 'src/app/data/filter-rule-type' import { FILTER_FULLTEXT_MORELIKE } from 'src/app/data/filter-rule-type'
import { PaperlessDocument } from 'src/app/data/paperless-document' import { PaperlessDocument } from 'src/app/data/paperless-document'
import { PaperlessSavedView } from 'src/app/data/paperless-saved-view' import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'
import { SETTINGS_KEYS } from 'src/app/data/paperless-uisettings'
import { import {
SortableDirective, SortableDirective,
SortEvent, SortEvent,
@ -29,6 +30,7 @@ import {
DOCUMENT_SORT_FIELDS_FULLTEXT, DOCUMENT_SORT_FIELDS_FULLTEXT,
} from 'src/app/services/rest/document.service' } from 'src/app/services/rest/document.service'
import { SavedViewService } from 'src/app/services/rest/saved-view.service' import { SavedViewService } from 'src/app/services/rest/saved-view.service'
import { SettingsService } from 'src/app/services/settings.service'
import { ToastService } from 'src/app/services/toast.service' import { ToastService } from 'src/app/services/toast.service'
import { ComponentWithPermissions } from '../with-permissions/with-permissions.component' import { ComponentWithPermissions } from '../with-permissions/with-permissions.component'
import { FilterEditorComponent } from './filter-editor/filter-editor.component' import { FilterEditorComponent } from './filter-editor/filter-editor.component'
@ -51,7 +53,8 @@ export class DocumentListComponent
private toastService: ToastService, private toastService: ToastService,
private modalService: NgbModal, private modalService: NgbModal,
private consumerStatusService: ConsumerStatusService, private consumerStatusService: ConsumerStatusService,
public openDocumentsService: OpenDocumentsService public openDocumentsService: OpenDocumentsService,
private settingsService: SettingsService
) { ) {
super() super()
} }
@ -289,4 +292,8 @@ export class DocumentListComponent
trackByDocumentId(index, item: PaperlessDocument) { trackByDocumentId(index, item: PaperlessDocument) {
return item.id return item.id
} }
get notesEnabled(): boolean {
return this.settingsService.get(SETTINGS_KEYS.NOTES_ENABLED)
}
} }

View File

@ -1,28 +1,28 @@
<div *ngIf="comments"> <div *ngIf="notes">
<form [formGroup]="commentForm" class="needs-validation mt-3" *appIfPermissions="{ action: PermissionAction.Add, type: PermissionType.Comment }" novalidate> <form [formGroup]="noteForm" class="needs-validation mt-3" *appIfPermissions="{ action: PermissionAction.Add, type: PermissionType.Note }" novalidate>
<div class="form-group"> <div class="form-group">
<textarea class="form-control form-control-sm" [class.is-invalid]="newCommentError" rows="3" formControlName="newComment" placeholder="Enter comment" i18n-placeholder (keydown)="commentFormKeydown($event)" required></textarea> <textarea class="form-control form-control-sm" [class.is-invalid]="newNoteError" rows="3" formControlName="newNote" placeholder="Enter note" i18n-placeholder (keydown)="noteFormKeydown($event)" required></textarea>
<div class="invalid-feedback" i18n> <div class="invalid-feedback" i18n>
Please enter a comment. Please enter a note.
</div> </div>
</div> </div>
<div class="form-group mt-2 d-flex justify-content-end align-items-center"> <div class="form-group mt-2 d-flex justify-content-end align-items-center">
<div *ngIf="networkActive" class="spinner-border spinner-border-sm fw-normal me-auto" role="status"></div> <div *ngIf="networkActive" class="spinner-border spinner-border-sm fw-normal me-auto" role="status"></div>
<button type="button" class="btn btn-primary btn-sm" [disabled]="networkActive" (click)="addComment()" i18n>Add comment</button> <button type="button" class="btn btn-primary btn-sm" [disabled]="networkActive" (click)="addNote()" i18n>Add note</button>
</div> </div>
</form> </form>
<hr> <hr>
<div *ngFor="let comment of comments" class="card border mb-3"> <div *ngFor="let note of notes" class="card border mb-3">
<div class="card-body text-dark"> <div class="card-body text-dark">
<p class="card-text">{{comment.comment}}</p> <p class="card-text">{{note.note}}</p>
</div> </div>
<div class="d-flex card-footer small bg-light text-primary justify-content-between align-items-center"> <div class="d-flex card-footer small bg-light text-primary justify-content-between align-items-center">
<span>{{displayName(comment)}} - {{ comment.created | customDate}}</span> <span>{{displayName(note)}} - {{ note.created | customDate}}</span>
<button type="button" class="btn btn-link btn-sm p-0 fade" title="Delete comment" i18n-title (click)="deleteComment(comment.id)" *appIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.Comment }"> <button type="button" class="btn btn-link btn-sm p-0 fade" title="Delete note" i18n-title (click)="deleteNote(note.id)" *appIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.Note }">
<svg width="13" height="13" fill="currentColor"> <svg width="13" height="13" fill="currentColor">
<use xlink:href="assets/bootstrap-icons.svg#trash" /> <use xlink:href="assets/bootstrap-icons.svg#trash" />
</svg> </svg>
<span class="visually-hidden" i18n>Delete comment</span> <span class="visually-hidden" i18n>Delete note</span>
</button> </button>
</div> </div>
</div> </div>

View File

@ -0,0 +1,106 @@
import { Component, Input, Output, EventEmitter } from '@angular/core'
import { DocumentNotesService } from 'src/app/services/rest/document-notes.service'
import { PaperlessDocumentNote } from 'src/app/data/paperless-document-note'
import { FormControl, FormGroup } from '@angular/forms'
import { first } from 'rxjs/operators'
import { ToastService } from 'src/app/services/toast.service'
import { ComponentWithPermissions } from '../with-permissions/with-permissions.component'
import { UserService } from 'src/app/services/rest/user.service'
import { PaperlessUser } from 'src/app/data/paperless-user'
@Component({
selector: 'app-document-notes',
templateUrl: './document-notes.component.html',
styleUrls: ['./document-notes.component.scss'],
})
export class DocumentNotesComponent extends ComponentWithPermissions {
noteForm: FormGroup = new FormGroup({
newNote: new FormControl(''),
})
networkActive = false
newNoteError: boolean = false
@Input()
documentId: number
@Input()
notes: PaperlessDocumentNote[] = []
@Output()
updated: EventEmitter<PaperlessDocumentNote[]> = new EventEmitter()
users: PaperlessUser[]
constructor(
private notesService: DocumentNotesService,
private toastService: ToastService,
private usersService: UserService
) {
super()
this.usersService.listAll().subscribe({
next: (users) => {
this.users = users.results
},
})
}
addNote() {
const note: string = this.noteForm.get('newNote').value.toString().trim()
if (note.length == 0) {
this.newNoteError = true
return
}
this.newNoteError = false
this.networkActive = true
this.notesService.addNote(this.documentId, note).subscribe({
next: (result) => {
this.notes = result
this.noteForm.get('newNote').reset()
this.networkActive = false
this.updated.emit(this.notes)
},
error: (e) => {
this.networkActive = false
this.toastService.showError(
$localize`Error saving note: ${e.toString()}`
)
},
})
}
deleteNote(noteId: number) {
this.notesService.deleteNote(this.documentId, noteId).subscribe({
next: (result) => {
this.notes = result
this.networkActive = false
this.updated.emit(this.notes)
},
error: (e) => {
this.networkActive = false
this.toastService.showError(
$localize`Error deleting note: ${e.toString()}`
)
},
})
}
displayName(note: PaperlessDocumentNote): string {
if (!note.user) return ''
const user = this.users.find((u) => u.id === note.user)
if (!user) return ''
const nameComponents = []
if (user.first_name) nameComponents.unshift(user.first_name)
if (user.last_name) nameComponents.unshift(user.last_name)
if (user.username) {
if (nameComponents.length > 0) nameComponents.push(`(${user.username})`)
else nameComponents.push(user.username)
}
return nameComponents.join(' ')
}
noteFormKeydown(event: KeyboardEvent) {
if ((event.metaKey || event.ctrlKey) && event.key === 'Enter') {
this.addNote()
}
}
}

View File

@ -156,11 +156,11 @@
</div> </div>
</div> </div>
<h4 class="mt-4" i18n>Comments</h4> <h4 class="mt-4" i18n>Notes</h4>
<div class="row mb-3"> <div class="row mb-3">
<div class="offset-md-3 col"> <div class="offset-md-3 col">
<app-input-check i18n-title title="Enable comments" formControlName="commentsEnabled"></app-input-check> <app-input-check i18n-title title="Enable notes" formControlName="notesEnabled"></app-input-check>
</div> </div>
</div> </div>

View File

@ -85,7 +85,7 @@ export class SettingsComponent
displayLanguage: new FormControl(null), displayLanguage: new FormControl(null),
dateLocale: new FormControl(null), dateLocale: new FormControl(null),
dateFormat: new FormControl(null), dateFormat: new FormControl(null),
commentsEnabled: new FormControl(null), notesEnabled: new FormControl(null),
updateCheckingEnabled: new FormControl(null), updateCheckingEnabled: new FormControl(null),
notificationsConsumerNewDocument: new FormControl(null), notificationsConsumerNewDocument: new FormControl(null),
@ -196,7 +196,7 @@ export class SettingsComponent
displayLanguage: this.settings.getLanguage(), displayLanguage: this.settings.getLanguage(),
dateLocale: this.settings.get(SETTINGS_KEYS.DATE_LOCALE), dateLocale: this.settings.get(SETTINGS_KEYS.DATE_LOCALE),
dateFormat: this.settings.get(SETTINGS_KEYS.DATE_FORMAT), dateFormat: this.settings.get(SETTINGS_KEYS.DATE_FORMAT),
commentsEnabled: this.settings.get(SETTINGS_KEYS.COMMENTS_ENABLED), notesEnabled: this.settings.get(SETTINGS_KEYS.NOTES_ENABLED),
updateCheckingEnabled: this.settings.get( updateCheckingEnabled: this.settings.get(
SETTINGS_KEYS.UPDATE_CHECKING_ENABLED SETTINGS_KEYS.UPDATE_CHECKING_ENABLED
), ),
@ -552,8 +552,8 @@ export class SettingsComponent
this.settingsForm.value.notificationsConsumerSuppressOnDashboard this.settingsForm.value.notificationsConsumerSuppressOnDashboard
) )
this.settings.set( this.settings.set(
SETTINGS_KEYS.COMMENTS_ENABLED, SETTINGS_KEYS.NOTES_ENABLED,
this.settingsForm.value.commentsEnabled this.settingsForm.value.notesEnabled
) )
this.settings.set( this.settings.set(
SETTINGS_KEYS.UPDATE_CHECKING_ENABLED, SETTINGS_KEYS.UPDATE_CHECKING_ENABLED,

View File

@ -1,8 +0,0 @@
import { ObjectWithId } from './object-with-id'
import { PaperlessUser } from './paperless-user'
export interface PaperlessDocumentComment extends ObjectWithId {
created?: Date
comment?: string
user?: PaperlessUser
}

View File

@ -0,0 +1,7 @@
import { ObjectWithId } from './object-with-id'
export interface PaperlessDocumentNote extends ObjectWithId {
created?: Date
note?: string
user?: number // PaperlessUser
}

View File

@ -4,14 +4,14 @@ import { PaperlessDocumentType } from './paperless-document-type'
import { Observable } from 'rxjs' import { Observable } from 'rxjs'
import { PaperlessStoragePath } from './paperless-storage-path' import { PaperlessStoragePath } from './paperless-storage-path'
import { ObjectWithPermissions } from './object-with-permissions' import { ObjectWithPermissions } from './object-with-permissions'
import { PaperlessDocumentComment } from './paperless-document-comment' import { PaperlessDocumentNote } from './paperless-document-note'
export interface SearchHit { export interface SearchHit {
score?: number score?: number
rank?: number rank?: number
highlights?: string highlights?: string
comment_highlights?: string note_highlights?: string
} }
export interface PaperlessDocument extends ObjectWithPermissions { export interface PaperlessDocument extends ObjectWithPermissions {
@ -55,7 +55,7 @@ export interface PaperlessDocument extends ObjectWithPermissions {
archive_serial_number?: number archive_serial_number?: number
comments?: PaperlessDocumentComment[] notes?: PaperlessDocumentNote[]
__search_hit__?: SearchHit __search_hit__?: SearchHit
} }

View File

@ -34,7 +34,7 @@ export const SETTINGS_KEYS = {
'general-settings:notifications:consumer-failed', 'general-settings:notifications:consumer-failed',
NOTIFICATIONS_CONSUMER_SUPPRESS_ON_DASHBOARD: NOTIFICATIONS_CONSUMER_SUPPRESS_ON_DASHBOARD:
'general-settings:notifications:consumer-suppress-on-dashboard', 'general-settings:notifications:consumer-suppress-on-dashboard',
COMMENTS_ENABLED: 'general-settings:comments-enabled', NOTES_ENABLED: 'general-settings:notes-enabled',
SLIM_SIDEBAR: 'general-settings:slim-sidebar', SLIM_SIDEBAR: 'general-settings:slim-sidebar',
UPDATE_CHECKING_ENABLED: 'general-settings:update-checking:enabled', UPDATE_CHECKING_ENABLED: 'general-settings:update-checking:enabled',
UPDATE_CHECKING_BACKEND_SETTING: UPDATE_CHECKING_BACKEND_SETTING:
@ -125,7 +125,7 @@ export const SETTINGS: PaperlessUiSetting[] = [
default: true, default: true,
}, },
{ {
key: SETTINGS_KEYS.COMMENTS_ENABLED, key: SETTINGS_KEYS.NOTES_ENABLED,
type: 'boolean', type: 'boolean',
default: true, default: true,
}, },

View File

@ -18,7 +18,7 @@ export enum PermissionType {
SavedView = '%s_savedview', SavedView = '%s_savedview',
PaperlessTask = '%s_paperlesstask', PaperlessTask = '%s_paperlesstask',
UISettings = '%s_uisettings', UISettings = '%s_uisettings',
Comment = '%s_comment', Note = '%s_note',
MailAccount = '%s_mailaccount', MailAccount = '%s_mailaccount',
MailRule = '%s_mailrule', MailRule = '%s_mailrule',
User = '%s_user', User = '%s_user',

View File

@ -1,37 +0,0 @@
import { Injectable } from '@angular/core'
import { HttpClient, HttpParams } from '@angular/common/http'
import { PaperlessDocumentComment } from 'src/app/data/paperless-document-comment'
import { AbstractPaperlessService } from './abstract-paperless-service'
import { Observable } from 'rxjs'
@Injectable({
providedIn: 'root',
})
export class DocumentCommentsService extends AbstractPaperlessService<PaperlessDocumentComment> {
constructor(http: HttpClient) {
super(http, 'documents')
}
getComments(documentId: number): Observable<PaperlessDocumentComment[]> {
return this.http.get<PaperlessDocumentComment[]>(
this.getResourceUrl(documentId, 'comments')
)
}
addComment(id: number, comment): Observable<PaperlessDocumentComment[]> {
return this.http.post<PaperlessDocumentComment[]>(
this.getResourceUrl(id, 'comments'),
{ comment: comment }
)
}
deleteComment(
documentId: number,
commentId: number
): Observable<PaperlessDocumentComment[]> {
return this.http.delete<PaperlessDocumentComment[]>(
this.getResourceUrl(documentId, 'comments'),
{ params: new HttpParams({ fromString: `id=${commentId}` }) }
)
}
}

View File

@ -0,0 +1,37 @@
import { Injectable } from '@angular/core'
import { HttpClient, HttpParams } from '@angular/common/http'
import { PaperlessDocumentNote } from 'src/app/data/paperless-document-note'
import { AbstractPaperlessService } from './abstract-paperless-service'
import { Observable } from 'rxjs'
@Injectable({
providedIn: 'root',
})
export class DocumentNotesService extends AbstractPaperlessService<PaperlessDocumentNote> {
constructor(http: HttpClient) {
super(http, 'documents')
}
getNotes(documentId: number): Observable<PaperlessDocumentNote[]> {
return this.http.get<PaperlessDocumentNote[]>(
this.getResourceUrl(documentId, 'notes')
)
}
addNote(id: number, note: string): Observable<PaperlessDocumentNote[]> {
return this.http.post<PaperlessDocumentNote[]>(
this.getResourceUrl(id, 'notes'),
{ note: note }
)
}
deleteNote(
documentId: number,
noteId: number
): Observable<PaperlessDocumentNote[]> {
return this.http.delete<PaperlessDocumentNote[]>(
this.getResourceUrl(documentId, 'notes'),
{ params: new HttpParams({ fromString: `id=${noteId}` }) }
)
}
}

View File

@ -22,6 +22,7 @@ export const DOCUMENT_SORT_FIELDS = [
{ field: 'created', name: $localize`Created` }, { field: 'created', name: $localize`Created` },
{ field: 'added', name: $localize`Added` }, { field: 'added', name: $localize`Added` },
{ field: 'modified', name: $localize`Modified` }, { field: 'modified', name: $localize`Modified` },
{ field: 'num_notes', name: $localize`Notes` },
] ]
export const DOCUMENT_SORT_FIELDS_FULLTEXT = [ export const DOCUMENT_SORT_FIELDS_FULLTEXT = [

View File

@ -137,6 +137,10 @@ $form-check-radio-checked-bg-image-dark: url("data:image/svg+xml,<svg xmlns='htt
border-color: rgba(0,0,0,0) !important; border-color: rgba(0,0,0,0) !important;
} }
.document-card .card-body.bg-light {
background-color: var(--bs-body-bg);
}
.doc-img { .doc-img {
mix-blend-mode: normal; mix-blend-mode: normal;
border-radius: 0; border-radius: 0;

View File

@ -4,6 +4,7 @@ from guardian.admin import GuardedModelAdmin
from .models import Correspondent from .models import Correspondent
from .models import Document from .models import Document
from .models import DocumentType from .models import DocumentType
from .models import Note
from .models import PaperlessTask from .models import PaperlessTask
from .models import SavedView from .models import SavedView
from .models import SavedViewFilterRule from .models import SavedViewFilterRule
@ -131,6 +132,13 @@ class TaskAdmin(admin.ModelAdmin):
) )
class NotesAdmin(GuardedModelAdmin):
list_display = ("user", "created", "note", "document")
list_filter = ("created", "user")
list_display_links = ("created",)
admin.site.register(Correspondent, CorrespondentAdmin) admin.site.register(Correspondent, CorrespondentAdmin)
admin.site.register(Tag, TagAdmin) admin.site.register(Tag, TagAdmin)
admin.site.register(DocumentType, DocumentTypeAdmin) admin.site.register(DocumentType, DocumentTypeAdmin)
@ -138,3 +146,4 @@ admin.site.register(Document, DocumentAdmin)
admin.site.register(SavedView, SavedViewAdmin) admin.site.register(SavedView, SavedViewAdmin)
admin.site.register(StoragePath, StoragePathAdmin) admin.site.register(StoragePath, StoragePathAdmin)
admin.site.register(PaperlessTask, TaskAdmin) admin.site.register(PaperlessTask, TaskAdmin)
admin.site.register(Note, NotesAdmin)

View File

@ -6,8 +6,8 @@ from contextlib import contextmanager
from dateutil.parser import isoparse from dateutil.parser import isoparse
from django.conf import settings from django.conf import settings
from django.utils import timezone from django.utils import timezone
from documents.models import Comment
from documents.models import Document from documents.models import Document
from documents.models import Note
from guardian.shortcuts import get_users_with_perms from guardian.shortcuts import get_users_with_perms
from whoosh import classify from whoosh import classify
from whoosh import highlight from whoosh import highlight
@ -52,7 +52,7 @@ def get_schema():
path=TEXT(sortable=True), path=TEXT(sortable=True),
path_id=NUMERIC(), path_id=NUMERIC(),
has_path=BOOLEAN(), has_path=BOOLEAN(),
comments=TEXT(), notes=TEXT(),
owner=TEXT(), owner=TEXT(),
owner_id=NUMERIC(), owner_id=NUMERIC(),
has_owner=BOOLEAN(), has_owner=BOOLEAN(),
@ -98,7 +98,7 @@ def open_index_searcher():
def update_document(writer: AsyncWriter, doc: Document): def update_document(writer: AsyncWriter, doc: Document):
tags = ",".join([t.name for t in doc.tags.all()]) tags = ",".join([t.name for t in doc.tags.all()])
tags_ids = ",".join([str(t.id) for t in doc.tags.all()]) tags_ids = ",".join([str(t.id) for t in doc.tags.all()])
comments = ",".join([str(c.comment) for c in Comment.objects.filter(document=doc)]) notes = ",".join([str(c.note) for c in Note.objects.filter(document=doc)])
asn = doc.archive_serial_number asn = doc.archive_serial_number
if asn is not None and ( if asn is not None and (
asn < Document.ARCHIVE_SERIAL_NUMBER_MIN asn < Document.ARCHIVE_SERIAL_NUMBER_MIN
@ -136,7 +136,7 @@ def update_document(writer: AsyncWriter, doc: Document):
path=doc.storage_path.name if doc.storage_path else None, path=doc.storage_path.name if doc.storage_path else None,
path_id=doc.storage_path.id if doc.storage_path else None, path_id=doc.storage_path.id if doc.storage_path else None,
has_path=doc.storage_path is not None, has_path=doc.storage_path is not None,
comments=comments, notes=notes,
owner=doc.owner.username if doc.owner else None, owner=doc.owner.username if doc.owner else None,
owner_id=doc.owner.id if doc.owner else None, owner_id=doc.owner.id if doc.owner else None,
has_owner=doc.owner is not None, has_owner=doc.owner is not None,
@ -293,7 +293,7 @@ class DelayedFullTextQuery(DelayedQuery):
def _get_query(self): def _get_query(self):
q_str = self.query_params["query"] q_str = self.query_params["query"]
qp = MultifieldParser( qp = MultifieldParser(
["content", "title", "correspondent", "tag", "type", "comments"], ["content", "title", "correspondent", "tag", "type", "notes"],
self.searcher.ixreader.schema, self.searcher.ixreader.schema,
) )
qp.add_plugin(DateParserPlugin(basedate=timezone.now())) qp.add_plugin(DateParserPlugin(basedate=timezone.now()))

View File

@ -17,10 +17,10 @@ from django.core.management.base import BaseCommand
from django.core.management.base import CommandError from django.core.management.base import CommandError
from django.db import transaction from django.db import transaction
from django.utils import timezone from django.utils import timezone
from documents.models import Comment
from documents.models import Correspondent from documents.models import Correspondent
from documents.models import Document from documents.models import Document
from documents.models import DocumentType from documents.models import DocumentType
from documents.models import Note
from documents.models import SavedView from documents.models import SavedView
from documents.models import SavedViewFilterRule from documents.models import SavedViewFilterRule
from documents.models import StoragePath from documents.models import StoragePath
@ -206,7 +206,7 @@ class Command(BaseCommand):
self.files_in_export_dir.add(x.resolve()) self.files_in_export_dir.add(x.resolve())
# 2. Create manifest, containing all correspondents, types, tags, storage paths # 2. Create manifest, containing all correspondents, types, tags, storage paths
# comments, documents and ui_settings # note, documents and ui_settings
with transaction.atomic(): with transaction.atomic():
manifest = json.loads( manifest = json.loads(
serializers.serialize("json", Correspondent.objects.all()), serializers.serialize("json", Correspondent.objects.all()),
@ -222,11 +222,11 @@ class Command(BaseCommand):
serializers.serialize("json", StoragePath.objects.all()), serializers.serialize("json", StoragePath.objects.all()),
) )
comments = json.loads( notes = json.loads(
serializers.serialize("json", Comment.objects.all()), serializers.serialize("json", Note.objects.all()),
) )
if not self.split_manifest: if not self.split_manifest:
manifest += comments manifest += notes
documents = Document.objects.order_by("id") documents = Document.objects.order_by("id")
document_map = {d.pk: d for d in documents} document_map = {d.pk: d for d in documents}
@ -359,7 +359,7 @@ class Command(BaseCommand):
content += list( content += list(
filter( filter(
lambda d: d["fields"]["document"] == document_dict["pk"], lambda d: d["fields"]["document"] == document_dict["pk"],
comments, notes,
), ),
) )
manifest_name.write_text(json.dumps(content, indent=2)) manifest_name.write_text(json.dumps(content, indent=2))

View File

@ -1,40 +0,0 @@
# Generated by Django 4.1.5 on 2023-03-17 21:23
from django.conf import settings
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
("documents", "1033_alter_documenttype_options_alter_tag_options_and_more"),
]
operations = [
migrations.AlterField(
model_name="comment",
name="document",
field=models.ForeignKey(
blank=True,
null=True,
on_delete=django.db.models.deletion.CASCADE,
related_name="comments",
to="documents.document",
verbose_name="document",
),
),
migrations.AlterField(
model_name="comment",
name="user",
field=models.ForeignKey(
blank=True,
null=True,
on_delete=django.db.models.deletion.SET_NULL,
related_name="comments",
to=settings.AUTH_USER_MODEL,
verbose_name="user",
),
),
]

View File

@ -0,0 +1,61 @@
# Generated by Django 4.1.5 on 2023-03-17 22:15
from django.conf import settings
from django.db import migrations
from django.db import models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
("documents", "1034_alter_savedviewfilterrule_rule_type"),
]
operations = [
migrations.RenameModel(
old_name="Comment",
new_name="Note",
),
migrations.RenameField(model_name="note", old_name="comment", new_name="note"),
migrations.AlterModelOptions(
name="note",
options={
"ordering": ("created",),
"verbose_name": "note",
"verbose_name_plural": "notes",
},
),
migrations.AlterField(
model_name="note",
name="document",
field=models.ForeignKey(
blank=True,
null=True,
on_delete=django.db.models.deletion.CASCADE,
related_name="notes",
to="documents.document",
verbose_name="document",
),
),
migrations.AlterField(
model_name="note",
name="note",
field=models.TextField(
blank=True, help_text="Note for the document", verbose_name="content"
),
),
migrations.AlterField(
model_name="note",
name="user",
field=models.ForeignKey(
blank=True,
null=True,
on_delete=django.db.models.deletion.SET_NULL,
related_name="notes",
to=settings.AUTH_USER_MODEL,
verbose_name="user",
),
),
]

View File

@ -635,11 +635,11 @@ class PaperlessTask(models.Model):
) )
class Comment(models.Model): class Note(models.Model):
comment = models.TextField( note = models.TextField(
_("content"), _("content"),
blank=True, blank=True,
help_text=_("Comment for the document"), help_text=_("Note for the document"),
) )
created = models.DateTimeField( created = models.DateTimeField(
@ -652,7 +652,7 @@ class Comment(models.Model):
Document, Document,
blank=True, blank=True,
null=True, null=True,
related_name="comments", related_name="notes",
on_delete=models.CASCADE, on_delete=models.CASCADE,
verbose_name=_("document"), verbose_name=_("document"),
) )
@ -661,15 +661,15 @@ class Comment(models.Model):
User, User,
blank=True, blank=True,
null=True, null=True,
related_name="comments", related_name="notes",
on_delete=models.SET_NULL, on_delete=models.SET_NULL,
verbose_name=_("user"), verbose_name=_("user"),
) )
class Meta: class Meta:
ordering = ("created",) ordering = ("created",)
verbose_name = _("comment") verbose_name = _("note")
verbose_name_plural = _("comments") verbose_name_plural = _("notes")
def __str__(self): def __str__(self):
return self.content return self.note

View File

@ -382,8 +382,6 @@ class DocumentSerializer(OwnedObjectSerializer, DynamicFieldsModelSerializer):
archived_file_name = SerializerMethodField() archived_file_name = SerializerMethodField()
created_date = serializers.DateField(required=False) created_date = serializers.DateField(required=False)
num_comments = serializers.IntegerField(read_only=True)
owner = serializers.PrimaryKeyRelatedField( owner = serializers.PrimaryKeyRelatedField(
queryset=User.objects.all(), queryset=User.objects.all(),
required=False, required=False,
@ -444,8 +442,7 @@ class DocumentSerializer(OwnedObjectSerializer, DynamicFieldsModelSerializer):
"owner", "owner",
"permissions", "permissions",
"set_permissions", "set_permissions",
"comments", "notes",
"num_comments",
) )

View File

@ -38,7 +38,7 @@ from documents.models import PaperlessTask
from documents.models import SavedView from documents.models import SavedView
from documents.models import StoragePath from documents.models import StoragePath
from documents.models import Tag from documents.models import Tag
from documents.models import Comment from documents.models import Note
from documents.tests.utils import DirectoriesMixin from documents.tests.utils import DirectoriesMixin
from paperless import version from paperless import version
from rest_framework.test import APITestCase from rest_framework.test import APITestCase
@ -1717,28 +1717,28 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
1, 1,
) )
def test_get_existing_comments(self): def test_get_existing_notes(self):
""" """
GIVEN: GIVEN:
- A document with a single comment - A document with a single note
WHEN: WHEN:
- API reuqest for document comments is made - API reuqest for document notes is made
THEN: THEN:
- The associated comment is returned - The associated note is returned
""" """
doc = Document.objects.create( doc = Document.objects.create(
title="test", title="test",
mime_type="application/pdf", mime_type="application/pdf",
content="this is a document which will have comments!", content="this is a document which will have notes!",
) )
comment = Comment.objects.create( note = Note.objects.create(
comment="This is a comment.", note="This is a note.",
document=doc, document=doc,
user=self.user, user=self.user,
) )
response = self.client.get( response = self.client.get(
f"/api/documents/{doc.pk}/comments/", f"/api/documents/{doc.pk}/notes/",
format="json", format="json",
) )
@ -1754,39 +1754,39 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
self.assertDictEqual( self.assertDictEqual(
resp_data, resp_data,
{ {
"id": comment.id, "id": note.id,
"comment": comment.comment, "note": note.note,
"user": { "user": {
"id": comment.user.id, "id": note.user.id,
"username": comment.user.username, "username": note.user.username,
"first_name": comment.user.first_name, "first_name": note.user.first_name,
"last_name": comment.user.last_name, "last_name": note.user.last_name,
}, },
}, },
) )
def test_create_comment(self): def test_create_note(self):
""" """
GIVEN: GIVEN:
- Existing document - Existing document
WHEN: WHEN:
- API request is made to add a comment - API request is made to add a note
THEN: THEN:
- Comment is created and associated with document - note is created and associated with document
""" """
doc = Document.objects.create( doc = Document.objects.create(
title="test", title="test",
mime_type="application/pdf", mime_type="application/pdf",
content="this is a document which will have comments added", content="this is a document which will have notes added",
) )
resp = self.client.post( resp = self.client.post(
f"/api/documents/{doc.pk}/comments/", f"/api/documents/{doc.pk}/notes/",
data={"comment": "this is a posted comment"}, data={"note": "this is a posted note"},
) )
self.assertEqual(resp.status_code, status.HTTP_200_OK) self.assertEqual(resp.status_code, status.HTTP_200_OK)
response = self.client.get( response = self.client.get(
f"/api/documents/{doc.pk}/comments/", f"/api/documents/{doc.pk}/notes/",
format="json", format="json",
) )
@ -1798,48 +1798,48 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
resp_data = resp_data[0] resp_data = resp_data[0]
self.assertEqual(resp_data["comment"], "this is a posted comment") self.assertEqual(resp_data["note"], "this is a posted note")
def test_delete_comment(self): def test_delete_note(self):
""" """
GIVEN: GIVEN:
- Existing document - Existing document
WHEN: WHEN:
- API request is made to add a comment - API request is made to add a note
THEN: THEN:
- Comment is created and associated with document - note is created and associated with document
""" """
doc = Document.objects.create( doc = Document.objects.create(
title="test", title="test",
mime_type="application/pdf", mime_type="application/pdf",
content="this is a document which will have comments!", content="this is a document which will have notes!",
) )
comment = Comment.objects.create( note = Note.objects.create(
comment="This is a comment.", note="This is a note.",
document=doc, document=doc,
user=self.user, user=self.user,
) )
response = self.client.delete( response = self.client.delete(
f"/api/documents/{doc.pk}/comments/?id={comment.pk}", f"/api/documents/{doc.pk}/notes/?id={note.pk}",
format="json", format="json",
) )
self.assertEqual(response.status_code, status.HTTP_200_OK) self.assertEqual(response.status_code, status.HTTP_200_OK)
self.assertEqual(len(Comment.objects.all()), 0) self.assertEqual(len(Note.objects.all()), 0)
def test_get_comments_no_doc(self): def test_get_notes_no_doc(self):
""" """
GIVEN: GIVEN:
- A request to get comments from a non-existent document - A request to get notes from a non-existent document
WHEN: WHEN:
- API request for document comments is made - API request for document notes is made
THEN: THEN:
- HTTP status.HTTP_404_NOT_FOUND is returned - HTTP status.HTTP_404_NOT_FOUND is returned
""" """
response = self.client.get( response = self.client.get(
"/api/documents/500/comments/", "/api/documents/500/notes/",
format="json", format="json",
) )
self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND) self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)

View File

@ -13,10 +13,10 @@ from django.test import override_settings
from django.test import TestCase from django.test import TestCase
from django.utils import timezone from django.utils import timezone
from documents.management.commands import document_exporter from documents.management.commands import document_exporter
from documents.models import Comment
from documents.models import Correspondent from documents.models import Correspondent
from documents.models import Document from documents.models import Document
from documents.models import DocumentType from documents.models import DocumentType
from documents.models import Note
from documents.models import StoragePath from documents.models import StoragePath
from documents.models import Tag from documents.models import Tag
from documents.models import User from documents.models import User
@ -66,8 +66,8 @@ class TestExportImport(DirectoriesMixin, FileSystemAssertsMixin, TestCase):
storage_type=Document.STORAGE_TYPE_GPG, storage_type=Document.STORAGE_TYPE_GPG,
) )
self.comment = Comment.objects.create( self.note = Note.objects.create(
comment="This is a comment. amaze.", note="This is a note. amaze.",
document=self.d1, document=self.d1,
user=self.user, user=self.user,
) )
@ -199,8 +199,8 @@ class TestExportImport(DirectoriesMixin, FileSystemAssertsMixin, TestCase):
checksum = hashlib.md5(f.read()).hexdigest() checksum = hashlib.md5(f.read()).hexdigest()
self.assertEqual(checksum, element["fields"]["archive_checksum"]) self.assertEqual(checksum, element["fields"]["archive_checksum"])
elif element["model"] == "documents.comment": elif element["model"] == "documents.note":
self.assertEqual(element["fields"]["comment"], self.comment.comment) self.assertEqual(element["fields"]["note"], self.note.note)
self.assertEqual(element["fields"]["document"], self.d1.id) self.assertEqual(element["fields"]["document"], self.d1.id)
self.assertEqual(element["fields"]["user"], self.user.id) self.assertEqual(element["fields"]["user"], self.user.id)

View File

@ -72,10 +72,10 @@ from .matching import match_correspondents
from .matching import match_document_types from .matching import match_document_types
from .matching import match_storage_paths from .matching import match_storage_paths
from .matching import match_tags from .matching import match_tags
from .models import Comment
from .models import Correspondent from .models import Correspondent
from .models import Document from .models import Document
from .models import DocumentType from .models import DocumentType
from .models import Note
from .models import PaperlessTask from .models import PaperlessTask
from .models import SavedView from .models import SavedView
from .models import StoragePath from .models import StoragePath
@ -230,7 +230,7 @@ class DocumentViewSet(
GenericViewSet, GenericViewSet,
): ):
model = Document model = Document
queryset = Document.objects.annotate(num_comments=Count("comments")) queryset = Document.objects.annotate(num_notes=Count("notes"))
serializer_class = DocumentSerializer serializer_class = DocumentSerializer
pagination_class = StandardPagination pagination_class = StandardPagination
permission_classes = (IsAuthenticated, PaperlessObjectPermissions) permission_classes = (IsAuthenticated, PaperlessObjectPermissions)
@ -251,11 +251,11 @@ class DocumentViewSet(
"modified", "modified",
"added", "added",
"archive_serial_number", "archive_serial_number",
"num_comments", "num_notes",
) )
def get_queryset(self): def get_queryset(self):
return Document.objects.distinct() return Document.objects.distinct().annotate(num_notes=Count("notes"))
def get_serializer(self, *args, **kwargs): def get_serializer(self, *args, **kwargs):
super().get_serializer(*args, **kwargs) super().get_serializer(*args, **kwargs)
@ -442,11 +442,11 @@ class DocumentViewSet(
except (FileNotFoundError, Document.DoesNotExist): except (FileNotFoundError, Document.DoesNotExist):
raise Http404() raise Http404()
def getComments(self, doc): def getNotes(self, doc):
return [ return [
{ {
"id": c.id, "id": c.id,
"comment": c.comment, "note": c.note,
"created": c.created, "created": c.created,
"user": { "user": {
"id": c.user.id, "id": c.user.id,
@ -455,11 +455,11 @@ class DocumentViewSet(
"last_name": c.user.last_name, "last_name": c.user.last_name,
}, },
} }
for c in Comment.objects.filter(document=doc).order_by("-created") for c in Note.objects.filter(document=doc).order_by("-created")
] ]
@action(methods=["get", "post", "delete"], detail=True) @action(methods=["get", "post", "delete"], detail=True)
def comments(self, request, pk=None): def notes(self, request, pk=None):
try: try:
doc = Document.objects.get(pk=pk) doc = Document.objects.get(pk=pk)
except Document.DoesNotExist: except Document.DoesNotExist:
@ -469,17 +469,17 @@ class DocumentViewSet(
if request.method == "GET": if request.method == "GET":
try: try:
return Response(self.getComments(doc)) return Response(self.getNotes(doc))
except Exception as e: except Exception as e:
logger.warning(f"An error occurred retrieving comments: {str(e)}") logger.warning(f"An error occurred retrieving notes: {str(e)}")
return Response( return Response(
{"error": "Error retreiving comments, check logs for more detail."}, {"error": "Error retreiving notes, check logs for more detail."},
) )
elif request.method == "POST": elif request.method == "POST":
try: try:
c = Comment.objects.create( c = Note.objects.create(
document=doc, document=doc,
comment=request.data["comment"], note=request.data["note"],
user=currentUser, user=currentUser,
) )
c.save() c.save()
@ -488,23 +488,23 @@ class DocumentViewSet(
index.add_or_update_document(self.get_object()) index.add_or_update_document(self.get_object())
return Response(self.getComments(doc)) return Response(self.getNotes(doc))
except Exception as e: except Exception as e:
logger.warning(f"An error occurred saving comment: {str(e)}") logger.warning(f"An error occurred saving note: {str(e)}")
return Response( return Response(
{ {
"error": "Error saving comment, check logs for more detail.", "error": "Error saving note, check logs for more detail.",
}, },
) )
elif request.method == "DELETE": elif request.method == "DELETE":
comment = Comment.objects.get(id=int(request.GET.get("id"))) note = Note.objects.get(id=int(request.GET.get("id")))
comment.delete() note.delete()
from documents import index from documents import index
index.add_or_update_document(self.get_object()) index.add_or_update_document(self.get_object())
return Response(self.getComments(doc)) return Response(self.getNotes(doc))
return Response( return Response(
{ {
@ -516,14 +516,14 @@ class DocumentViewSet(
class SearchResultSerializer(DocumentSerializer, PassUserMixin): class SearchResultSerializer(DocumentSerializer, PassUserMixin):
def to_representation(self, instance): def to_representation(self, instance):
doc = Document.objects.get(id=instance["id"]) doc = Document.objects.get(id=instance["id"])
comments = ",".join( notes = ",".join(
[str(c.comment) for c in Comment.objects.filter(document=instance["id"])], [str(c.note) for c in Note.objects.filter(document=instance["id"])],
) )
r = super().to_representation(doc) r = super().to_representation(doc)
r["__search_hit__"] = { r["__search_hit__"] = {
"score": instance.score, "score": instance.score,
"highlights": instance.highlights("content", text=doc.content), "highlights": instance.highlights("content", text=doc.content),
"comment_highlights": instance.highlights("comments", text=comments) "note_highlights": instance.highlights("notes", text=notes)
if doc if doc
else None, else None,
"rank": instance.rank, "rank": instance.rank,