mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-30 03:56:23 -05:00 
			
		
		
		
	Feature: email document button (#8950)
This commit is contained in:
		| @@ -385,7 +385,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">100</context> | ||||
|           <context context-type="linenumber">117</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="1241348629231510663" datatype="html"> | ||||
| @@ -534,7 +534,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">353</context> | ||||
|           <context context-type="linenumber">370</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3768927257183755959" datatype="html"> | ||||
| @@ -593,7 +593,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">346</context> | ||||
|           <context context-type="linenumber">363</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/custom-fields-bulk-edit-dialog/custom-fields-bulk-edit-dialog.component.html</context> | ||||
| @@ -739,7 +739,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">366</context> | ||||
|           <context context-type="linenumber">383</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/document-list.component.html</context> | ||||
| @@ -1190,7 +1190,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">322</context> | ||||
|           <context context-type="linenumber">339</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.html</context> | ||||
| @@ -2077,8 +2077,8 @@ | ||||
|           <context context-type="linenumber">19</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">37</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">36</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
| @@ -3391,7 +3391,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">94</context> | ||||
|           <context context-type="linenumber">111</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
| @@ -4288,7 +4288,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">288</context> | ||||
|           <context context-type="linenumber">305</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8057014866157903311" datatype="html"> | ||||
| @@ -4390,6 +4390,10 @@ | ||||
|           <context context-type="sourcefile">src/app/components/common/profile-edit-dialog/profile-edit-dialog.component.html</context> | ||||
|           <context context-type="linenumber">10</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">96</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5342432350421167093" datatype="html"> | ||||
|         <source>First name</source> | ||||
| @@ -5061,6 +5065,62 @@ | ||||
|           <context context-type="linenumber">229</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="7376342558017986274" datatype="html"> | ||||
|         <source>Email address(es)</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.html</context> | ||||
|           <context context-type="linenumber">7</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="9127604588498960753" datatype="html"> | ||||
|         <source>Subject</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.html</context> | ||||
|           <context context-type="linenumber">11</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8066608938393600549" datatype="html"> | ||||
|         <source>Message</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.html</context> | ||||
|           <context context-type="linenumber">15</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5867799091834207531" datatype="html"> | ||||
|         <source>Use archive version</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.html</context> | ||||
|           <context context-type="linenumber">23</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4312183290449350804" datatype="html"> | ||||
|         <source>Send email</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.html</context> | ||||
|           <context context-type="linenumber">29</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="1342170399958833675" datatype="html"> | ||||
|         <source>Email Document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">17</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="9049148856403142491" datatype="html"> | ||||
|         <source>Email sent</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">65</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3742745894977668908" datatype="html"> | ||||
|         <source>Error emailing document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/email-document-dialog/email-document-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">69</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6381578200008167206" datatype="html"> | ||||
|         <source>Include</source> | ||||
|         <context-group purpose="location"> | ||||
| @@ -5082,8 +5142,8 @@ | ||||
|           <context context-type="linenumber">58</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">64</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">65</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/manage/management-list/management-list.component.html</context> | ||||
| @@ -5543,8 +5603,8 @@ | ||||
|           <context context-type="linenumber">155</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">29</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">28</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context> | ||||
| @@ -5585,8 +5645,8 @@ | ||||
|           <context context-type="linenumber">162</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">40</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">39</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4369881772624105142" datatype="html"> | ||||
| @@ -5765,103 +5825,103 @@ | ||||
|           <context context-type="linenumber">320</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="686374493515618129" datatype="html"> | ||||
|         <source>Share Links</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">4</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">32</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6617773613987957957" datatype="html"> | ||||
|         <source> No existing links </source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">9,11</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">8,10</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="7419704019640008953" datatype="html"> | ||||
|         <source>Share</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">33</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">32</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6811921365829755679" datatype="html"> | ||||
|         <source>Share archive version</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">47</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">48</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8037476586059399916" datatype="html"> | ||||
|         <source>Expires</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.html</context> | ||||
|           <context context-type="linenumber">51</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.html</context> | ||||
|           <context context-type="linenumber">52</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4776429682428363094" datatype="html"> | ||||
|         <source>1 day</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">25</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">20</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">111</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">104</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8542568275115626925" datatype="html"> | ||||
|         <source>7 days</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">26</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">21</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="7152095234138763013" datatype="html"> | ||||
|         <source>30 days</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">27</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">22</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8372007266188249803" datatype="html"> | ||||
|         <source>Never</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">28</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">23</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="686374493515618129" datatype="html"> | ||||
|         <source>Share Links</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">27</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">92</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3429210839568770054" datatype="html"> | ||||
|         <source>Error retrieving links</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">92</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">85</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3242255798983858463" datatype="html"> | ||||
|         <source><x id="PH" equiv-text="days"/> days</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">111</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">104</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2897042887615940599" datatype="html"> | ||||
|         <source>Error deleting link</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">140</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">133</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8400747326190565173" datatype="html"> | ||||
|         <source>Error creating link</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts</context> | ||||
|           <context context-type="linenumber">168</context> | ||||
|           <context context-type="sourcefile">src/app/components/common/share-links-dialog/share-links-dialog.component.ts</context> | ||||
|           <context context-type="linenumber">161</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="9180110319941008393" datatype="html"> | ||||
| @@ -6379,25 +6439,32 @@ | ||||
|           <context context-type="linenumber">70</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6490688569532630280" datatype="html"> | ||||
|         <source>Send</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">88</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4452427314943113135" datatype="html"> | ||||
|         <source>Previous</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">97</context> | ||||
|           <context context-type="linenumber">114</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5028777105388019087" datatype="html"> | ||||
|         <source>Details</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">110</context> | ||||
|           <context context-type="linenumber">127</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5701618810648052610" datatype="html"> | ||||
|         <source>Title</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">113</context> | ||||
|           <context context-type="linenumber">130</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/document-list.component.html</context> | ||||
| @@ -6420,21 +6487,21 @@ | ||||
|         <source>Archive serial number</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">114</context> | ||||
|           <context context-type="linenumber">131</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5114742157723900905" datatype="html"> | ||||
|         <source>Date created</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">115</context> | ||||
|           <context context-type="linenumber">132</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2691296884221415710" datatype="html"> | ||||
|         <source>Correspondent</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">117</context> | ||||
|           <context context-type="linenumber">134</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.html</context> | ||||
| @@ -6461,7 +6528,7 @@ | ||||
|         <source>Document type</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">119</context> | ||||
|           <context context-type="linenumber">136</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.html</context> | ||||
| @@ -6488,7 +6555,7 @@ | ||||
|         <source>Storage path</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">121</context> | ||||
|           <context context-type="linenumber">138</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.html</context> | ||||
| @@ -6511,7 +6578,7 @@ | ||||
|         <source>Default</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">122</context> | ||||
|           <context context-type="linenumber">139</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/manage/saved-views/saved-views.component.html</context> | ||||
| @@ -6522,14 +6589,14 @@ | ||||
|         <source>Content</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">218</context> | ||||
|           <context context-type="linenumber">235</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="218403386307979629" datatype="html"> | ||||
|         <source>Metadata</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">227</context> | ||||
|           <context context-type="linenumber">244</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts</context> | ||||
| @@ -6540,119 +6607,119 @@ | ||||
|         <source>Date modified</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">234</context> | ||||
|           <context context-type="linenumber">251</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6392918669949841614" datatype="html"> | ||||
|         <source>Date added</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">238</context> | ||||
|           <context context-type="linenumber">255</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="146828917013192897" datatype="html"> | ||||
|         <source>Media filename</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">242</context> | ||||
|           <context context-type="linenumber">259</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4500855521601039868" datatype="html"> | ||||
|         <source>Original filename</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">246</context> | ||||
|           <context context-type="linenumber">263</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="7985558498848210210" datatype="html"> | ||||
|         <source>Original MD5 checksum</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">250</context> | ||||
|           <context context-type="linenumber">267</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5888243105821763422" datatype="html"> | ||||
|         <source>Original file size</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">254</context> | ||||
|           <context context-type="linenumber">271</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2696647325713149563" datatype="html"> | ||||
|         <source>Original mime type</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">258</context> | ||||
|           <context context-type="linenumber">275</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="342875990758166588" datatype="html"> | ||||
|         <source>Archive MD5 checksum</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">263</context> | ||||
|           <context context-type="linenumber">280</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6033581412811562084" datatype="html"> | ||||
|         <source>Archive file size</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">269</context> | ||||
|           <context context-type="linenumber">286</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6992781481378431874" datatype="html"> | ||||
|         <source>Original document metadata</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">278</context> | ||||
|           <context context-type="linenumber">295</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2846565152091361585" datatype="html"> | ||||
|         <source>Archived document metadata</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">281</context> | ||||
|           <context context-type="linenumber">298</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="7206723502037428235" datatype="html"> | ||||
|         <source>Notes <x id="START_BLOCK_IF" equiv-text="@if (document?.notes.length) {"/><x id="START_TAG_SPAN" ctype="x-span" equiv-text="<span class="badge text-bg-secondary ms-1">"/><x id="INTERPOLATION" equiv-text="ngth}}"/><x id="CLOSE_TAG_SPAN" ctype="x-span"/><x id="CLOSE_BLOCK_IF" equiv-text="}"/></source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">300,303</context> | ||||
|           <context context-type="linenumber">317,320</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="186236568870281953" datatype="html"> | ||||
|         <source>History</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">311</context> | ||||
|           <context context-type="linenumber">328</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5129524307369213584" datatype="html"> | ||||
|         <source>Save & next</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">348</context> | ||||
|           <context context-type="linenumber">365</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4910102545766233758" datatype="html"> | ||||
|         <source>Save & close</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">351</context> | ||||
|           <context context-type="linenumber">368</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="1309556917227148591" datatype="html"> | ||||
|         <source>Document loading...</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">361</context> | ||||
|           <context context-type="linenumber">378</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8191371354890763172" datatype="html"> | ||||
|         <source>Enter Password</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context> | ||||
|           <context context-type="linenumber">415</context> | ||||
|           <context context-type="linenumber">432</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2218903673684131427" datatype="html"> | ||||
| @@ -6949,11 +7016,11 @@ | ||||
|         <source>An error occurred loading tiff: <x id="PH" equiv-text="err.toString()"/></source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1461</context> | ||||
|           <context context-type="linenumber">1481</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1465</context> | ||||
|           <context context-type="linenumber">1485</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4958946940233632319" datatype="html"> | ||||
|   | ||||
| @@ -0,0 +1,32 @@ | ||||
| <div class="modal-header"> | ||||
|     <h4 class="modal-title" id="modal-basic-title">{{title}}</h4> | ||||
|     <button type="button" class="btn-close" aria-label="Close" (click)="close()"></button> | ||||
| </div> | ||||
| <div class="modal-body"> | ||||
|     <div class="mb-1"> | ||||
|         <label for="email" class="form-label" i18n>Email address(es)</label> | ||||
|         <input type="email" class="form-control" id="email" [(ngModel)]="emailAddress"> | ||||
|     </div> | ||||
|     <div class="mb-1"> | ||||
|         <label for="email" class="form-label" i18n>Subject</label> | ||||
|         <input type="email" class="form-control" id="subject" [(ngModel)]="emailSubject"> | ||||
|     </div> | ||||
|     <div> | ||||
|         <label for="message" class="form-label" i18n>Message</label> | ||||
|         <textarea class="form-control" id="message" rows="3" [(ngModel)]="emailMessage"></textarea> | ||||
|     </div> | ||||
| </div> | ||||
| <div class="modal-footer"> | ||||
|     <div class="input-group"> | ||||
|         <div class="input-group-text flex-grow-1"> | ||||
|             <input class="form-check-input mt-0 me-2" type="checkbox" role="switch" id="useArchiveVersion" [disabled]="!hasArchiveVersion" [(ngModel)]="useArchiveVersion"> | ||||
|             <label class="form-check-label w-100 text-start" for="useArchiveVersion" i18n>Use archive version</label> | ||||
|         </div> | ||||
|         <button type="submit" class="btn btn-outline-primary" (click)="emailDocument()" [disabled]="loading || emailAddress.length === 0 || emailMessage.length === 0 || emailSubject.length === 0"> | ||||
|             @if (loading) { | ||||
|                 <div class="spinner-border spinner-border-sm me-2" role="status"></div> | ||||
|             } | ||||
|             <ng-container i18n>Send email</ng-container> | ||||
|         </button> | ||||
|     </div> | ||||
| </div> | ||||
| @@ -0,0 +1,72 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing' | ||||
|  | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { provideHttpClientTesting } from '@angular/common/http/testing' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { of, throwError } from 'rxjs' | ||||
| import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive' | ||||
| import { PermissionsService } from 'src/app/services/permissions.service' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { EmailDocumentDialogComponent } from './email-document-dialog.component' | ||||
|  | ||||
| describe('EmailDocumentDialogComponent', () => { | ||||
|   let component: EmailDocumentDialogComponent | ||||
|   let fixture: ComponentFixture<EmailDocumentDialogComponent> | ||||
|   let documentService: DocumentService | ||||
|   let permissionsService: PermissionsService | ||||
|   let toastService: ToastService | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       imports: [ | ||||
|         EmailDocumentDialogComponent, | ||||
|         IfPermissionsDirective, | ||||
|         NgxBootstrapIconsModule.pick(allIcons), | ||||
|       ], | ||||
|       providers: [ | ||||
|         provideHttpClient(withInterceptorsFromDi()), | ||||
|         provideHttpClientTesting(), | ||||
|         NgbActiveModal, | ||||
|       ], | ||||
|     }).compileComponents() | ||||
|  | ||||
|     fixture = TestBed.createComponent(EmailDocumentDialogComponent) | ||||
|     documentService = TestBed.inject(DocumentService) | ||||
|     toastService = TestBed.inject(ToastService) | ||||
|     component = fixture.componentInstance | ||||
|     fixture.detectChanges() | ||||
|   }) | ||||
|  | ||||
|   it('should set hasArchiveVersion and useArchiveVersion', () => { | ||||
|     expect(component.hasArchiveVersion).toBeTruthy() | ||||
|     component.hasArchiveVersion = false | ||||
|     expect(component.hasArchiveVersion).toBeFalsy() | ||||
|     expect(component.useArchiveVersion).toBeFalsy() | ||||
|   }) | ||||
|  | ||||
|   it('should support sending document via email, showing error if needed', () => { | ||||
|     const toastErrorSpy = jest.spyOn(toastService, 'showError') | ||||
|     const toastSuccessSpy = jest.spyOn(toastService, 'showInfo') | ||||
|     component.emailAddress = 'hello@paperless-ngx.com' | ||||
|     component.emailSubject = 'Hello' | ||||
|     component.emailMessage = 'World' | ||||
|     jest | ||||
|       .spyOn(documentService, 'emailDocument') | ||||
|       .mockReturnValue(throwError(() => new Error('Unable to email document'))) | ||||
|     component.emailDocument() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|  | ||||
|     jest.spyOn(documentService, 'emailDocument').mockReturnValue(of(true)) | ||||
|     component.emailDocument() | ||||
|     expect(toastSuccessSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
|   it('should close the dialog', () => { | ||||
|     const activeModal = TestBed.inject(NgbActiveModal) | ||||
|     const closeSpy = jest.spyOn(activeModal, 'close') | ||||
|     component.close() | ||||
|     expect(closeSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
| @@ -0,0 +1,77 @@ | ||||
| import { Component, Input } from '@angular/core' | ||||
| import { FormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { LoadingComponentWithPermissions } from '../../loading-component/loading.component' | ||||
|  | ||||
| @Component({ | ||||
|   selector: 'pngx-email-document-dialog', | ||||
|   templateUrl: './email-document-dialog.component.html', | ||||
|   styleUrl: './email-document-dialog.component.scss', | ||||
|   imports: [FormsModule, NgxBootstrapIconsModule], | ||||
| }) | ||||
| export class EmailDocumentDialogComponent extends LoadingComponentWithPermissions { | ||||
|   @Input() | ||||
|   title = $localize`Email Document` | ||||
|  | ||||
|   @Input() | ||||
|   documentId: number | ||||
|  | ||||
|   private _hasArchiveVersion: boolean = true | ||||
|  | ||||
|   @Input() | ||||
|   set hasArchiveVersion(value: boolean) { | ||||
|     this._hasArchiveVersion = value | ||||
|     this.useArchiveVersion = value | ||||
|   } | ||||
|  | ||||
|   get hasArchiveVersion(): boolean { | ||||
|     return this._hasArchiveVersion | ||||
|   } | ||||
|  | ||||
|   public useArchiveVersion: boolean = true | ||||
|  | ||||
|   public emailAddress: string = '' | ||||
|   public emailSubject: string = '' | ||||
|   public emailMessage: string = '' | ||||
|  | ||||
|   constructor( | ||||
|     private activeModal: NgbActiveModal, | ||||
|     private documentService: DocumentService, | ||||
|     private toastService: ToastService | ||||
|   ) { | ||||
|     super() | ||||
|     this.loading = false | ||||
|   } | ||||
|  | ||||
|   public emailDocument() { | ||||
|     this.loading = true | ||||
|     this.documentService | ||||
|       .emailDocument( | ||||
|         this.documentId, | ||||
|         this.emailAddress, | ||||
|         this.emailSubject, | ||||
|         this.emailMessage, | ||||
|         this.useArchiveVersion | ||||
|       ) | ||||
|       .subscribe({ | ||||
|         next: () => { | ||||
|           this.loading = false | ||||
|           this.emailAddress = '' | ||||
|           this.emailSubject = '' | ||||
|           this.emailMessage = '' | ||||
|           this.toastService.showInfo($localize`Email sent`) | ||||
|         }, | ||||
|         error: (e) => { | ||||
|           this.loading = false | ||||
|           this.toastService.showError($localize`Error emailing document`, e) | ||||
|         }, | ||||
|       }) | ||||
|   } | ||||
|  | ||||
|   public close() { | ||||
|     this.activeModal.close() | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,68 @@ | ||||
| <div class="modal-header"> | ||||
|   <h4 class="modal-title" id="modal-basic-title">{{title}}</h4> | ||||
|   <button type="button" class="btn-close" aria-label="Close" (click)="close()"></button> | ||||
| </div> | ||||
| <div class="modal-body p-0"> | ||||
|   <ul class="list-group list-group-flush"> | ||||
|     @if (!shareLinks || shareLinks.length === 0) { | ||||
|       <li class="list-group-item fst-italic small text-center text-secondary" i18n> | ||||
|         No existing links | ||||
|       </li> | ||||
|     } | ||||
|     @for (link of shareLinks; track link) { | ||||
|       <li class="list-group-item"> | ||||
|         <div class="input-group w-100"> | ||||
|           <input type="text" class="form-control" aria-label="Share link" [value]="getShareUrl(link)" readonly> | ||||
|           @if (link.expiration) { | ||||
|             <span class="input-group-text"> | ||||
|               {{ getDaysRemaining(link) }} | ||||
|             </span> | ||||
|           } | ||||
|           <button type="button" class="btn btn-outline-primary" (click)="copy(link)"> | ||||
|               @if (copied !== link.id) { | ||||
|                 <i-bs width="1.2em" height="1.2em" name="clipboard-fill"></i-bs> | ||||
|               } | ||||
|               @if (copied === link.id) { | ||||
|                 <i-bs width="1.2em" height="1.2em" name="clipboard-check-fill"></i-bs> | ||||
|               } | ||||
|               <span class="visually-hidden" i18n>Copy</span> | ||||
|           </button> | ||||
|           @if (canShare(link)) { | ||||
|             <button type="button" class="btn btn-outline-primary" (click)="share(link)"> | ||||
|               <i-bs width="1.2em" height="1.2em" name="box-arrow-up"></i-bs><span class="visually-hidden" i18n>Share</span> | ||||
|             </button> | ||||
|           } | ||||
|           <button type="button" class="btn btn-outline-danger" (click)="delete(link)"> | ||||
|             <i-bs width="1.2em" height="1.2em" name="trash"></i-bs><span class="visually-hidden" i18n>Delete</span> | ||||
|           </button> | ||||
|         </div> | ||||
|         <span class="badge copied-badge bg-primary small fade ms-4 position-absolute top-50 translate-middle-y pe-none z-3" [class.show]="copied === link.id" i18n>Copied!</span> | ||||
|       </li> | ||||
|     } | ||||
|   </ul> | ||||
| </div> | ||||
| <div class="modal-footer"> | ||||
|   <div class="input-group w-100"> | ||||
|     <div class="form-check form-switch ms-auto"> | ||||
|       <input class="form-check-input" type="checkbox" role="switch" id="versionSwitch" [disabled]="!hasArchiveVersion" [(ngModel)]="useArchiveVersion"> | ||||
|       <label class="form-check-label" for="versionSwitch" i18n>Share archive version</label> | ||||
|     </div> | ||||
|   </div> | ||||
|   <div class="input-group w-100 mt-2"> | ||||
|     <label class="input-group-text" for="addLink"><ng-container i18n>Expires</ng-container>:</label> | ||||
|     <select class="form-select fs-6" [(ngModel)]="expirationDays"> | ||||
|       @for (option of EXPIRATION_OPTIONS; track option) { | ||||
|         <option [ngValue]="option.value">{{ option.label }}</option> | ||||
|       } | ||||
|     </select> | ||||
|     <button class="btn btn-outline-primary ms-auto" type="button" (click)="createLink()" [disabled]="loading"> | ||||
|       @if (loading) { | ||||
|         <div class="spinner-border spinner-border-sm me-2" role="status"></div> | ||||
|       } | ||||
|       @if (!loading) { | ||||
|         <i-bs name="plus"></i-bs> | ||||
|       } | ||||
|       <ng-container i18n>Create</ng-container> | ||||
|     </button> | ||||
|   </div> | ||||
| </div> | ||||
| @@ -0,0 +1,3 @@ | ||||
| .copied-badge { | ||||
|     right: 15em; | ||||
| } | ||||
| @@ -11,17 +11,18 @@ import { | ||||
|   tick, | ||||
| } from '@angular/core/testing' | ||||
| import { By } from '@angular/platform-browser' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { of, throwError } from 'rxjs' | ||||
| import { FileVersion, ShareLink } from 'src/app/data/share-link' | ||||
| import { ShareLinkService } from 'src/app/services/rest/share-link.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { environment } from 'src/environments/environment' | ||||
| import { ShareLinksDropdownComponent } from './share-links-dropdown.component' | ||||
| import { ShareLinksDialogComponent } from './share-links-dialog.component' | ||||
| 
 | ||||
| describe('ShareLinksDropdownComponent', () => { | ||||
|   let component: ShareLinksDropdownComponent | ||||
|   let fixture: ComponentFixture<ShareLinksDropdownComponent> | ||||
| describe('ShareLinksDialogComponent', () => { | ||||
|   let component: ShareLinksDialogComponent | ||||
|   let fixture: ComponentFixture<ShareLinksDialogComponent> | ||||
|   let shareLinkService: ShareLinkService | ||||
|   let toastService: ToastService | ||||
|   let httpController: HttpTestingController | ||||
| @@ -30,16 +31,17 @@ describe('ShareLinksDropdownComponent', () => { | ||||
|   beforeEach(() => { | ||||
|     TestBed.configureTestingModule({ | ||||
|       imports: [ | ||||
|         ShareLinksDropdownComponent, | ||||
|         ShareLinksDialogComponent, | ||||
|         NgxBootstrapIconsModule.pick(allIcons), | ||||
|       ], | ||||
|       providers: [ | ||||
|         provideHttpClient(withInterceptorsFromDi()), | ||||
|         provideHttpClientTesting(), | ||||
|         NgbActiveModal, | ||||
|       ], | ||||
|     }) | ||||
| 
 | ||||
|     fixture = TestBed.createComponent(ShareLinksDropdownComponent) | ||||
|     fixture = TestBed.createComponent(ShareLinksDialogComponent) | ||||
|     shareLinkService = TestBed.inject(ShareLinkService) | ||||
|     toastService = TestBed.inject(ToastService) | ||||
|     httpController = TestBed.inject(HttpTestingController) | ||||
| @@ -232,4 +234,11 @@ describe('ShareLinksDropdownComponent', () => { | ||||
|       ] | ||||
|     ).toBeTruthy() | ||||
|   }) | ||||
| 
 | ||||
|   it('should support close', () => { | ||||
|     const activeModal = TestBed.inject(NgbActiveModal) | ||||
|     const closeSpy = jest.spyOn(activeModal, 'close') | ||||
|     component.close() | ||||
|     expect(closeSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { Clipboard } from '@angular/cdk/clipboard' | ||||
| import { Component, Input, OnInit } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbDropdownModule } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { first } from 'rxjs' | ||||
| import { FileVersion, ShareLink } from 'src/app/data/share-link' | ||||
| @@ -10,17 +10,12 @@ import { ToastService } from 'src/app/services/toast.service' | ||||
| import { environment } from 'src/environments/environment' | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'pngx-share-links-dropdown', | ||||
|   templateUrl: './share-links-dropdown.component.html', | ||||
|   styleUrls: ['./share-links-dropdown.component.scss'], | ||||
|   imports: [ | ||||
|     FormsModule, | ||||
|     ReactiveFormsModule, | ||||
|     NgbDropdownModule, | ||||
|     NgxBootstrapIconsModule, | ||||
|   ], | ||||
|   selector: 'pngx-share-links-dialog', | ||||
|   templateUrl: './share-links-dialog.component.html', | ||||
|   styleUrls: ['./share-links-dialog.component.scss'], | ||||
|   imports: [FormsModule, ReactiveFormsModule, NgxBootstrapIconsModule], | ||||
| }) | ||||
| export class ShareLinksDropdownComponent implements OnInit { | ||||
| export class ShareLinksDialogComponent implements OnInit { | ||||
|   EXPIRATION_OPTIONS = [ | ||||
|     { label: $localize`1 day`, value: 1 }, | ||||
|     { label: $localize`7 days`, value: 7 }, | ||||
| @@ -41,9 +36,6 @@ export class ShareLinksDropdownComponent implements OnInit { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @Input() | ||||
|   disabled: boolean = false | ||||
| 
 | ||||
|   private _hasArchiveVersion: boolean = true | ||||
| 
 | ||||
|   @Input() | ||||
| @@ -67,6 +59,7 @@ export class ShareLinksDropdownComponent implements OnInit { | ||||
|   useArchiveVersion: boolean = true | ||||
| 
 | ||||
|   constructor( | ||||
|     private activeModal: NgbActiveModal, | ||||
|     private shareLinkService: ShareLinkService, | ||||
|     private toastService: ToastService, | ||||
|     private clipboard: Clipboard | ||||
| @@ -169,4 +162,8 @@ export class ShareLinksDropdownComponent implements OnInit { | ||||
|         }, | ||||
|       }) | ||||
|   } | ||||
| 
 | ||||
|   close() { | ||||
|     this.activeModal.close() | ||||
|   } | ||||
| } | ||||
| @@ -1,70 +0,0 @@ | ||||
| <div ngbDropdown> | ||||
|   <button class="btn btn-sm btn-outline-primary" id="shareLinksDropdown" [disabled]="disabled" ngbDropdownToggle> | ||||
|     <i-bs name="link"></i-bs> | ||||
|     <div class="d-none d-sm-inline"> <ng-container i18n>Share Links</ng-container></div> | ||||
|   </button> | ||||
|   <div ngbDropdownMenu aria-labelledby="shareLinksDropdown" class="shadow share-links-dropdown"> | ||||
|     <ul class="list-group list-group-flush"> | ||||
|       @if (!shareLinks || shareLinks.length === 0) { | ||||
|         <li class="list-group-item fst-italic small text-center text-secondary" i18n> | ||||
|           No existing links | ||||
|         </li> | ||||
|       } | ||||
|       @for (link of shareLinks; track link) { | ||||
|         <li class="list-group-item"> | ||||
|           <div class="input-group input-group-sm w-100"> | ||||
|             <input type="text" class="form-control" aria-label="Share link" [value]="getShareUrl(link)" readonly> | ||||
|             @if (link.expiration) { | ||||
|               <span class="input-group-text"> | ||||
|                 {{ getDaysRemaining(link) }} | ||||
|               </span> | ||||
|             } | ||||
|             <button type="button" class="btn btn-sm btn-outline-primary" (click)="copy(link)"> | ||||
|                 @if (copied !== link.id) { | ||||
|                   <i-bs width="1.2em" height="1.2em" name="clipboard-fill"></i-bs> | ||||
|                 } | ||||
|                 @if (copied === link.id) { | ||||
|                   <i-bs width="1.2em" height="1.2em" name="clipboard-check-fill"></i-bs> | ||||
|                 } | ||||
|                 <span class="visually-hidden" i18n>Copy</span> | ||||
|               </button> | ||||
|               @if (canShare(link)) { | ||||
|                 <button type="button" class="btn btn-sm btn-outline-primary" (click)="share(link)"> | ||||
|                   <i-bs width="1.2em" height="1.2em" name="box-arrow-up"></i-bs><span class="visually-hidden" i18n>Share</span> | ||||
|                   </button> | ||||
|                 } | ||||
|                 <button type="button" class="btn btn-sm btn-outline-danger" (click)="delete(link)"> | ||||
|                   <i-bs width="1.2em" height="1.2em" name="trash"></i-bs><span class="visually-hidden" i18n>Delete</span> | ||||
|                   </button> | ||||
|                 </div> | ||||
|                 <span class="badge copied-badge bg-primary small fade ms-4 position-absolute top-50 translate-middle-y pe-none z-3" [class.show]="copied === link.id" i18n>Copied!</span> | ||||
|               </li> | ||||
|             } | ||||
|             <li class="list-group-item pt-3 pb-2"> | ||||
|               <div class="input-group input-group-sm w-100"> | ||||
|                 <div class="form-check form-switch ms-auto small"> | ||||
|                   <input class="form-check-input" type="checkbox" role="switch" id="versionSwitch" [disabled]="!hasArchiveVersion" [(ngModel)]="useArchiveVersion"> | ||||
|                   <label class="form-check-label" for="versionSwitch" i18n>Share archive version</label> | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="input-group input-group-sm w-100 mt-2"> | ||||
|                 <label class="input-group-text" for="addLink"><ng-container i18n>Expires</ng-container>:</label> | ||||
|                 <select class="form-select form-select-sm" [(ngModel)]="expirationDays"> | ||||
|                   @for (option of EXPIRATION_OPTIONS; track option) { | ||||
|                     <option [ngValue]="option.value">{{ option.label }}</option> | ||||
|                   } | ||||
|                 </select> | ||||
|                 <button class="btn btn-sm btn-outline-primary ms-auto" type="button" (click)="createLink()" [disabled]="loading"> | ||||
|                   @if (loading) { | ||||
|                     <div class="spinner-border spinner-border-sm me-2" role="status"></div> | ||||
|                   } | ||||
|                   @if (!loading) { | ||||
|                     <i-bs name="plus"></i-bs> | ||||
|                   } | ||||
|                   <ng-container i18n>Create</ng-container> | ||||
|                 </button> | ||||
|               </div> | ||||
|             </li> | ||||
|           </ul> | ||||
|         </div> | ||||
|       </div> | ||||
| @@ -1,14 +0,0 @@ | ||||
| .share-links-dropdown { | ||||
|     min-width: 350px; | ||||
|  | ||||
|     // correct position on mobile | ||||
|     @media (max-width: 575.98px) { | ||||
|         &.show { | ||||
|             margin-left: -175px !important; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| .copied-badge { | ||||
|     right: 7.5em; | ||||
| } | ||||
| @@ -81,7 +81,24 @@ | ||||
|     (added)="addField($event)"> | ||||
|   </pngx-custom-fields-dropdown> | ||||
|  | ||||
|   <pngx-share-links-dropdown [documentId]="documentId" [hasArchiveVersion]="!!document?.archived_file_name" [disabled]="!userCanEdit && !userIsOwner" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.ShareLink }"></pngx-share-links-dropdown> | ||||
|  | ||||
|   <div class="ms-auto" ngbDropdown> | ||||
|     <button class="btn btn-sm btn-outline-primary" id="sendDropdown" ngbDropdownToggle> | ||||
|       <i-bs name="send"></i-bs> | ||||
|       <div class="d-none d-sm-inline"> <ng-container i18n>Send</ng-container></div> | ||||
|     </button> | ||||
|     <div ngbDropdownMenu aria-labelledby="actionsDropdown" class="shadow"> | ||||
|       <button ngbDropdownItem (click)="openShareLinks()" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.ShareLink }"> | ||||
|         <i-bs name="link"></i-bs> <span i18n>Share Links</span> | ||||
|       </button> | ||||
|       @if (emailEnabled) { | ||||
|         <button ngbDropdownItem (click)="openEmailDocument()"> | ||||
|           <i-bs name="envelope"></i-bs> <span i18n>Email</span> | ||||
|         </button> | ||||
|       } | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
| </pngx-page-header> | ||||
|  | ||||
| <div class="row"> | ||||
|   | ||||
| @@ -1330,4 +1330,18 @@ describe('DocumentDetailComponent', () => { | ||||
|     expect(createSpy).toHaveBeenCalledWith('a') | ||||
|     expect(urlRevokeSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
|   it('should get email enabled status from settings', () => { | ||||
|     jest.spyOn(settingsService, 'get').mockReturnValue(true) | ||||
|     expect(component.emailEnabled).toBeTruthy() | ||||
|   }) | ||||
|  | ||||
|   it('should support open share links and email modals', () => { | ||||
|     const modalSpy = jest.spyOn(modalService, 'open') | ||||
|     initNormally() | ||||
|     component.openShareLinks() | ||||
|     expect(modalSpy).toHaveBeenCalled() | ||||
|     component.openEmailDocument() | ||||
|     expect(modalSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
|   | ||||
| @@ -88,6 +88,7 @@ import { CorrespondentEditDialogComponent } from '../common/edit-dialog/correspo | ||||
| import { DocumentTypeEditDialogComponent } from '../common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component' | ||||
| import { EditDialogMode } from '../common/edit-dialog/edit-dialog.component' | ||||
| import { StoragePathEditDialogComponent } from '../common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component' | ||||
| import { EmailDocumentDialogComponent } from '../common/email-document-dialog/email-document-dialog.component' | ||||
| import { CheckComponent } from '../common/input/check/check.component' | ||||
| import { DateComponent } from '../common/input/date/date.component' | ||||
| import { DocumentLinkComponent } from '../common/input/document-link/document-link.component' | ||||
| @@ -99,7 +100,7 @@ import { TagsComponent } from '../common/input/tags/tags.component' | ||||
| import { TextComponent } from '../common/input/text/text.component' | ||||
| import { UrlComponent } from '../common/input/url/url.component' | ||||
| import { PageHeaderComponent } from '../common/page-header/page-header.component' | ||||
| import { ShareLinksDropdownComponent } from '../common/share-links-dropdown/share-links-dropdown.component' | ||||
| import { ShareLinksDialogComponent } from '../common/share-links-dialog/share-links-dialog.component' | ||||
| import { DocumentHistoryComponent } from '../document-history/document-history.component' | ||||
| import { DocumentNotesComponent } from '../document-notes/document-notes.component' | ||||
| import { ComponentWithPermissions } from '../with-permissions/with-permissions.component' | ||||
| @@ -145,7 +146,6 @@ export enum ZoomSetting { | ||||
|     CustomFieldsDropdownComponent, | ||||
|     DocumentNotesComponent, | ||||
|     DocumentHistoryComponent, | ||||
|     ShareLinksDropdownComponent, | ||||
|     CheckComponent, | ||||
|     DateComponent, | ||||
|     DocumentLinkComponent, | ||||
| @@ -1426,6 +1426,26 @@ export class DocumentDetailComponent | ||||
|       }) | ||||
|   } | ||||
|  | ||||
|   public openShareLinks() { | ||||
|     const modal = this.modalService.open(ShareLinksDialogComponent) | ||||
|     modal.componentInstance.documentId = this.document.id | ||||
|     modal.componentInstance.hasArchiveVersion = | ||||
|       !!this.document?.archived_file_name | ||||
|   } | ||||
|  | ||||
|   get emailEnabled(): boolean { | ||||
|     return this.settings.get(SETTINGS_KEYS.EMAIL_ENABLED) | ||||
|   } | ||||
|  | ||||
|   public openEmailDocument() { | ||||
|     const modal = this.modalService.open(EmailDocumentDialogComponent, { | ||||
|       backdrop: 'static', | ||||
|     }) | ||||
|     modal.componentInstance.documentId = this.document.id | ||||
|     modal.componentInstance.hasArchiveVersion = | ||||
|       !!this.document?.archived_file_name | ||||
|   } | ||||
|  | ||||
|   private tryRenderTiff() { | ||||
|     this.http.get(this.previewUrl, { responseType: 'arraybuffer' }).subscribe({ | ||||
|       next: (res) => { | ||||
|   | ||||
| @@ -355,6 +355,21 @@ it('should include custom fields in sort fields if user has permission', () => { | ||||
|   ]) | ||||
| }) | ||||
|  | ||||
| it('should call appropriate api endpoint for email document', () => { | ||||
|   subscription = service | ||||
|     .emailDocument( | ||||
|       documents[0].id, | ||||
|       'hello@paperless-ngx.com', | ||||
|       'hello', | ||||
|       'world', | ||||
|       true | ||||
|     ) | ||||
|     .subscribe() | ||||
|   httpTestingController.expectOne( | ||||
|     `${environment.apiBaseUrl}${endpoint}/${documents[0].id}/email/` | ||||
|   ) | ||||
| }) | ||||
|  | ||||
| afterEach(() => { | ||||
|   subscription?.unsubscribe() | ||||
|   httpTestingController.verify() | ||||
|   | ||||
| @@ -258,4 +258,19 @@ export class DocumentService extends AbstractPaperlessService<Document> { | ||||
|   public get searchQuery(): string { | ||||
|     return this._searchQuery | ||||
|   } | ||||
|  | ||||
|   emailDocument( | ||||
|     documentId: number, | ||||
|     addresses: string, | ||||
|     subject: string, | ||||
|     message: string, | ||||
|     useArchiveVersion: boolean | ||||
|   ): Observable<any> { | ||||
|     return this.http.post(this.getResourceUrl(documentId, 'email'), { | ||||
|       addresses: addresses, | ||||
|       subject: subject, | ||||
|       message: message, | ||||
|       use_archive_version: useArchiveVersion, | ||||
|     }) | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -112,6 +112,7 @@ import { | ||||
|   questionCircle, | ||||
|   scissors, | ||||
|   search, | ||||
|   send, | ||||
|   slashCircle, | ||||
|   sliders2Vertical, | ||||
|   sortAlphaDown, | ||||
| @@ -316,6 +317,7 @@ const icons = { | ||||
|   questionCircle, | ||||
|   scissors, | ||||
|   search, | ||||
|   send, | ||||
|   slashCircle, | ||||
|   sliders2Vertical, | ||||
|   sortAlphaDown, | ||||
|   | ||||
| @@ -15,6 +15,7 @@ from dateutil import parser | ||||
| from django.conf import settings | ||||
| from django.contrib.auth.models import Permission | ||||
| from django.contrib.auth.models import User | ||||
| from django.core import mail | ||||
| from django.core.cache import cache | ||||
| from django.db import DataError | ||||
| from django.test import override_settings | ||||
| @@ -2651,6 +2652,153 @@ class TestDocumentApi(DirectoriesMixin, DocumentConsumeDelayMixin, APITestCase): | ||||
|         self.assertEqual(resp.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(doc1.tags.count(), 2) | ||||
|  | ||||
|     @override_settings( | ||||
|         EMAIL_ENABLED=True, | ||||
|         EMAIL_BACKEND="django.core.mail.backends.locmem.EmailBackend", | ||||
|     ) | ||||
|     def test_email_document(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing document | ||||
|         WHEN: | ||||
|             - API request is made to email document action | ||||
|         THEN: | ||||
|             - Email is sent, with document (original or archive) attached | ||||
|         """ | ||||
|         doc = Document.objects.create( | ||||
|             title="test", | ||||
|             mime_type="application/pdf", | ||||
|             content="this is a document 1", | ||||
|             checksum="1", | ||||
|             filename="test.pdf", | ||||
|             archive_checksum="A", | ||||
|             archive_filename="archive.pdf", | ||||
|         ) | ||||
|         doc2 = Document.objects.create( | ||||
|             title="test2", | ||||
|             mime_type="application/pdf", | ||||
|             content="this is a document 2", | ||||
|             checksum="2", | ||||
|             filename="test2.pdf", | ||||
|         ) | ||||
|  | ||||
|         archive_file = Path(__file__).parent / "samples" / "simple.pdf" | ||||
|         source_file = Path(__file__).parent / "samples" / "simple.pdf" | ||||
|  | ||||
|         shutil.copy(archive_file, doc.archive_path) | ||||
|         shutil.copy(source_file, doc2.source_path) | ||||
|  | ||||
|         self.client.post( | ||||
|             f"/api/documents/{doc.pk}/email/", | ||||
|             { | ||||
|                 "addresses": "hello@paperless-ngx.com", | ||||
|                 "subject": "test", | ||||
|                 "message": "hello", | ||||
|             }, | ||||
|         ) | ||||
|  | ||||
|         self.assertEqual(len(mail.outbox), 1) | ||||
|         self.assertEqual(mail.outbox[0].attachments[0][0], "archive.pdf") | ||||
|  | ||||
|         self.client.post( | ||||
|             f"/api/documents/{doc2.pk}/email/", | ||||
|             { | ||||
|                 "addresses": "hello@paperless-ngx.com", | ||||
|                 "subject": "test", | ||||
|                 "message": "hello", | ||||
|                 "use_archive_version": False, | ||||
|             }, | ||||
|         ) | ||||
|  | ||||
|         self.assertEqual(len(mail.outbox), 2) | ||||
|         self.assertEqual(mail.outbox[1].attachments[0][0], "test2.pdf") | ||||
|  | ||||
|     @mock.patch("django.core.mail.message.EmailMessage.send", side_effect=Exception) | ||||
|     def test_email_document_errors(self, mocked_send): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing document | ||||
|         WHEN: | ||||
|             - API request is made to email document action with insufficient permissions | ||||
|             - API request is made to email document action with invalid document id | ||||
|             - API request is made to email document action with missing data | ||||
|             - API request is made to email document action with invalid email address | ||||
|             - API request is made to email document action and error occurs during email send | ||||
|         THEN: | ||||
|             - Error response is returned | ||||
|         """ | ||||
|         user1 = User.objects.create_user(username="test1") | ||||
|         user1.user_permissions.add(*Permission.objects.all()) | ||||
|         user1.save() | ||||
|  | ||||
|         doc = Document.objects.create( | ||||
|             title="test", | ||||
|             mime_type="application/pdf", | ||||
|             content="this is a document 1", | ||||
|             checksum="1", | ||||
|             filename="test.pdf", | ||||
|             archive_checksum="A", | ||||
|             archive_filename="archive.pdf", | ||||
|         ) | ||||
|  | ||||
|         doc2 = Document.objects.create( | ||||
|             title="test2", | ||||
|             mime_type="application/pdf", | ||||
|             content="this is a document 2", | ||||
|             checksum="2", | ||||
|             owner=self.user, | ||||
|         ) | ||||
|  | ||||
|         self.client.force_authenticate(user1) | ||||
|  | ||||
|         resp = self.client.post( | ||||
|             f"/api/documents/{doc2.pk}/email/", | ||||
|             { | ||||
|                 "addresses": "hello@paperless-ngx.com", | ||||
|                 "subject": "test", | ||||
|                 "message": "hello", | ||||
|             }, | ||||
|         ) | ||||
|         self.assertEqual(resp.status_code, status.HTTP_403_FORBIDDEN) | ||||
|  | ||||
|         resp = self.client.post( | ||||
|             "/api/documents/999/email/", | ||||
|             { | ||||
|                 "addresses": "hello@paperless-ngx.com", | ||||
|                 "subject": "test", | ||||
|                 "message": "hello", | ||||
|             }, | ||||
|         ) | ||||
|         self.assertEqual(resp.status_code, status.HTTP_404_NOT_FOUND) | ||||
|  | ||||
|         resp = self.client.post( | ||||
|             f"/api/documents/{doc.pk}/email/", | ||||
|             { | ||||
|                 "addresses": "hello@paperless-ngx.com", | ||||
|             }, | ||||
|         ) | ||||
|         self.assertEqual(resp.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|  | ||||
|         resp = self.client.post( | ||||
|             f"/api/documents/{doc.pk}/email/", | ||||
|             { | ||||
|                 "addresses": "hello@paperless-ngx.com,hello", | ||||
|                 "subject": "test", | ||||
|                 "message": "hello", | ||||
|             }, | ||||
|         ) | ||||
|         self.assertEqual(resp.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|  | ||||
|         resp = self.client.post( | ||||
|             f"/api/documents/{doc.pk}/email/", | ||||
|             { | ||||
|                 "addresses": "hello@paperless-ngx.com", | ||||
|                 "subject": "test", | ||||
|                 "message": "hello", | ||||
|             }, | ||||
|         ) | ||||
|         self.assertEqual(resp.status_code, status.HTTP_500_INTERNAL_SERVER_ERROR) | ||||
|  | ||||
|     @mock.patch("django_softdelete.models.SoftDeleteModel.delete") | ||||
|     def test_warn_on_delete_with_old_uuid_field(self, mocked_delete): | ||||
|         """ | ||||
|   | ||||
| @@ -37,6 +37,7 @@ from django.http import HttpResponse | ||||
| from django.http import HttpResponseBadRequest | ||||
| from django.http import HttpResponseForbidden | ||||
| from django.http import HttpResponseRedirect | ||||
| from django.http import HttpResponseServerError | ||||
| from django.shortcuts import get_object_or_404 | ||||
| from django.utils import timezone | ||||
| from django.utils.decorators import method_decorator | ||||
| @@ -106,6 +107,7 @@ from documents.filters import ObjectOwnedPermissionsFilter | ||||
| from documents.filters import ShareLinkFilterSet | ||||
| from documents.filters import StoragePathFilterSet | ||||
| from documents.filters import TagFilterSet | ||||
| from documents.mail import send_email | ||||
| from documents.matching import match_correspondents | ||||
| from documents.matching import match_document_types | ||||
| from documents.matching import match_storage_paths | ||||
| @@ -1023,6 +1025,57 @@ class DocumentViewSet( | ||||
|  | ||||
|         return Response(sorted(entries, key=lambda x: x["timestamp"], reverse=True)) | ||||
|  | ||||
|     @action(methods=["post"], detail=True) | ||||
|     def email(self, request, pk=None): | ||||
|         try: | ||||
|             doc = Document.objects.select_related("owner").get(pk=pk) | ||||
|             if request.user is not None and not has_perms_owner_aware( | ||||
|                 request.user, | ||||
|                 "view_document", | ||||
|                 doc, | ||||
|             ): | ||||
|                 return HttpResponseForbidden("Insufficient permissions") | ||||
|         except Document.DoesNotExist: | ||||
|             raise Http404 | ||||
|  | ||||
|         try: | ||||
|             if ( | ||||
|                 "addresses" not in request.data | ||||
|                 or "subject" not in request.data | ||||
|                 or "message" not in request.data | ||||
|             ): | ||||
|                 return HttpResponseBadRequest("Missing required fields") | ||||
|  | ||||
|             use_archive_version = request.data.get("use_archive_version", True) | ||||
|  | ||||
|             addresses = request.data.get("addresses").split(",") | ||||
|             if not all( | ||||
|                 re.match(r"[^@]+@[^@]+\.[^@]+", address.strip()) | ||||
|                 for address in addresses | ||||
|             ): | ||||
|                 return HttpResponseBadRequest("Invalid email address found") | ||||
|  | ||||
|             send_email( | ||||
|                 subject=request.data.get("subject"), | ||||
|                 body=request.data.get("message"), | ||||
|                 to=addresses, | ||||
|                 attachment=( | ||||
|                     doc.archive_path | ||||
|                     if use_archive_version and doc.has_archive_version | ||||
|                     else doc.source_path | ||||
|                 ), | ||||
|                 attachment_mime_type=doc.mime_type, | ||||
|             ) | ||||
|             logger.debug( | ||||
|                 f"Sent document {doc.id} via email to {addresses}", | ||||
|             ) | ||||
|             return Response({"message": "Email sent"}) | ||||
|         except Exception as e: | ||||
|             logger.warning(f"An error occurred emailing document: {e!s}") | ||||
|             return HttpResponseServerError( | ||||
|                 "Error emailing document, check logs for more detail.", | ||||
|             ) | ||||
|  | ||||
|  | ||||
| @extend_schema_view( | ||||
|     list=extend_schema( | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 shamoon
					shamoon