Compare commits
	
		
			162 Commits
		
	
	
		
			v0.13.0-be
			...
			v0.19.1
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					f43f0e10a1 | ||
| 
						 | 
					6d842a65a2 | ||
| 
						 | 
					50c4de021c | ||
| 
						 | 
					936d8449f6 | ||
| 
						 | 
					462bc0edd5 | ||
| 
						 | 
					35ef3c8470 | ||
| 
						 | 
					5117d43e29 | ||
| 
						 | 
					7c9ac488e8 | ||
| 
						 | 
					fec1574447 | ||
| 
						 | 
					f7587de452 | ||
| 
						 | 
					41a6e777ea | ||
| 
						 | 
					8fb0de900b | ||
| 
						 | 
					a40bf71fd4 | ||
| 
						 | 
					2a53bb03ae | ||
| 
						 | 
					a684879b91 | ||
| 
						 | 
					ddbd4f73c8 | ||
| 
						 | 
					b0ed790edf | ||
| 
						 | 
					3424406ff1 | ||
| 
						 | 
					ce5c385c15 | ||
| 
						 | 
					cd9eef32b0 | ||
| 
						 | 
					12d82e3b33 | ||
| 
						 | 
					f071d3f651 | ||
| 
						 | 
					297b536ebc | ||
| 
						 | 
					7cca2d9247 | ||
| 
						 | 
					36dc802d16 | ||
| 
						 | 
					c78ddb70cb | ||
| 
						 | 
					9fb0599c45 | ||
| 
						 | 
					13f524fb39 | ||
| 
						 | 
					27be3b4c90 | ||
| 
						 | 
					af4ea66742 | ||
| 
						 | 
					0f42c396f3 | ||
| 
						 | 
					9e96272eb3 | ||
| 
						 | 
					965dbcbc9a | ||
| 
						 | 
					7ac109e7f7 | ||
| 
						 | 
					ac25770c0e | ||
| 
						 | 
					5b15424498 | ||
| 
						 | 
					f1240c26bf | ||
| 
						 | 
					1c0fd243d1 | ||
| 
						 | 
					3491235533 | ||
| 
						 | 
					5f36856571 | ||
| 
						 | 
					d3e44b37e9 | ||
| 
						 | 
					90e9297ec5 | ||
| 
						 | 
					c568ef2f8a | ||
| 
						 | 
					fcf6141cde | ||
| 
						 | 
					21551d7b77 | ||
| 
						 | 
					12031d369f | ||
| 
						 | 
					b44c523845 | ||
| 
						 | 
					49989695ff | ||
| 
						 | 
					a55d3530e9 | ||
| 
						 | 
					2aab3ad281 | ||
| 
						 | 
					194ce4f10f | ||
| 
						 | 
					2089c32839 | ||
| 
						 | 
					f437be7af0 | ||
| 
						 | 
					96dc56098d | ||
| 
						 | 
					61987e46f7 | ||
| 
						 | 
					509093b755 | ||
| 
						 | 
					097114c0f2 | ||
| 
						 | 
					040f9185f8 | ||
| 
						 | 
					6dc934abbe | ||
| 
						 | 
					2d24bf81dd | ||
| 
						 | 
					9452fc236b | ||
| 
						 | 
					365c37604b | ||
| 
						 | 
					01c7e58d47 | ||
| 
						 | 
					d3d49923b1 | ||
| 
						 | 
					263ac299d0 | ||
| 
						 | 
					3d185a5178 | ||
| 
						 | 
					2ff7a890bc | ||
| 
						 | 
					2eb1a9705f | ||
| 
						 | 
					ed1381103a | ||
| 
						 | 
					170d317589 | ||
| 
						 | 
					ededc063df | ||
| 
						 | 
					986eace1be | ||
| 
						 | 
					29086d8dfe | ||
| 
						 | 
					9b3f3fde05 | ||
| 
						 | 
					6a50afd952 | ||
| 
						 | 
					697eee2706 | ||
| 
						 | 
					8a95afd756 | ||
| 
						 | 
					4d6eda8fe6 | ||
| 
						 | 
					e4f459fa2b | ||
| 
						 | 
					f578e001b0 | ||
| 
						 | 
					2a08aef885 | ||
| 
						 | 
					7564bf388c | ||
| 
						 | 
					7e4d70259f | ||
| 
						 | 
					5b98c1c0f3 | ||
| 
						 | 
					02dc7b199b | ||
| 
						 | 
					d39cdbfada | ||
| 
						 | 
					50bb4a47ee | ||
| 
						 | 
					a4627f2ddb | ||
| 
						 | 
					c8253caae9 | ||
| 
						 | 
					0ece9bd1be | ||
| 
						 | 
					b6935abcc9 | ||
| 
						 | 
					37ab7b4641 | ||
| 
						 | 
					013714cb5c | ||
| 
						 | 
					1fe7c62f5a | ||
| 
						 | 
					a06618d851 | ||
| 
						 | 
					e7460ca3a9 | ||
| 
						 | 
					073300bbcd | ||
| 
						 | 
					a201661ce5 | ||
| 
						 | 
					6235a3c886 | ||
| 
						 | 
					3972c27e7a | ||
| 
						 | 
					14cffbbe62 | ||
| 
						 | 
					599c3c04af | ||
| 
						 | 
					f1412b631d | ||
| 
						 | 
					41908050bb | ||
| 
						 | 
					f07033423c | ||
| 
						 | 
					daf96fcbf2 | ||
| 
						 | 
					2bca94529e | ||
| 
						 | 
					b2c9a0da21 | ||
| 
						 | 
					7a9542b4fc | ||
| 
						 | 
					3a95c9e1bc | ||
| 
						 | 
					3d2ef6be01 | ||
| 
						 | 
					d67246699a | ||
| 
						 | 
					14c704d6db | ||
| 
						 | 
					4c8eeb2e6f | ||
| 
						 | 
					c1b245c8b1 | ||
| 
						 | 
					74202d67bb | ||
| 
						 | 
					26066f39b1 | ||
| 
						 | 
					b255cf190c | ||
| 
						 | 
					bc77b143b0 | ||
| 
						 | 
					9f0ff6ae7a | ||
| 
						 | 
					736704c7d6 | ||
| 
						 | 
					654c116c58 | ||
| 
						 | 
					89a5cab98f | ||
| 
						 | 
					c39d0be8cd | ||
| 
						 | 
					e75b4cd848 | ||
| 
						 | 
					378e8f35e5 | ||
| 
						 | 
					bdb5e2f13f | ||
| 
						 | 
					8211bed449 | ||
| 
						 | 
					b243632483 | ||
| 
						 | 
					e4d2513451 | ||
| 
						 | 
					385144451b | ||
| 
						 | 
					c8c533844e | ||
| 
						 | 
					0e69f0c079 | ||
| 
						 | 
					aee60c444f | ||
| 
						 | 
					e7a504c66b | ||
| 
						 | 
					45d9c7164c | ||
| 
						 | 
					bd913a63a8 | ||
| 
						 | 
					5a1938c078 | ||
| 
						 | 
					015cd68756 | ||
| 
						 | 
					76c0e5b2b8 | ||
| 
						 | 
					0f8f707acd | ||
| 
						 | 
					083cccea28 | ||
| 
						 | 
					31b76b23ce | ||
| 
						 | 
					af529f82e5 | ||
| 
						 | 
					fc6669d254 | ||
| 
						 | 
					c07785be67 | ||
| 
						 | 
					80d2457b23 | ||
| 
						 | 
					5dde2752d2 | ||
| 
						 | 
					8bf4633cd0 | ||
| 
						 | 
					bd66b8a1c8 | ||
| 
						 | 
					be51e533fc | ||
| 
						 | 
					f47ae12019 | ||
| 
						 | 
					cab54a458f | ||
| 
						 | 
					a30734f1bc | ||
| 
						 | 
					7ad9f7b129 | ||
| 
						 | 
					40a32e6826 | ||
| 
						 | 
					ab0486aaf1 | ||
| 
						 | 
					874593a167 | ||
| 
						 | 
					03bf33630e | ||
| 
						 | 
					933cce1b94 | ||
| 
						 | 
					4a6ff573f8 | ||
| 
						 | 
					1a737f7d19 | 
							
								
								
									
										4
									
								
								.dockerignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,4 @@
 | 
				
			|||||||
 | 
					node_modules
 | 
				
			||||||
 | 
					npm-debug.log
 | 
				
			||||||
 | 
					dist
 | 
				
			||||||
 | 
					.idea
 | 
				
			||||||
@@ -1,6 +1,6 @@
 | 
				
			|||||||
<?xml version="1.0" encoding="UTF-8"?>
 | 
					<?xml version="1.0" encoding="UTF-8"?>
 | 
				
			||||||
<dataSource name="document.db">
 | 
					<dataSource name="document.db">
 | 
				
			||||||
  <database-model serializer="dbm" rdbms="SQLITE" format-version="4.8">
 | 
					  <database-model serializer="dbm" rdbms="SQLITE" format-version="4.11">
 | 
				
			||||||
    <root id="1">
 | 
					    <root id="1">
 | 
				
			||||||
      <ServerVersion>3.16.1</ServerVersion>
 | 
					      <ServerVersion>3.16.1</ServerVersion>
 | 
				
			||||||
    </root>
 | 
					    </root>
 | 
				
			||||||
@@ -12,10 +12,10 @@
 | 
				
			|||||||
    <collation id="4" parent="1" name="NOCASE"/>
 | 
					    <collation id="4" parent="1" name="NOCASE"/>
 | 
				
			||||||
    <collation id="5" parent="1" name="RTRIM"/>
 | 
					    <collation id="5" parent="1" name="RTRIM"/>
 | 
				
			||||||
    <table id="6" parent="2" name="api_tokens"/>
 | 
					    <table id="6" parent="2" name="api_tokens"/>
 | 
				
			||||||
    <table id="7" parent="2" name="branches"/>
 | 
					    <table id="7" parent="2" name="attributes"/>
 | 
				
			||||||
    <table id="8" parent="2" name="event_log"/>
 | 
					    <table id="8" parent="2" name="branches"/>
 | 
				
			||||||
    <table id="9" parent="2" name="images"/>
 | 
					    <table id="9" parent="2" name="event_log"/>
 | 
				
			||||||
    <table id="10" parent="2" name="labels"/>
 | 
					    <table id="10" parent="2" name="images"/>
 | 
				
			||||||
    <table id="11" parent="2" name="note_images"/>
 | 
					    <table id="11" parent="2" name="note_images"/>
 | 
				
			||||||
    <table id="12" parent="2" name="note_revisions"/>
 | 
					    <table id="12" parent="2" name="note_revisions"/>
 | 
				
			||||||
    <table id="13" parent="2" name="notes"/>
 | 
					    <table id="13" parent="2" name="notes"/>
 | 
				
			||||||
@@ -50,546 +50,600 @@
 | 
				
			|||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="24" parent="6" name="sqlite_autoindex_api_tokens_1">
 | 
					    <column id="24" parent="6" name="hash">
 | 
				
			||||||
 | 
					      <Position>5</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <index id="25" parent="6" name="sqlite_autoindex_api_tokens_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>apiTokenId</ColNames>
 | 
					      <ColNames>apiTokenId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="25" parent="6">
 | 
					    <key id="26" parent="6">
 | 
				
			||||||
      <ColNames>apiTokenId</ColNames>
 | 
					      <ColNames>apiTokenId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_api_tokens_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_api_tokens_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="26" parent="7" name="branchId">
 | 
					    <column id="27" parent="7" name="attributeId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="27" parent="7" name="noteId">
 | 
					    <column id="28" parent="7" name="noteId">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="28" parent="7" name="parentNoteId">
 | 
					    <column id="29" parent="7" name="type">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="29" parent="7" name="notePosition">
 | 
					    <column id="30" parent="7" name="name">
 | 
				
			||||||
 | 
					      <Position>4</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="31" parent="7" name="value">
 | 
				
			||||||
 | 
					      <Position>5</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>''</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="32" parent="7" name="position">
 | 
				
			||||||
 | 
					      <Position>6</Position>
 | 
				
			||||||
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="33" parent="7" name="dateCreated">
 | 
				
			||||||
 | 
					      <Position>7</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="34" parent="7" name="dateModified">
 | 
				
			||||||
 | 
					      <Position>8</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="35" parent="7" name="isDeleted">
 | 
				
			||||||
 | 
					      <Position>9</Position>
 | 
				
			||||||
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="36" parent="7" name="hash">
 | 
				
			||||||
 | 
					      <Position>10</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="37" parent="7" name="isInheritable">
 | 
				
			||||||
 | 
					      <Position>11</Position>
 | 
				
			||||||
 | 
					      <DataType>int|0s</DataType>
 | 
				
			||||||
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <index id="38" parent="7" name="sqlite_autoindex_attributes_1">
 | 
				
			||||||
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
 | 
					      <ColNames>attributeId</ColNames>
 | 
				
			||||||
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
 | 
					    </index>
 | 
				
			||||||
 | 
					    <key id="39" parent="7">
 | 
				
			||||||
 | 
					      <ColNames>attributeId</ColNames>
 | 
				
			||||||
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
 | 
					      <UnderlyingIndexName>sqlite_autoindex_attributes_1</UnderlyingIndexName>
 | 
				
			||||||
 | 
					    </key>
 | 
				
			||||||
 | 
					    <column id="40" parent="8" name="branchId">
 | 
				
			||||||
 | 
					      <Position>1</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="41" parent="8" name="noteId">
 | 
				
			||||||
 | 
					      <Position>2</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="42" parent="8" name="parentNoteId">
 | 
				
			||||||
 | 
					      <Position>3</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="43" parent="8" name="notePosition">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>INTEGER|0s</DataType>
 | 
					      <DataType>INTEGER|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="30" parent="7" name="prefix">
 | 
					    <column id="44" parent="8" name="prefix">
 | 
				
			||||||
      <Position>5</Position>
 | 
					      <Position>5</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="31" parent="7" name="isExpanded">
 | 
					    <column id="45" parent="8" name="isExpanded">
 | 
				
			||||||
      <Position>6</Position>
 | 
					      <Position>6</Position>
 | 
				
			||||||
      <DataType>BOOLEAN|0s</DataType>
 | 
					      <DataType>BOOLEAN|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="32" parent="7" name="isDeleted">
 | 
					    <column id="46" parent="8" name="isDeleted">
 | 
				
			||||||
      <Position>7</Position>
 | 
					      <Position>7</Position>
 | 
				
			||||||
      <DataType>INTEGER|0s</DataType>
 | 
					      <DataType>INTEGER|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="33" parent="7" name="dateModified">
 | 
					    <column id="47" parent="8" name="dateModified">
 | 
				
			||||||
      <Position>8</Position>
 | 
					      <Position>8</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="34" parent="7" name="sqlite_autoindex_branches_1">
 | 
					    <column id="48" parent="8" name="hash">
 | 
				
			||||||
 | 
					      <Position>9</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="49" parent="8" name="dateCreated">
 | 
				
			||||||
 | 
					      <Position>10</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>'1970-01-01T00:00:00.000Z'</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <index id="50" parent="8" name="sqlite_autoindex_branches_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>branchId</ColNames>
 | 
					      <ColNames>branchId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="35" parent="7" name="IDX_branches_noteId_parentNoteId">
 | 
					    <index id="51" parent="8" name="IDX_branches_noteId_parentNoteId">
 | 
				
			||||||
      <ColNames>noteId
 | 
					      <ColNames>noteId
 | 
				
			||||||
parentNoteId</ColNames>
 | 
					parentNoteId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="36" parent="7" name="IDX_branches_noteId">
 | 
					    <index id="52" parent="8" name="IDX_branches_noteId">
 | 
				
			||||||
      <ColNames>noteId</ColNames>
 | 
					      <ColNames>noteId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="37" parent="7">
 | 
					    <index id="53" parent="8" name="IDX_branches_parentNoteId">
 | 
				
			||||||
 | 
					      <ColNames>parentNoteId</ColNames>
 | 
				
			||||||
 | 
					    </index>
 | 
				
			||||||
 | 
					    <key id="54" parent="8">
 | 
				
			||||||
      <ColNames>branchId</ColNames>
 | 
					      <ColNames>branchId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_branches_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_branches_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="38" parent="8" name="id">
 | 
					    <column id="55" parent="9" name="eventId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>INTEGER|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <SequenceIdentity>1</SequenceIdentity>
 | 
					 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="39" parent="8" name="noteId">
 | 
					    <column id="56" parent="9" name="noteId">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="40" parent="8" name="comment">
 | 
					    <column id="57" parent="9" name="comment">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="41" parent="8" name="dateAdded">
 | 
					    <column id="58" parent="9" name="dateCreated">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <key id="42" parent="8">
 | 
					    <index id="59" parent="9" name="sqlite_autoindex_event_log_1">
 | 
				
			||||||
      <ColNames>id</ColNames>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
 | 
					      <ColNames>eventId</ColNames>
 | 
				
			||||||
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
 | 
					    </index>
 | 
				
			||||||
 | 
					    <key id="60" parent="9">
 | 
				
			||||||
 | 
					      <ColNames>eventId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
 | 
					      <UnderlyingIndexName>sqlite_autoindex_event_log_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="43" parent="9" name="imageId">
 | 
					    <column id="61" parent="10" name="imageId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="44" parent="9" name="format">
 | 
					    <column id="62" parent="10" name="format">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="45" parent="9" name="checksum">
 | 
					    <column id="63" parent="10" name="checksum">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="46" parent="9" name="name">
 | 
					    <column id="64" parent="10" name="name">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="47" parent="9" name="data">
 | 
					    <column id="65" parent="10" name="data">
 | 
				
			||||||
      <Position>5</Position>
 | 
					      <Position>5</Position>
 | 
				
			||||||
      <DataType>BLOB|0s</DataType>
 | 
					      <DataType>BLOB|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="48" parent="9" name="isDeleted">
 | 
					    <column id="66" parent="10" name="isDeleted">
 | 
				
			||||||
      <Position>6</Position>
 | 
					      <Position>6</Position>
 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="49" parent="9" name="dateModified">
 | 
					    <column id="67" parent="10" name="dateModified">
 | 
				
			||||||
      <Position>7</Position>
 | 
					      <Position>7</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="50" parent="9" name="dateCreated">
 | 
					    <column id="68" parent="10" name="dateCreated">
 | 
				
			||||||
      <Position>8</Position>
 | 
					      <Position>8</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="51" parent="9" name="sqlite_autoindex_images_1">
 | 
					    <column id="69" parent="10" name="hash">
 | 
				
			||||||
 | 
					      <Position>9</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <index id="70" parent="10" name="sqlite_autoindex_images_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>imageId</ColNames>
 | 
					      <ColNames>imageId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="52" parent="9">
 | 
					    <key id="71" parent="10">
 | 
				
			||||||
      <ColNames>imageId</ColNames>
 | 
					      <ColNames>imageId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_images_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_images_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="53" parent="10" name="labelId">
 | 
					    <column id="72" parent="11" name="noteImageId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="54" parent="10" name="noteId">
 | 
					    <column id="73" parent="11" name="noteId">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="55" parent="10" name="name">
 | 
					    <column id="74" parent="11" name="imageId">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="56" parent="10" name="value">
 | 
					    <column id="75" parent="11" name="isDeleted">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
      <DefaultExpression>''</DefaultExpression>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <column id="57" parent="10" name="position">
 | 
					 | 
				
			||||||
      <Position>5</Position>
 | 
					 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="58" parent="10" name="dateCreated">
 | 
					    <column id="76" parent="11" name="dateModified">
 | 
				
			||||||
 | 
					      <Position>5</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="77" parent="11" name="dateCreated">
 | 
				
			||||||
      <Position>6</Position>
 | 
					      <Position>6</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="59" parent="10" name="dateModified">
 | 
					    <column id="78" parent="11" name="hash">
 | 
				
			||||||
      <Position>7</Position>
 | 
					      <Position>7</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="60" parent="10" name="isDeleted">
 | 
					    <index id="79" parent="11" name="sqlite_autoindex_note_images_1">
 | 
				
			||||||
      <Position>8</Position>
 | 
					 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <index id="61" parent="10" name="sqlite_autoindex_labels_1">
 | 
					 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					 | 
				
			||||||
      <ColNames>labelId</ColNames>
 | 
					 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					 | 
				
			||||||
    </index>
 | 
					 | 
				
			||||||
    <index id="62" parent="10" name="IDX_labels_noteId">
 | 
					 | 
				
			||||||
      <ColNames>noteId</ColNames>
 | 
					 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					 | 
				
			||||||
    <index id="63" parent="10" name="IDX_labels_name_value">
 | 
					 | 
				
			||||||
      <ColNames>name
 | 
					 | 
				
			||||||
value</ColNames>
 | 
					 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					 | 
				
			||||||
    <key id="64" parent="10">
 | 
					 | 
				
			||||||
      <ColNames>labelId</ColNames>
 | 
					 | 
				
			||||||
      <Primary>1</Primary>
 | 
					 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_labels_1</UnderlyingIndexName>
 | 
					 | 
				
			||||||
    </key>
 | 
					 | 
				
			||||||
    <column id="65" parent="11" name="noteImageId">
 | 
					 | 
				
			||||||
      <Position>1</Position>
 | 
					 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <column id="66" parent="11" name="noteId">
 | 
					 | 
				
			||||||
      <Position>2</Position>
 | 
					 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <column id="67" parent="11" name="imageId">
 | 
					 | 
				
			||||||
      <Position>3</Position>
 | 
					 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <column id="68" parent="11" name="isDeleted">
 | 
					 | 
				
			||||||
      <Position>4</Position>
 | 
					 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <column id="69" parent="11" name="dateModified">
 | 
					 | 
				
			||||||
      <Position>5</Position>
 | 
					 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <column id="70" parent="11" name="dateCreated">
 | 
					 | 
				
			||||||
      <Position>6</Position>
 | 
					 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					 | 
				
			||||||
    </column>
 | 
					 | 
				
			||||||
    <index id="71" parent="11" name="sqlite_autoindex_note_images_1">
 | 
					 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>noteImageId</ColNames>
 | 
					      <ColNames>noteImageId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="72" parent="11" name="IDX_note_images_noteId_imageId">
 | 
					    <index id="80" parent="11" name="IDX_note_images_noteId_imageId">
 | 
				
			||||||
      <ColNames>noteId
 | 
					      <ColNames>noteId
 | 
				
			||||||
imageId</ColNames>
 | 
					imageId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="73" parent="11" name="IDX_note_images_noteId">
 | 
					    <index id="81" parent="11" name="IDX_note_images_noteId">
 | 
				
			||||||
      <ColNames>noteId</ColNames>
 | 
					      <ColNames>noteId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="74" parent="11" name="IDX_note_images_imageId">
 | 
					    <index id="82" parent="11" name="IDX_note_images_imageId">
 | 
				
			||||||
      <ColNames>imageId</ColNames>
 | 
					      <ColNames>imageId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="75" parent="11">
 | 
					    <key id="83" parent="11">
 | 
				
			||||||
      <ColNames>noteImageId</ColNames>
 | 
					      <ColNames>noteImageId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_note_images_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_note_images_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="76" parent="12" name="noteRevisionId">
 | 
					    <column id="84" parent="12" name="noteRevisionId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="77" parent="12" name="noteId">
 | 
					    <column id="85" parent="12" name="noteId">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="78" parent="12" name="title">
 | 
					    <column id="86" parent="12" name="title">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="79" parent="12" name="content">
 | 
					    <column id="87" parent="12" name="content">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="80" parent="12" name="isProtected">
 | 
					    <column id="88" parent="12" name="isProtected">
 | 
				
			||||||
      <Position>5</Position>
 | 
					      <Position>5</Position>
 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="81" parent="12" name="dateModifiedFrom">
 | 
					    <column id="89" parent="12" name="dateModifiedFrom">
 | 
				
			||||||
      <Position>6</Position>
 | 
					      <Position>6</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="82" parent="12" name="dateModifiedTo">
 | 
					    <column id="90" parent="12" name="dateModifiedTo">
 | 
				
			||||||
      <Position>7</Position>
 | 
					      <Position>7</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="83" parent="12" name="type">
 | 
					    <column id="91" parent="12" name="type">
 | 
				
			||||||
      <Position>8</Position>
 | 
					      <Position>8</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>''</DefaultExpression>
 | 
					      <DefaultExpression>''</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="84" parent="12" name="mime">
 | 
					    <column id="92" parent="12" name="mime">
 | 
				
			||||||
      <Position>9</Position>
 | 
					      <Position>9</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>''</DefaultExpression>
 | 
					      <DefaultExpression>''</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="85" parent="12" name="sqlite_autoindex_note_revisions_1">
 | 
					    <column id="93" parent="12" name="hash">
 | 
				
			||||||
 | 
					      <Position>10</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <index id="94" parent="12" name="sqlite_autoindex_note_revisions_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>noteRevisionId</ColNames>
 | 
					      <ColNames>noteRevisionId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="86" parent="12" name="IDX_note_revisions_noteId">
 | 
					    <index id="95" parent="12" name="IDX_note_revisions_noteId">
 | 
				
			||||||
      <ColNames>noteId</ColNames>
 | 
					      <ColNames>noteId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="87" parent="12" name="IDX_note_revisions_dateModifiedFrom">
 | 
					    <index id="96" parent="12" name="IDX_note_revisions_dateModifiedFrom">
 | 
				
			||||||
      <ColNames>dateModifiedFrom</ColNames>
 | 
					      <ColNames>dateModifiedFrom</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="88" parent="12" name="IDX_note_revisions_dateModifiedTo">
 | 
					    <index id="97" parent="12" name="IDX_note_revisions_dateModifiedTo">
 | 
				
			||||||
      <ColNames>dateModifiedTo</ColNames>
 | 
					      <ColNames>dateModifiedTo</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="89" parent="12">
 | 
					    <key id="98" parent="12">
 | 
				
			||||||
      <ColNames>noteRevisionId</ColNames>
 | 
					      <ColNames>noteRevisionId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_note_revisions_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_note_revisions_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="90" parent="13" name="noteId">
 | 
					    <column id="99" parent="13" name="noteId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="91" parent="13" name="title">
 | 
					    <column id="100" parent="13" name="title">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>"unnamed"</DefaultExpression>
 | 
					      <DefaultExpression>"unnamed"</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="92" parent="13" name="content">
 | 
					    <column id="101" parent="13" name="content">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>""</DefaultExpression>
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="93" parent="13" name="isProtected">
 | 
					    <column id="102" parent="13" name="isProtected">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="94" parent="13" name="isDeleted">
 | 
					    <column id="103" parent="13" name="isDeleted">
 | 
				
			||||||
      <Position>5</Position>
 | 
					      <Position>5</Position>
 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="95" parent="13" name="dateCreated">
 | 
					    <column id="104" parent="13" name="dateCreated">
 | 
				
			||||||
      <Position>6</Position>
 | 
					      <Position>6</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="96" parent="13" name="dateModified">
 | 
					    <column id="105" parent="13" name="dateModified">
 | 
				
			||||||
      <Position>7</Position>
 | 
					      <Position>7</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="97" parent="13" name="type">
 | 
					    <column id="106" parent="13" name="type">
 | 
				
			||||||
      <Position>8</Position>
 | 
					      <Position>8</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>'text'</DefaultExpression>
 | 
					      <DefaultExpression>'text'</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="98" parent="13" name="mime">
 | 
					    <column id="107" parent="13" name="mime">
 | 
				
			||||||
      <Position>9</Position>
 | 
					      <Position>9</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>'text/html'</DefaultExpression>
 | 
					      <DefaultExpression>'text/html'</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="99" parent="13" name="sqlite_autoindex_notes_1">
 | 
					    <column id="108" parent="13" name="hash">
 | 
				
			||||||
 | 
					      <Position>10</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <index id="109" parent="13" name="sqlite_autoindex_notes_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>noteId</ColNames>
 | 
					      <ColNames>noteId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="100" parent="13" name="IDX_notes_isDeleted">
 | 
					    <index id="110" parent="13" name="IDX_notes_type">
 | 
				
			||||||
      <ColNames>isDeleted</ColNames>
 | 
					      <ColNames>type</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="101" parent="13">
 | 
					    <key id="111" parent="13">
 | 
				
			||||||
      <ColNames>noteId</ColNames>
 | 
					      <ColNames>noteId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_notes_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_notes_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="102" parent="14" name="name">
 | 
					    <column id="112" parent="14" name="name">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="103" parent="14" name="value">
 | 
					    <column id="113" parent="14" name="value">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="104" parent="14" name="dateModified">
 | 
					    <column id="114" parent="14" name="dateModified">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="105" parent="14" name="isSynced">
 | 
					    <column id="115" parent="14" name="isSynced">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>INTEGER|0s</DataType>
 | 
					      <DataType>INTEGER|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <DefaultExpression>0</DefaultExpression>
 | 
					      <DefaultExpression>0</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="106" parent="14" name="sqlite_autoindex_options_1">
 | 
					    <column id="116" parent="14" name="hash">
 | 
				
			||||||
 | 
					      <Position>5</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="117" parent="14" name="dateCreated">
 | 
				
			||||||
 | 
					      <Position>6</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>'1970-01-01T00:00:00.000Z'</DefaultExpression>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <index id="118" parent="14" name="sqlite_autoindex_options_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>name</ColNames>
 | 
					      <ColNames>name</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="107" parent="14">
 | 
					    <key id="119" parent="14">
 | 
				
			||||||
      <ColNames>name</ColNames>
 | 
					      <ColNames>name</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_options_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_options_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="108" parent="15" name="branchId">
 | 
					    <column id="120" parent="15" name="branchId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="109" parent="15" name="notePath">
 | 
					    <column id="121" parent="15" name="notePath">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="110" parent="15" name="dateAccessed">
 | 
					    <column id="122" parent="15" name="hash">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					      <DefaultExpression>""</DefaultExpression>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="111" parent="15" name="isDeleted">
 | 
					    <column id="123" parent="15" name="dateCreated">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
 | 
					    </column>
 | 
				
			||||||
 | 
					    <column id="124" parent="15" name="isDeleted">
 | 
				
			||||||
 | 
					      <Position>5</Position>
 | 
				
			||||||
      <DataType>INT|0s</DataType>
 | 
					      <DataType>INT|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="112" parent="15" name="sqlite_autoindex_recent_notes_1">
 | 
					    <index id="125" parent="15" name="sqlite_autoindex_recent_notes_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>branchId</ColNames>
 | 
					      <ColNames>branchId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="113" parent="15">
 | 
					    <key id="126" parent="15">
 | 
				
			||||||
      <ColNames>branchId</ColNames>
 | 
					      <ColNames>branchId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_recent_notes_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_recent_notes_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="114" parent="16" name="sourceId">
 | 
					    <column id="127" parent="16" name="sourceId">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="115" parent="16" name="dateCreated">
 | 
					    <column id="128" parent="16" name="dateCreated">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="116" parent="16" name="sqlite_autoindex_source_ids_1">
 | 
					    <index id="129" parent="16" name="sqlite_autoindex_source_ids_1">
 | 
				
			||||||
      <NameSurrogate>1</NameSurrogate>
 | 
					      <NameSurrogate>1</NameSurrogate>
 | 
				
			||||||
      <ColNames>sourceId</ColNames>
 | 
					      <ColNames>sourceId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="117" parent="16">
 | 
					    <key id="130" parent="16">
 | 
				
			||||||
      <ColNames>sourceId</ColNames>
 | 
					      <ColNames>sourceId</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
      <UnderlyingIndexName>sqlite_autoindex_source_ids_1</UnderlyingIndexName>
 | 
					      <UnderlyingIndexName>sqlite_autoindex_source_ids_1</UnderlyingIndexName>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
    <column id="118" parent="17" name="type">
 | 
					    <column id="131" parent="17" name="type">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>text|0s</DataType>
 | 
					      <DataType>text|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="119" parent="17" name="name">
 | 
					    <column id="132" parent="17" name="name">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>text|0s</DataType>
 | 
					      <DataType>text|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="120" parent="17" name="tbl_name">
 | 
					    <column id="133" parent="17" name="tbl_name">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>text|0s</DataType>
 | 
					      <DataType>text|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="121" parent="17" name="rootpage">
 | 
					    <column id="134" parent="17" name="rootpage">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>integer|0s</DataType>
 | 
					      <DataType>integer|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="122" parent="17" name="sql">
 | 
					    <column id="135" parent="17" name="sql">
 | 
				
			||||||
      <Position>5</Position>
 | 
					      <Position>5</Position>
 | 
				
			||||||
      <DataType>text|0s</DataType>
 | 
					      <DataType>text|0s</DataType>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="123" parent="18" name="name">
 | 
					    <column id="136" parent="18" name="name">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="124" parent="18" name="seq">
 | 
					    <column id="137" parent="18" name="seq">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="125" parent="19" name="id">
 | 
					    <column id="138" parent="19" name="id">
 | 
				
			||||||
      <Position>1</Position>
 | 
					      <Position>1</Position>
 | 
				
			||||||
      <DataType>INTEGER|0s</DataType>
 | 
					      <DataType>INTEGER|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
      <SequenceIdentity>1</SequenceIdentity>
 | 
					      <SequenceIdentity>1</SequenceIdentity>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="126" parent="19" name="entityName">
 | 
					    <column id="139" parent="19" name="entityName">
 | 
				
			||||||
      <Position>2</Position>
 | 
					      <Position>2</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="127" parent="19" name="entityId">
 | 
					    <column id="140" parent="19" name="entityId">
 | 
				
			||||||
      <Position>3</Position>
 | 
					      <Position>3</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="128" parent="19" name="sourceId">
 | 
					    <column id="141" parent="19" name="sourceId">
 | 
				
			||||||
      <Position>4</Position>
 | 
					      <Position>4</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <column id="129" parent="19" name="syncDate">
 | 
					    <column id="142" parent="19" name="syncDate">
 | 
				
			||||||
      <Position>5</Position>
 | 
					      <Position>5</Position>
 | 
				
			||||||
      <DataType>TEXT|0s</DataType>
 | 
					      <DataType>TEXT|0s</DataType>
 | 
				
			||||||
      <NotNull>1</NotNull>
 | 
					      <NotNull>1</NotNull>
 | 
				
			||||||
    </column>
 | 
					    </column>
 | 
				
			||||||
    <index id="130" parent="19" name="IDX_sync_entityName_entityId">
 | 
					    <index id="143" parent="19" name="IDX_sync_entityName_entityId">
 | 
				
			||||||
      <ColNames>entityName
 | 
					      <ColNames>entityName
 | 
				
			||||||
entityId</ColNames>
 | 
					entityId</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
      <Unique>1</Unique>
 | 
					      <Unique>1</Unique>
 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <index id="131" parent="19" name="IDX_sync_syncDate">
 | 
					    <index id="144" parent="19" name="IDX_sync_syncDate">
 | 
				
			||||||
      <ColNames>syncDate</ColNames>
 | 
					      <ColNames>syncDate</ColNames>
 | 
				
			||||||
      <ColumnCollations></ColumnCollations>
 | 
					 | 
				
			||||||
    </index>
 | 
					    </index>
 | 
				
			||||||
    <key id="132" parent="19">
 | 
					    <key id="145" parent="19">
 | 
				
			||||||
      <ColNames>id</ColNames>
 | 
					      <ColNames>id</ColNames>
 | 
				
			||||||
      <Primary>1</Primary>
 | 
					      <Primary>1</Primary>
 | 
				
			||||||
    </key>
 | 
					    </key>
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										6
									
								
								.idea/sqldialects.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,6 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="UTF-8"?>
 | 
				
			||||||
 | 
					<project version="4">
 | 
				
			||||||
 | 
					  <component name="SqlDialectMappings">
 | 
				
			||||||
 | 
					    <file url="PROJECT" dialect="SQLite" />
 | 
				
			||||||
 | 
					  </component>
 | 
				
			||||||
 | 
					</project>
 | 
				
			||||||
							
								
								
									
										21
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					FROM node:8.11.2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					RUN apt-get update && apt-get install -y nasm
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Create app directory
 | 
				
			||||||
 | 
					WORKDIR /usr/src/app
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Install app dependencies
 | 
				
			||||||
 | 
					# A wildcard is used to ensure both package.json AND package-lock.json are copied
 | 
				
			||||||
 | 
					# where available (npm@5+)
 | 
				
			||||||
 | 
					COPY package*.json ./
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					RUN npm install --production
 | 
				
			||||||
 | 
					# If you are building your code for production
 | 
				
			||||||
 | 
					# RUN npm install --only=production
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Bundle app source
 | 
				
			||||||
 | 
					COPY . .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					EXPOSE 8080
 | 
				
			||||||
 | 
					CMD [ "node", "src/www" ]
 | 
				
			||||||
							
								
								
									
										8
									
								
								bin/build-docker.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						@@ -0,0 +1,8 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/env bash
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if [[ $# -eq 0 ]] ; then
 | 
				
			||||||
 | 
					    echo "Missing argument of new version"
 | 
				
			||||||
 | 
					    exit 1
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					sudo docker build -t zadam/trilium:latest -t zadam/trilium:$1 .
 | 
				
			||||||
							
								
								
									
										23
									
								
								bin/build-pkg.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						@@ -0,0 +1,23 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/env bash
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if [[ $# -eq 0 ]] ; then
 | 
				
			||||||
 | 
					    echo "Missing argument of new version"
 | 
				
			||||||
 | 
					    exit 1
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					VERSION=$1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					PKG_DIR=dist/trilium-linux-x64-server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mkdir $PKG_DIR
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pkg . --targets node8-linux-x64 --output ${PKG_DIR}/trilium
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					chmod +x ${PKG_DIR}/trilium
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					cp node_modules/sqlite3/lib/binding/node-v57-linux-x64/node_sqlite3.node ${PKG_DIR}/
 | 
				
			||||||
 | 
					cp node_modules/scrypt/build/Release/scrypt.node ${PKG_DIR}/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					cd dist
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					7z a trilium-linux-x64-${VERSION}-server.7z trilium-linux-x64-server
 | 
				
			||||||
							
								
								
									
										9
									
								
								bin/push-docker-image.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						@@ -0,0 +1,9 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/env bash
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if [[ $# -eq 0 ]] ; then
 | 
				
			||||||
 | 
					    echo "Missing argument of new version"
 | 
				
			||||||
 | 
					    exit 1
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					sudo docker push zadam/trilium:latest
 | 
				
			||||||
 | 
					sudo docker push zadam/trilium:$1
 | 
				
			||||||
@@ -47,6 +47,7 @@ bin/package.sh
 | 
				
			|||||||
LINUX_X64_BUILD=trilium-linux-x64-$VERSION.7z
 | 
					LINUX_X64_BUILD=trilium-linux-x64-$VERSION.7z
 | 
				
			||||||
LINUX_IA32_BUILD=trilium-linux-ia32-$VERSION.7z
 | 
					LINUX_IA32_BUILD=trilium-linux-ia32-$VERSION.7z
 | 
				
			||||||
WINDOWS_X64_BUILD=trilium-windows-x64-$VERSION.7z
 | 
					WINDOWS_X64_BUILD=trilium-windows-x64-$VERSION.7z
 | 
				
			||||||
 | 
					SERVER_BUILD=trilium-linux-x64-server.7z
 | 
				
			||||||
 | 
					
 | 
				
			||||||
echo "Creating release in GitHub"
 | 
					echo "Creating release in GitHub"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -75,4 +76,21 @@ github-release upload \
 | 
				
			|||||||
    --name "$WINDOWS_X64_BUILD" \
 | 
					    --name "$WINDOWS_X64_BUILD" \
 | 
				
			||||||
    --file "dist/$WINDOWS_X64_BUILD"
 | 
					    --file "dist/$WINDOWS_X64_BUILD"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo "Packaging server version"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bin/build-pkg.sh $VERSION
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					github-release upload \
 | 
				
			||||||
 | 
					    --tag $TAG \
 | 
				
			||||||
 | 
					    --name "$SERVER_BUILD" \
 | 
				
			||||||
 | 
					    --file "dist/$SERVER_BUILD"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo "Building docker image"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bin/build-docker.sh $VERSION
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo "Pushing docker image to dockerhub"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bin/push-docker-image.sh $VERSION
 | 
				
			||||||
 | 
					
 | 
				
			||||||
echo "Release finished!"
 | 
					echo "Release finished!"
 | 
				
			||||||
@@ -3,15 +3,10 @@
 | 
				
			|||||||
instanceName=
 | 
					instanceName=
 | 
				
			||||||
 | 
					
 | 
				
			||||||
[Network]
 | 
					[Network]
 | 
				
			||||||
 | 
					# port setting is relevant only for web deployments, desktop builds run on random free port
 | 
				
			||||||
port=8080
 | 
					port=8080
 | 
				
			||||||
# true for TLS/SSL/HTTPS (secure), false for HTTP (unsecure).
 | 
					# true for TLS/SSL/HTTPS (secure), false for HTTP (unsecure).
 | 
				
			||||||
https=false
 | 
					https=false
 | 
				
			||||||
# path to certificate (run "bash generate-cert.sh" to generate self-signed certificate). Relevant only if https=true
 | 
					# path to certificate (run "bash generate-cert.sh" to generate self-signed certificate). Relevant only if https=true
 | 
				
			||||||
certPath=
 | 
					certPath=
 | 
				
			||||||
keyPath=
 | 
					keyPath=
 | 
				
			||||||
 | 
					 | 
				
			||||||
[Sync]
 | 
					 | 
				
			||||||
syncServerHost=
 | 
					 | 
				
			||||||
syncServerTimeout=10000
 | 
					 | 
				
			||||||
syncProxy=
 | 
					 | 
				
			||||||
syncServerCertificate=
 | 
					 | 
				
			||||||
							
								
								
									
										30
									
								
								db/migrations/0094__unify_auditing_fields.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,30 @@
 | 
				
			|||||||
 | 
					ALTER TABLE branches ADD dateCreated TEXT NOT NULL DEFAULT '1970-01-01T00:00:00.000Z';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CREATE TABLE `event_log_mig` (
 | 
				
			||||||
 | 
					  `id`	INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
 | 
				
			||||||
 | 
					  `noteId`	TEXT,
 | 
				
			||||||
 | 
					  `comment`	TEXT,
 | 
				
			||||||
 | 
					  `dateCreated`	TEXT NOT NULL
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO event_log_mig (id, noteId, comment, dateCreated)
 | 
				
			||||||
 | 
					SELECT id, noteId, comment, dateAdded FROM event_log;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					DROP TABLE event_log;
 | 
				
			||||||
 | 
					ALTER TABLE event_log_mig RENAME TO event_log;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ALTER TABLE options ADD dateCreated TEXT NOT NULL DEFAULT '1970-01-01T00:00:00.000Z';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CREATE TABLE `recent_notes_mig` (
 | 
				
			||||||
 | 
					  `branchId` TEXT NOT NULL PRIMARY KEY,
 | 
				
			||||||
 | 
					  `notePath` TEXT NOT NULL,
 | 
				
			||||||
 | 
					  hash TEXT DEFAULT "" NOT NULL,
 | 
				
			||||||
 | 
					  `dateCreated` TEXT NOT NULL,
 | 
				
			||||||
 | 
					  isDeleted INT
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO recent_notes_mig (branchId, notePath, hash, dateCreated, isDeleted)
 | 
				
			||||||
 | 
					SELECT branchId, notePath, hash, dateAccessed, isDeleted FROM recent_notes;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					DROP TABLE recent_notes;
 | 
				
			||||||
 | 
					ALTER TABLE recent_notes_mig RENAME TO recent_notes;
 | 
				
			||||||
							
								
								
									
										1
									
								
								db/migrations/0095__mime_type_for_render.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					UPDATE notes SET mime = 'text/html' WHERE type = 'render';
 | 
				
			||||||
							
								
								
									
										29
									
								
								db/migrations/0096__unify_surrogate_keys.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,29 @@
 | 
				
			|||||||
 | 
					CREATE TABLE `event_log_mig` (
 | 
				
			||||||
 | 
					  `eventId`	TEXT NOT NULL PRIMARY KEY,
 | 
				
			||||||
 | 
					  `noteId`	TEXT,
 | 
				
			||||||
 | 
					  `comment`	TEXT,
 | 
				
			||||||
 | 
					  `dateCreated`	TEXT NOT NULL
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO event_log_mig (eventId, noteId, comment, dateCreated)
 | 
				
			||||||
 | 
					SELECT id, noteId, comment, dateCreated FROM event_log;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					DROP TABLE event_log;
 | 
				
			||||||
 | 
					ALTER TABLE event_log_mig RENAME TO event_log;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					create table options_mig
 | 
				
			||||||
 | 
					(
 | 
				
			||||||
 | 
					  optionId TEXT NOT NULL PRIMARY KEY,
 | 
				
			||||||
 | 
					  name TEXT not null,
 | 
				
			||||||
 | 
					  value TEXT,
 | 
				
			||||||
 | 
					  dateModified INT,
 | 
				
			||||||
 | 
					  isSynced INTEGER default 0 not null,
 | 
				
			||||||
 | 
					  hash TEXT default "" not null,
 | 
				
			||||||
 | 
					  dateCreated TEXT default '1970-01-01T00:00:00.000Z' not null
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO options_mig (optionId, name, value, dateModified, isSynced, hash, dateCreated)
 | 
				
			||||||
 | 
					  SELECT name || "_key", name, value, dateModified, isSynced, hash, dateCreated FROM options;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					DROP TABLE options;
 | 
				
			||||||
 | 
					ALTER TABLE options_mig RENAME TO options;
 | 
				
			||||||
							
								
								
									
										2
									
								
								db/migrations/0097__add_zoomFactor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					INSERT INTO options (optionId, name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('zoomFactor_key', 'zoomFactor', '1.0', '2018-06-01T03:35:55.041Z', '2018-06-01T03:35:55.041Z', 0);
 | 
				
			||||||
							
								
								
									
										1
									
								
								db/migrations/0098__rename_hideInAutocomplete.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					UPDATE labels SET name = 'archived' WHERE name = 'hideInAutocomplete'
 | 
				
			||||||
							
								
								
									
										2
									
								
								db/migrations/0099__add_theme_option.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					INSERT INTO options (optionId, name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('theme_key', 'theme', 'white', '2018-06-01T03:35:55.041Z', '2018-06-01T03:35:55.041Z', 0);
 | 
				
			||||||
							
								
								
									
										15
									
								
								db/migrations/0100__remove_optionId.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,15 @@
 | 
				
			|||||||
 | 
					create table options_mig
 | 
				
			||||||
 | 
					(
 | 
				
			||||||
 | 
					  name TEXT not null PRIMARY KEY,
 | 
				
			||||||
 | 
					  value TEXT,
 | 
				
			||||||
 | 
					  dateModified INT,
 | 
				
			||||||
 | 
					  isSynced INTEGER default 0 not null,
 | 
				
			||||||
 | 
					  hash TEXT default "" not null,
 | 
				
			||||||
 | 
					  dateCreated TEXT default '1970-01-01T00:00:00.000Z' not null
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO options_mig (name, value, dateModified, isSynced, hash, dateCreated)
 | 
				
			||||||
 | 
					SELECT name, value, dateModified, isSynced, hash, dateCreated FROM options;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					DROP TABLE options;
 | 
				
			||||||
 | 
					ALTER TABLE options_mig RENAME TO options;
 | 
				
			||||||
							
								
								
									
										8
									
								
								db/migrations/0101__add_sync_options.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,8 @@
 | 
				
			|||||||
 | 
					INSERT INTO options (name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('syncServerHost', '', '2018-06-01T03:35:55.041Z', '2018-06-01T03:35:55.041Z', 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO options (name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('syncServerTimeout', '5000', '2018-06-01T03:35:55.041Z', '2018-06-01T03:35:55.041Z', 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO options (name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('syncProxy', '', '2018-06-01T03:35:55.041Z', '2018-06-01T03:35:55.041Z', 0);
 | 
				
			||||||
							
								
								
									
										2
									
								
								db/migrations/0102__fix_sync_entityIds.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					DELETE FROM sync WHERE entityName = 'note_tree';
 | 
				
			||||||
 | 
					DELETE FROM sync WHERE entityName = 'attributes';
 | 
				
			||||||
							
								
								
									
										2
									
								
								db/migrations/0103__add_initialized_option.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					INSERT INTO options (name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('initialized', 'true', '2018-06-01T03:35:55.041Z', '2018-06-01T03:35:55.041Z', 0);
 | 
				
			||||||
							
								
								
									
										4
									
								
								db/migrations/0104__fill_sync_rows_for_options.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,4 @@
 | 
				
			|||||||
 | 
					const syncTableService = require('../../src/services/sync_table');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// options has not been filled so far which caused problems with clean-slate sync.
 | 
				
			||||||
 | 
					module.exports = async () => await syncTableService.fillAllSyncRows();
 | 
				
			||||||
@@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					UPDATE notes SET content = '' WHERE isDeleted = 1;
 | 
				
			||||||
 | 
					UPDATE note_revisions SET content = '' WHERE (SELECT isDeleted FROM notes WHERE noteId = note_revisions.noteId) = 1;
 | 
				
			||||||
							
								
								
									
										15
									
								
								db/migrations/0106__add_relations_table.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,15 @@
 | 
				
			|||||||
 | 
					CREATE TABLE relations
 | 
				
			||||||
 | 
					(
 | 
				
			||||||
 | 
					    relationId  TEXT not null primary key,
 | 
				
			||||||
 | 
					    sourceNoteId       TEXT not null,
 | 
				
			||||||
 | 
					    name         TEXT not null,
 | 
				
			||||||
 | 
					    targetNoteId        TEXT not null,
 | 
				
			||||||
 | 
					    position     INT  default 0 not null,
 | 
				
			||||||
 | 
					    dateCreated  TEXT not null,
 | 
				
			||||||
 | 
					    dateModified TEXT not null,
 | 
				
			||||||
 | 
					    isDeleted    INT  not null
 | 
				
			||||||
 | 
					  , hash TEXT DEFAULT "" NOT NULL);
 | 
				
			||||||
 | 
					CREATE INDEX IDX_relation_sourceNoteId
 | 
				
			||||||
 | 
					  on relations (sourceNoteId);
 | 
				
			||||||
 | 
					CREATE INDEX IDX_relation_targetNoteId
 | 
				
			||||||
 | 
					  on relations (targetNoteId);
 | 
				
			||||||
@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					ALTER TABLE relations ADD isInheritable int DEFAULT 0 NULL;
 | 
				
			||||||
							
								
								
									
										9
									
								
								db/migrations/0108__new_backup_options.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,9 @@
 | 
				
			|||||||
 | 
					UPDATE options SET name = 'lastDailyBackupDate' WHERE name = 'lastBackupDate';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO options (name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('lastWeeklyBackupDate', '2018-07-29T18:31:00.874Z', '2018-07-29T18:31:00.874Z', '2018-07-29T18:31:00.874Z', 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO options (name, value, dateCreated, dateModified, isSynced)
 | 
				
			||||||
 | 
					VALUES ('lastMonthlyBackupDate', '2018-07-29T18:31:00.874Z', '2018-07-29T18:31:00.874Z', '2018-07-29T18:31:00.874Z', 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- these options are not synced so no need to fix sync rows
 | 
				
			||||||
							
								
								
									
										27
									
								
								db/migrations/0109__create_attributes.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,27 @@
 | 
				
			|||||||
 | 
					create table attributes
 | 
				
			||||||
 | 
					(
 | 
				
			||||||
 | 
					  attributeId      TEXT not null primary key,
 | 
				
			||||||
 | 
					  noteId       TEXT not null,
 | 
				
			||||||
 | 
					  type         TEXT not null,
 | 
				
			||||||
 | 
					  name         TEXT not null,
 | 
				
			||||||
 | 
					  value        TEXT default '' not null,
 | 
				
			||||||
 | 
					  position     INT  default 0 not null,
 | 
				
			||||||
 | 
					  dateCreated  TEXT not null,
 | 
				
			||||||
 | 
					  dateModified TEXT not null,
 | 
				
			||||||
 | 
					  isDeleted    INT  not null,
 | 
				
			||||||
 | 
					  hash         TEXT default "" not null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					create index IDX_attributes_name_value
 | 
				
			||||||
 | 
					  on attributes (name, value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					create index IDX_attributes_value
 | 
				
			||||||
 | 
					  on attributes (value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					create index IDX_attributes_noteId
 | 
				
			||||||
 | 
					  on attributes (noteId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO attributes (attributeId, noteId, type, name, value, position, dateCreated, dateModified, isDeleted, hash)
 | 
				
			||||||
 | 
					SELECT labelId, noteId, 'label', name, value, position, dateCreated, dateModified, isDeleted, hash FROM labels;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					INSERT INTO attributes (attributeId, noteId, type, name, value, position, dateCreated, dateModified, isDeleted, hash)
 | 
				
			||||||
 | 
					SELECT relationId, sourceNoteId, 'relation', name, targetNoteId, position, dateCreated, dateModified, isDeleted, hash FROM relations;
 | 
				
			||||||
							
								
								
									
										1
									
								
								db/migrations/0110__add_isInheritable_to_attributes.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					ALTER TABLE attributes ADD isInheritable int DEFAULT 0 NULL;
 | 
				
			||||||
							
								
								
									
										4
									
								
								db/migrations/0111__cleanup_labels_and_relations.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,4 @@
 | 
				
			|||||||
 | 
					DROP TABLE relations;
 | 
				
			||||||
 | 
					DROP TABLE labels;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					DELETE FROM sync WHERE entityName = 'relations' OR entityName = 'labels';
 | 
				
			||||||
@@ -1,8 +1,3 @@
 | 
				
			|||||||
CREATE TABLE IF NOT EXISTS "options" (
 | 
					 | 
				
			||||||
    `name`	TEXT NOT NULL PRIMARY KEY,
 | 
					 | 
				
			||||||
    `value`	TEXT,
 | 
					 | 
				
			||||||
    `dateModified` INT,
 | 
					 | 
				
			||||||
    isSynced INTEGER NOT NULL DEFAULT 0);
 | 
					 | 
				
			||||||
CREATE TABLE IF NOT EXISTS "sync" (
 | 
					CREATE TABLE IF NOT EXISTS "sync" (
 | 
				
			||||||
  `id`	INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
 | 
					  `id`	INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
 | 
				
			||||||
  `entityName`	TEXT NOT NULL,
 | 
					  `entityName`	TEXT NOT NULL,
 | 
				
			||||||
@@ -29,7 +24,7 @@ CREATE TABLE IF NOT EXISTS "note_revisions" (
 | 
				
			|||||||
  `isProtected`	INT NOT NULL DEFAULT 0,
 | 
					  `isProtected`	INT NOT NULL DEFAULT 0,
 | 
				
			||||||
  `dateModifiedFrom` TEXT NOT NULL,
 | 
					  `dateModifiedFrom` TEXT NOT NULL,
 | 
				
			||||||
  `dateModifiedTo` TEXT NOT NULL
 | 
					  `dateModifiedTo` TEXT NOT NULL
 | 
				
			||||||
, type TEXT DEFAULT '' NOT NULL, mime TEXT DEFAULT '' NOT NULL);
 | 
					, type TEXT DEFAULT '' NOT NULL, mime TEXT DEFAULT '' NOT NULL, hash TEXT DEFAULT "" NOT NULL);
 | 
				
			||||||
CREATE INDEX `IDX_note_revisions_noteId` ON `note_revisions` (
 | 
					CREATE INDEX `IDX_note_revisions_noteId` ON `note_revisions` (
 | 
				
			||||||
  `noteId`
 | 
					  `noteId`
 | 
				
			||||||
);
 | 
					);
 | 
				
			||||||
@@ -49,7 +44,7 @@ CREATE TABLE IF NOT EXISTS "images"
 | 
				
			|||||||
  isDeleted INT NOT NULL DEFAULT 0,
 | 
					  isDeleted INT NOT NULL DEFAULT 0,
 | 
				
			||||||
  dateModified TEXT NOT NULL,
 | 
					  dateModified TEXT NOT NULL,
 | 
				
			||||||
  dateCreated TEXT NOT NULL
 | 
					  dateCreated TEXT NOT NULL
 | 
				
			||||||
);
 | 
					, hash TEXT DEFAULT "" NOT NULL);
 | 
				
			||||||
CREATE TABLE note_images
 | 
					CREATE TABLE note_images
 | 
				
			||||||
(
 | 
					(
 | 
				
			||||||
  noteImageId TEXT PRIMARY KEY NOT NULL,
 | 
					  noteImageId TEXT PRIMARY KEY NOT NULL,
 | 
				
			||||||
@@ -58,7 +53,7 @@ CREATE TABLE note_images
 | 
				
			|||||||
  isDeleted INT NOT NULL DEFAULT 0,
 | 
					  isDeleted INT NOT NULL DEFAULT 0,
 | 
				
			||||||
  dateModified TEXT NOT NULL,
 | 
					  dateModified TEXT NOT NULL,
 | 
				
			||||||
  dateCreated TEXT NOT NULL
 | 
					  dateCreated TEXT NOT NULL
 | 
				
			||||||
);
 | 
					, hash TEXT DEFAULT "" NOT NULL);
 | 
				
			||||||
CREATE INDEX IDX_note_images_noteId ON note_images (noteId);
 | 
					CREATE INDEX IDX_note_images_noteId ON note_images (noteId);
 | 
				
			||||||
CREATE INDEX IDX_note_images_imageId ON note_images (imageId);
 | 
					CREATE INDEX IDX_note_images_imageId ON note_images (imageId);
 | 
				
			||||||
CREATE INDEX IDX_note_images_noteId_imageId ON note_images (noteId, imageId);
 | 
					CREATE INDEX IDX_note_images_noteId_imageId ON note_images (noteId, imageId);
 | 
				
			||||||
@@ -68,7 +63,7 @@ CREATE TABLE IF NOT EXISTS "api_tokens"
 | 
				
			|||||||
  token TEXT NOT NULL,
 | 
					  token TEXT NOT NULL,
 | 
				
			||||||
  dateCreated TEXT NOT NULL,
 | 
					  dateCreated TEXT NOT NULL,
 | 
				
			||||||
  isDeleted INT NOT NULL DEFAULT 0
 | 
					  isDeleted INT NOT NULL DEFAULT 0
 | 
				
			||||||
);
 | 
					, hash TEXT DEFAULT "" NOT NULL);
 | 
				
			||||||
CREATE TABLE IF NOT EXISTS "branches" (
 | 
					CREATE TABLE IF NOT EXISTS "branches" (
 | 
				
			||||||
  `branchId`	TEXT NOT NULL,
 | 
					  `branchId`	TEXT NOT NULL,
 | 
				
			||||||
  `noteId`	TEXT NOT NULL,
 | 
					  `noteId`	TEXT NOT NULL,
 | 
				
			||||||
@@ -77,7 +72,7 @@ CREATE TABLE IF NOT EXISTS "branches" (
 | 
				
			|||||||
  `prefix`	TEXT,
 | 
					  `prefix`	TEXT,
 | 
				
			||||||
  `isExpanded`	BOOLEAN,
 | 
					  `isExpanded`	BOOLEAN,
 | 
				
			||||||
  `isDeleted`	INTEGER NOT NULL DEFAULT 0,
 | 
					  `isDeleted`	INTEGER NOT NULL DEFAULT 0,
 | 
				
			||||||
  `dateModified`	TEXT NOT NULL,
 | 
					  `dateModified`	TEXT NOT NULL, hash TEXT DEFAULT "" NOT NULL, dateCreated TEXT NOT NULL DEFAULT '1970-01-01T00:00:00.000Z',
 | 
				
			||||||
  PRIMARY KEY(`branchId`)
 | 
					  PRIMARY KEY(`branchId`)
 | 
				
			||||||
);
 | 
					);
 | 
				
			||||||
CREATE INDEX `IDX_branches_noteId` ON `branches` (
 | 
					CREATE INDEX `IDX_branches_noteId` ON `branches` (
 | 
				
			||||||
@@ -87,34 +82,6 @@ CREATE INDEX `IDX_branches_noteId_parentNoteId` ON `branches` (
 | 
				
			|||||||
  `noteId`,
 | 
					  `noteId`,
 | 
				
			||||||
  `parentNoteId`
 | 
					  `parentNoteId`
 | 
				
			||||||
);
 | 
					);
 | 
				
			||||||
CREATE TABLE IF NOT EXISTS "recent_notes" (
 | 
					 | 
				
			||||||
  `branchId` TEXT NOT NULL PRIMARY KEY,
 | 
					 | 
				
			||||||
  `notePath` TEXT NOT NULL,
 | 
					 | 
				
			||||||
  `dateAccessed` TEXT NOT NULL,
 | 
					 | 
				
			||||||
  isDeleted INT
 | 
					 | 
				
			||||||
);
 | 
					 | 
				
			||||||
CREATE TABLE labels
 | 
					 | 
				
			||||||
(
 | 
					 | 
				
			||||||
  labelId  TEXT not null primary key,
 | 
					 | 
				
			||||||
  noteId       TEXT not null,
 | 
					 | 
				
			||||||
  name         TEXT not null,
 | 
					 | 
				
			||||||
  value        TEXT default '' not null,
 | 
					 | 
				
			||||||
  position     INT  default 0 not null,
 | 
					 | 
				
			||||||
  dateCreated  TEXT not null,
 | 
					 | 
				
			||||||
  dateModified TEXT not null,
 | 
					 | 
				
			||||||
  isDeleted    INT  not null
 | 
					 | 
				
			||||||
);
 | 
					 | 
				
			||||||
CREATE INDEX IDX_labels_name_value
 | 
					 | 
				
			||||||
  on labels (name, value);
 | 
					 | 
				
			||||||
CREATE INDEX IDX_labels_noteId
 | 
					 | 
				
			||||||
  on labels (noteId);
 | 
					 | 
				
			||||||
CREATE TABLE IF NOT EXISTS "event_log"
 | 
					 | 
				
			||||||
(
 | 
					 | 
				
			||||||
  id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
 | 
					 | 
				
			||||||
  noteId TEXT,
 | 
					 | 
				
			||||||
  comment TEXT,
 | 
					 | 
				
			||||||
  dateAdded TEXT NOT NULL
 | 
					 | 
				
			||||||
);
 | 
					 | 
				
			||||||
CREATE TABLE IF NOT EXISTS "notes" (
 | 
					CREATE TABLE IF NOT EXISTS "notes" (
 | 
				
			||||||
  `noteId`	TEXT NOT NULL,
 | 
					  `noteId`	TEXT NOT NULL,
 | 
				
			||||||
  `title`	TEXT NOT NULL DEFAULT "unnamed",
 | 
					  `title`	TEXT NOT NULL DEFAULT "unnamed",
 | 
				
			||||||
@@ -124,9 +91,43 @@ CREATE TABLE IF NOT EXISTS "notes" (
 | 
				
			|||||||
  `dateCreated`	TEXT NOT NULL,
 | 
					  `dateCreated`	TEXT NOT NULL,
 | 
				
			||||||
  `dateModified`	TEXT NOT NULL,
 | 
					  `dateModified`	TEXT NOT NULL,
 | 
				
			||||||
  type TEXT NOT NULL DEFAULT 'text',
 | 
					  type TEXT NOT NULL DEFAULT 'text',
 | 
				
			||||||
  mime TEXT NOT NULL DEFAULT 'text/html',
 | 
					  mime TEXT NOT NULL DEFAULT 'text/html', hash TEXT DEFAULT "" NOT NULL,
 | 
				
			||||||
  PRIMARY KEY(`noteId`)
 | 
					  PRIMARY KEY(`noteId`)
 | 
				
			||||||
);
 | 
					);
 | 
				
			||||||
CREATE INDEX `IDX_notes_isDeleted` ON `notes` (
 | 
					CREATE INDEX IDX_branches_parentNoteId ON branches (parentNoteId);
 | 
				
			||||||
  `isDeleted`
 | 
					CREATE INDEX IDX_notes_type
 | 
				
			||||||
 | 
					  on notes (type);
 | 
				
			||||||
 | 
					CREATE TABLE IF NOT EXISTS "recent_notes" (
 | 
				
			||||||
 | 
					  `branchId` TEXT NOT NULL PRIMARY KEY,
 | 
				
			||||||
 | 
					  `notePath` TEXT NOT NULL,
 | 
				
			||||||
 | 
					  hash TEXT DEFAULT "" NOT NULL,
 | 
				
			||||||
 | 
					  `dateCreated` TEXT NOT NULL,
 | 
				
			||||||
 | 
					  isDeleted INT
 | 
				
			||||||
);
 | 
					);
 | 
				
			||||||
 | 
					CREATE TABLE IF NOT EXISTS "event_log" (
 | 
				
			||||||
 | 
					  `eventId`	TEXT NOT NULL PRIMARY KEY,
 | 
				
			||||||
 | 
					  `noteId`	TEXT,
 | 
				
			||||||
 | 
					  `comment`	TEXT,
 | 
				
			||||||
 | 
					  `dateCreated`	TEXT NOT NULL
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					CREATE TABLE IF NOT EXISTS "options"
 | 
				
			||||||
 | 
					(
 | 
				
			||||||
 | 
					  name TEXT not null PRIMARY KEY,
 | 
				
			||||||
 | 
					  value TEXT,
 | 
				
			||||||
 | 
					  dateModified INT,
 | 
				
			||||||
 | 
					  isSynced INTEGER default 0 not null,
 | 
				
			||||||
 | 
					  hash TEXT default "" not null,
 | 
				
			||||||
 | 
					  dateCreated TEXT default '1970-01-01T00:00:00.000Z' not null
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					CREATE TABLE attributes
 | 
				
			||||||
 | 
					(
 | 
				
			||||||
 | 
					  attributeId      TEXT not null primary key,
 | 
				
			||||||
 | 
					  noteId       TEXT not null,
 | 
				
			||||||
 | 
					  type         TEXT not null,
 | 
				
			||||||
 | 
					  name         TEXT not null,
 | 
				
			||||||
 | 
					  value        TEXT default '' not null,
 | 
				
			||||||
 | 
					  position     INT  default 0 not null,
 | 
				
			||||||
 | 
					  dateCreated  TEXT not null,
 | 
				
			||||||
 | 
					  dateModified TEXT not null,
 | 
				
			||||||
 | 
					  isDeleted    INT  not null,
 | 
				
			||||||
 | 
					  hash         TEXT default "" not null, isInheritable int DEFAULT 0 NULL);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -2,9 +2,9 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const electron = require('electron');
 | 
					const electron = require('electron');
 | 
				
			||||||
const path = require('path');
 | 
					const path = require('path');
 | 
				
			||||||
const config = require('./src/services/config');
 | 
					 | 
				
			||||||
const log = require('./src/services/log');
 | 
					const log = require('./src/services/log');
 | 
				
			||||||
const url = require("url");
 | 
					const url = require("url");
 | 
				
			||||||
 | 
					const port = require('./src/services/port');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const app = electron.app;
 | 
					const app = electron.app;
 | 
				
			||||||
const globalShortcut = electron.globalShortcut;
 | 
					const globalShortcut = electron.globalShortcut;
 | 
				
			||||||
@@ -23,7 +23,7 @@ function onClosed() {
 | 
				
			|||||||
    mainWindow = null;
 | 
					    mainWindow = null;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function createMainWindow() {
 | 
					async function createMainWindow() {
 | 
				
			||||||
    const win = new electron.BrowserWindow({
 | 
					    const win = new electron.BrowserWindow({
 | 
				
			||||||
        width: 1200,
 | 
					        width: 1200,
 | 
				
			||||||
        height: 900,
 | 
					        height: 900,
 | 
				
			||||||
@@ -31,10 +31,8 @@ function createMainWindow() {
 | 
				
			|||||||
        icon: path.join(__dirname, 'src/public/images/app-icons/png/256x256.png')
 | 
					        icon: path.join(__dirname, 'src/public/images/app-icons/png/256x256.png')
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const port = config['Network']['port'] || '3000';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    win.setMenu(null);
 | 
					    win.setMenu(null);
 | 
				
			||||||
    win.loadURL('http://localhost:' + port);
 | 
					    win.loadURL('http://localhost:' + await port);
 | 
				
			||||||
    win.on('closed', onClosed);
 | 
					    win.on('closed', onClosed);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    win.webContents.on('new-window', (e, url) => {
 | 
					    win.webContents.on('new-window', (e, url) => {
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										5998
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
							
								
								
									
										50
									
								
								package.json
									
									
									
									
									
								
							
							
						
						@@ -1,9 +1,12 @@
 | 
				
			|||||||
{
 | 
					{
 | 
				
			||||||
  "name": "trilium",
 | 
					  "name": "trilium",
 | 
				
			||||||
  "description": "Trilium Notes",
 | 
					  "description": "Trilium Notes",
 | 
				
			||||||
  "version": "0.13.0-beta",
 | 
					  "version": "0.19.1",
 | 
				
			||||||
  "license": "AGPL-3.0-only",
 | 
					  "license": "AGPL-3.0-only",
 | 
				
			||||||
  "main": "electron.js",
 | 
					  "main": "electron.js",
 | 
				
			||||||
 | 
					  "bin": {
 | 
				
			||||||
 | 
					    "trilium": "./src/www"
 | 
				
			||||||
 | 
					  },
 | 
				
			||||||
  "repository": {
 | 
					  "repository": {
 | 
				
			||||||
    "type": "git",
 | 
					    "type": "git",
 | 
				
			||||||
    "url": "https://github.com/zadam/trilium.git"
 | 
					    "url": "https://github.com/zadam/trilium.git"
 | 
				
			||||||
@@ -28,23 +31,24 @@
 | 
				
			|||||||
    "debug": "~3.1.0",
 | 
					    "debug": "~3.1.0",
 | 
				
			||||||
    "devtron": "^1.4.0",
 | 
					    "devtron": "^1.4.0",
 | 
				
			||||||
    "ejs": "~2.6.1",
 | 
					    "ejs": "~2.6.1",
 | 
				
			||||||
    "electron-debug": "^1.5.0",
 | 
					    "electron-debug": "^2.0.0",
 | 
				
			||||||
    "electron-dl": "^1.12.0",
 | 
					    "electron-dl": "^1.12.0",
 | 
				
			||||||
    "electron-in-page-search": "^1.3.2",
 | 
					    "electron-in-page-search": "^1.3.2",
 | 
				
			||||||
    "express": "~4.16.3",
 | 
					    "express": "~4.16.3",
 | 
				
			||||||
    "express-session": "^1.15.6",
 | 
					    "express-session": "^1.15.6",
 | 
				
			||||||
    "fs-extra": "^6.0.1",
 | 
					    "fs-extra": "^7.0.0",
 | 
				
			||||||
    "helmet": "^3.12.1",
 | 
					    "get-port": "^4.0.0",
 | 
				
			||||||
 | 
					    "helmet": "^3.13.0",
 | 
				
			||||||
    "html": "^1.0.0",
 | 
					    "html": "^1.0.0",
 | 
				
			||||||
    "image-type": "^3.0.0",
 | 
					    "image-type": "^3.0.0",
 | 
				
			||||||
    "imagemin": "^5.3.1",
 | 
					    "imagemin": "^6.0.0",
 | 
				
			||||||
    "imagemin-giflossy": "^5.1.10",
 | 
					    "imagemin-giflossy": "^5.1.10",
 | 
				
			||||||
    "imagemin-mozjpeg": "^7.0.0",
 | 
					    "imagemin-mozjpeg": "^7.0.0",
 | 
				
			||||||
    "imagemin-pngquant": "^5.1.0",
 | 
					    "imagemin-pngquant": "^6.0.0",
 | 
				
			||||||
    "ini": "^1.3.5",
 | 
					    "ini": "^1.3.5",
 | 
				
			||||||
    "jimp": "^0.2.28",
 | 
					    "jimp": "^0.3.0",
 | 
				
			||||||
    "moment": "^2.22.1",
 | 
					    "moment": "^2.22.2",
 | 
				
			||||||
    "multer": "^1.3.0",
 | 
					    "multer": "^1.3.1",
 | 
				
			||||||
    "open": "0.0.5",
 | 
					    "open": "0.0.5",
 | 
				
			||||||
    "rand-token": "^0.4.0",
 | 
					    "rand-token": "^0.4.0",
 | 
				
			||||||
    "rcedit": "^1.1.0",
 | 
					    "rcedit": "^1.1.0",
 | 
				
			||||||
@@ -59,17 +63,19 @@
 | 
				
			|||||||
    "sqlite": "^2.9.2",
 | 
					    "sqlite": "^2.9.2",
 | 
				
			||||||
    "tar-stream": "^1.6.1",
 | 
					    "tar-stream": "^1.6.1",
 | 
				
			||||||
    "unescape": "^1.0.1",
 | 
					    "unescape": "^1.0.1",
 | 
				
			||||||
    "ws": "^5.2.0"
 | 
					    "ws": "^6.0.0",
 | 
				
			||||||
 | 
					    "xml2js": "^0.4.19"
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "devDependencies": {
 | 
					  "devDependencies": {
 | 
				
			||||||
    "electron": "^2.0.1",
 | 
					    "electron": "^2.0.6",
 | 
				
			||||||
    "electron-compile": "^6.4.2",
 | 
					    "electron-compile": "^6.4.3",
 | 
				
			||||||
    "electron-packager": "^12.1.0",
 | 
					    "electron-packager": "^12.1.0",
 | 
				
			||||||
    "electron-prebuilt-compile": "2.0.0",
 | 
					    "electron-prebuilt-compile": "2.0.6",
 | 
				
			||||||
    "electron-rebuild": "^1.7.3",
 | 
					    "electron-rebuild": "^1.8.2",
 | 
				
			||||||
    "lorem-ipsum": "^1.0.4",
 | 
					    "lorem-ipsum": "^1.0.5",
 | 
				
			||||||
    "tape": "^4.9.0",
 | 
					    "tape": "^4.9.1",
 | 
				
			||||||
    "xo": "^0.21.1"
 | 
					    "xo": "^0.21.1",
 | 
				
			||||||
 | 
					    "pkg": "^4.3.3"
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "config": {
 | 
					  "config": {
 | 
				
			||||||
    "forge": {
 | 
					    "forge": {
 | 
				
			||||||
@@ -108,5 +114,15 @@
 | 
				
			|||||||
      "node",
 | 
					      "node",
 | 
				
			||||||
      "browser"
 | 
					      "browser"
 | 
				
			||||||
    ]
 | 
					    ]
 | 
				
			||||||
 | 
					  },
 | 
				
			||||||
 | 
					  "pkg": {
 | 
				
			||||||
 | 
					    "assets": [
 | 
				
			||||||
 | 
					      "./db/**/*",
 | 
				
			||||||
 | 
					      "./src/public/**/*",
 | 
				
			||||||
 | 
					      "./src/views/**/*",
 | 
				
			||||||
 | 
					      "./node_modules/mozjpeg/vendor/*",
 | 
				
			||||||
 | 
					      "./node_modules/pngquant-bin/vendor/*",
 | 
				
			||||||
 | 
					      "./node_modules/giflossy/vendor/*"
 | 
				
			||||||
 | 
					    ]
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -11,6 +11,7 @@ const os = require('os');
 | 
				
			|||||||
const sessionSecret = require('./services/session_secret');
 | 
					const sessionSecret = require('./services/session_secret');
 | 
				
			||||||
const cls = require('./services/cls');
 | 
					const cls = require('./services/cls');
 | 
				
			||||||
require('./entities/entity_constructor');
 | 
					require('./entities/entity_constructor');
 | 
				
			||||||
 | 
					require('./services/handlers');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const app = express();
 | 
					const app = express();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -47,7 +48,7 @@ const sessionParser = session({
 | 
				
			|||||||
    cookie: {
 | 
					    cookie: {
 | 
				
			||||||
        //    path: "/",
 | 
					        //    path: "/",
 | 
				
			||||||
        httpOnly: true,
 | 
					        httpOnly: true,
 | 
				
			||||||
        maxAge:  1800000
 | 
					        maxAge:  24 * 60 * 60 * 1000 // in milliseconds
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    store: new FileStore({
 | 
					    store: new FileStore({
 | 
				
			||||||
        ttl: 30 * 24 * 3600,
 | 
					        ttl: 30 * 24 * 3600,
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -9,8 +9,6 @@ class ApiToken extends Entity {
 | 
				
			|||||||
    static get hashedProperties() { return ["apiTokenId", "token", "dateCreated", "isDeleted"]; }
 | 
					    static get hashedProperties() { return ["apiTokenId", "token", "dateCreated", "isDeleted"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    beforeSaving() {
 | 
					    beforeSaving() {
 | 
				
			||||||
        super.beforeSaving();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!this.isDeleted) {
 | 
					        if (!this.isDeleted) {
 | 
				
			||||||
            this.isDeleted = false;
 | 
					            this.isDeleted = false;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -18,6 +16,8 @@ class ApiToken extends Entity {
 | 
				
			|||||||
        if (!this.dateCreated) {
 | 
					        if (!this.dateCreated) {
 | 
				
			||||||
            this.dateCreated = dateUtils.nowDate();
 | 
					            this.dateCreated = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										77
									
								
								src/entities/attribute.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,77 @@
 | 
				
			|||||||
 | 
					"use strict";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const Entity = require('./entity');
 | 
				
			||||||
 | 
					const repository = require('../services/repository');
 | 
				
			||||||
 | 
					const dateUtils = require('../services/date_utils');
 | 
				
			||||||
 | 
					const sql = require('../services/sql');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Attribute extends Entity {
 | 
				
			||||||
 | 
					    static get tableName() { return "attributes"; }
 | 
				
			||||||
 | 
					    static get primaryKeyName() { return "attributeId"; }
 | 
				
			||||||
 | 
					    static get hashedProperties() { return ["attributeId", "noteId", "type", "name", "value", "isInheritable", "isDeleted", "dateCreated"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(row) {
 | 
				
			||||||
 | 
					        super(row);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.isInheritable = !!this.isInheritable;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.isDefinition()) {
 | 
				
			||||||
 | 
					            try {
 | 
				
			||||||
 | 
					                this.value = JSON.parse(this.value);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            catch (e) {
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async getNote() {
 | 
				
			||||||
 | 
					        return await repository.getEntity("SELECT * FROM notes WHERE noteId = ?", [this.noteId]);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async getTargetNote() {
 | 
				
			||||||
 | 
					        if (this.type !== 'relation') {
 | 
				
			||||||
 | 
					            throw new Error(`Attribute ${this.attributeId} is not relation`);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!this.value) {
 | 
				
			||||||
 | 
					            return null;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return await repository.getEntity("SELECT * FROM notes WHERE noteId = ?", [this.value]);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    isDefinition() {
 | 
				
			||||||
 | 
					        return this.type === 'label-definition' || this.type === 'relation-definition';
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async beforeSaving() {
 | 
				
			||||||
 | 
					        if (!this.value) {
 | 
				
			||||||
 | 
					            // null value isn't allowed
 | 
				
			||||||
 | 
					            this.value = "";
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.position === undefined) {
 | 
				
			||||||
 | 
					            this.position = 1 + await sql.getValue(`SELECT COALESCE(MAX(position), 0) FROM attributes WHERE noteId = ?`, [this.noteId]);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!this.isInheritable) {
 | 
				
			||||||
 | 
					            this.isInheritable = false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!this.isDeleted) {
 | 
				
			||||||
 | 
					            this.isDeleted = false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!this.dateCreated) {
 | 
				
			||||||
 | 
					            this.dateCreated = dateUtils.nowDate();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.isChanged) {
 | 
				
			||||||
 | 
					            this.dateModified = dateUtils.nowDate();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					module.exports = Attribute;
 | 
				
			||||||
@@ -9,15 +9,13 @@ class Branch extends Entity {
 | 
				
			|||||||
    static get tableName() { return "branches"; }
 | 
					    static get tableName() { return "branches"; }
 | 
				
			||||||
    static get primaryKeyName() { return "branchId"; }
 | 
					    static get primaryKeyName() { return "branchId"; }
 | 
				
			||||||
    // notePosition is not part of hash because it would produce a lot of updates in case of reordering
 | 
					    // notePosition is not part of hash because it would produce a lot of updates in case of reordering
 | 
				
			||||||
    static get hashedProperties() { return ["branchId", "noteId", "parentNoteId", "dateModified", "isDeleted", "prefix"]; }
 | 
					    static get hashedProperties() { return ["branchId", "noteId", "parentNoteId", "isDeleted", "prefix"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async getNote() {
 | 
					    async getNote() {
 | 
				
			||||||
        return await repository.getEntity("SELECT * FROM notes WHERE noteId = ?", [this.noteId]);
 | 
					        return await repository.getEntity("SELECT * FROM notes WHERE noteId = ?", [this.noteId]);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async beforeSaving() {
 | 
					    async beforeSaving() {
 | 
				
			||||||
        super.beforeSaving();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (this.notePosition === undefined) {
 | 
					        if (this.notePosition === undefined) {
 | 
				
			||||||
            const maxNotePos = await sql.getValue('SELECT MAX(notePosition) FROM branches WHERE parentNoteId = ? AND isDeleted = 0', [this.parentNoteId]);
 | 
					            const maxNotePos = await sql.getValue('SELECT MAX(notePosition) FROM branches WHERE parentNoteId = ? AND isDeleted = 0', [this.parentNoteId]);
 | 
				
			||||||
            this.notePosition = maxNotePos === null ? 0 : maxNotePos + 1;
 | 
					            this.notePosition = maxNotePos === null ? 0 : maxNotePos + 1;
 | 
				
			||||||
@@ -27,7 +25,15 @@ class Branch extends Entity {
 | 
				
			|||||||
            this.isDeleted = false;
 | 
					            this.isDeleted = false;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        this.dateModified = dateUtils.nowDate()
 | 
					        if (!this.dateCreated) {
 | 
				
			||||||
 | 
					            this.dateCreated = dateUtils.nowDate();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.isChanged) {
 | 
				
			||||||
 | 
					            this.dateModified = dateUtils.nowDate();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,13 +1,16 @@
 | 
				
			|||||||
"use strict";
 | 
					"use strict";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const utils = require('../services/utils');
 | 
					const utils = require('../services/utils');
 | 
				
			||||||
const repository = require('../services/repository');
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
class Entity {
 | 
					class Entity {
 | 
				
			||||||
    constructor(row = {}) {
 | 
					    constructor(row = {}) {
 | 
				
			||||||
        for (const key in row) {
 | 
					        for (const key in row) {
 | 
				
			||||||
            this[key] = row[key];
 | 
					            this[key] = row[key];
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if ('isDeleted' in this) {
 | 
				
			||||||
 | 
					            this.isDeleted = !!this.isDeleted;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    beforeSaving() {
 | 
					    beforeSaving() {
 | 
				
			||||||
@@ -15,20 +18,25 @@ class Entity {
 | 
				
			|||||||
            this[this.constructor.primaryKeyName] = utils.newEntityId();
 | 
					            this[this.constructor.primaryKeyName] = utils.newEntityId();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const origHash = this.hash;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.hash = this.generateHash();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.isChanged = origHash !== this.hash;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    generateHash() {
 | 
				
			||||||
        let contentToHash = "";
 | 
					        let contentToHash = "";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        for (const propertyName of this.constructor.hashedProperties) {
 | 
					        for (const propertyName of this.constructor.hashedProperties) {
 | 
				
			||||||
            contentToHash += "|" + this[propertyName];
 | 
					            contentToHash += "|" + this[propertyName];
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // this IF is to ease the migration from before hashed options, can be later removed
 | 
					        return utils.hash(contentToHash).substr(0, 10);
 | 
				
			||||||
        if (this.constructor.tableName !== 'options' || this.isSynced) {
 | 
					 | 
				
			||||||
            this["hash"] = utils.hash(contentToHash).substr(0, 10);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async save() {
 | 
					    async save() {
 | 
				
			||||||
        await repository.updateEntity(this);
 | 
					        await require('../services/repository').updateEntity(this);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return this;
 | 
					        return this;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,17 +3,37 @@ const NoteRevision = require('../entities/note_revision');
 | 
				
			|||||||
const Image = require('../entities/image');
 | 
					const Image = require('../entities/image');
 | 
				
			||||||
const NoteImage = require('../entities/note_image');
 | 
					const NoteImage = require('../entities/note_image');
 | 
				
			||||||
const Branch = require('../entities/branch');
 | 
					const Branch = require('../entities/branch');
 | 
				
			||||||
const Label = require('../entities/label');
 | 
					const Attribute = require('../entities/attribute');
 | 
				
			||||||
const RecentNote = require('../entities/recent_note');
 | 
					const RecentNote = require('../entities/recent_note');
 | 
				
			||||||
const ApiToken = require('../entities/api_token');
 | 
					const ApiToken = require('../entities/api_token');
 | 
				
			||||||
const Option = require('../entities/option');
 | 
					const Option = require('../entities/option');
 | 
				
			||||||
const repository = require('../services/repository');
 | 
					const repository = require('../services/repository');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const TABLE_NAME_TO_ENTITY = {
 | 
				
			||||||
 | 
					    "attributes": Attribute,
 | 
				
			||||||
 | 
					    "images": Image,
 | 
				
			||||||
 | 
					    "note_images": NoteImage,
 | 
				
			||||||
 | 
					    "branches": Branch,
 | 
				
			||||||
 | 
					    "notes": Note,
 | 
				
			||||||
 | 
					    "note_revisions": NoteRevision,
 | 
				
			||||||
 | 
					    "recent_notes": RecentNote,
 | 
				
			||||||
 | 
					    "options": Option,
 | 
				
			||||||
 | 
					    "api_tokens": ApiToken
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getEntityFromTableName(tableName) {
 | 
				
			||||||
 | 
					    if (!(tableName in TABLE_NAME_TO_ENTITY)) {
 | 
				
			||||||
 | 
					        throw new Error(`Entity for table ${tableName} not found!`);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return TABLE_NAME_TO_ENTITY[tableName];
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function createEntityFromRow(row) {
 | 
					function createEntityFromRow(row) {
 | 
				
			||||||
    let entity;
 | 
					    let entity;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (row.labelId) {
 | 
					    if (row.attributeId) {
 | 
				
			||||||
        entity = new Label(row);
 | 
					        entity = new Attribute(row);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    else if (row.noteRevisionId) {
 | 
					    else if (row.noteRevisionId) {
 | 
				
			||||||
        entity = new NoteRevision(row);
 | 
					        entity = new NoteRevision(row);
 | 
				
			||||||
@@ -46,8 +66,9 @@ function createEntityFromRow(row) {
 | 
				
			|||||||
    return entity;
 | 
					    return entity;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
repository.setEntityConstructor(createEntityFromRow);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
module.exports = {
 | 
					module.exports = {
 | 
				
			||||||
    createEntityFromRow
 | 
					    createEntityFromRow,
 | 
				
			||||||
 | 
					    getEntityFromTableName
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					repository.setEntityConstructor(module.exports);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,11 +6,9 @@ const dateUtils = require('../services/date_utils');
 | 
				
			|||||||
class Image extends Entity {
 | 
					class Image extends Entity {
 | 
				
			||||||
    static get tableName() { return "images"; }
 | 
					    static get tableName() { return "images"; }
 | 
				
			||||||
    static get primaryKeyName() { return "imageId"; }
 | 
					    static get primaryKeyName() { return "imageId"; }
 | 
				
			||||||
    static get hashedProperties() { return ["imageId", "format", "checksum", "name", "isDeleted", "dateModified", "dateCreated"]; }
 | 
					    static get hashedProperties() { return ["imageId", "format", "checksum", "name", "isDeleted", "dateCreated"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    beforeSaving() {
 | 
					    beforeSaving() {
 | 
				
			||||||
        super.beforeSaving();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!this.isDeleted) {
 | 
					        if (!this.isDeleted) {
 | 
				
			||||||
            this.isDeleted = false;
 | 
					            this.isDeleted = false;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -19,8 +17,12 @@ class Image extends Entity {
 | 
				
			|||||||
            this.dateCreated = dateUtils.nowDate();
 | 
					            this.dateCreated = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.isChanged) {
 | 
				
			||||||
            this.dateModified = dateUtils.nowDate();
 | 
					            this.dateModified = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
module.exports = Image;
 | 
					module.exports = Image;
 | 
				
			||||||
@@ -1,41 +0,0 @@
 | 
				
			|||||||
"use strict";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const Entity = require('./entity');
 | 
					 | 
				
			||||||
const repository = require('../services/repository');
 | 
					 | 
				
			||||||
const dateUtils = require('../services/date_utils');
 | 
					 | 
				
			||||||
const sql = require('../services/sql');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Label extends Entity {
 | 
					 | 
				
			||||||
    static get tableName() { return "labels"; }
 | 
					 | 
				
			||||||
    static get primaryKeyName() { return "labelId"; }
 | 
					 | 
				
			||||||
    static get hashedProperties() { return ["labelId", "noteId", "name", "value", "dateModified", "dateCreated"]; }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    async getNote() {
 | 
					 | 
				
			||||||
        return await repository.getEntity("SELECT * FROM notes WHERE noteId = ?", [this.noteId]);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    async beforeSaving() {
 | 
					 | 
				
			||||||
        super.beforeSaving();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!this.value) {
 | 
					 | 
				
			||||||
            // null value isn't allowed
 | 
					 | 
				
			||||||
            this.value = "";
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (this.position === undefined) {
 | 
					 | 
				
			||||||
            this.position = 1 + await sql.getValue(`SELECT COALESCE(MAX(position), 0) FROM labels WHERE noteId = ?`, [this.noteId]);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!this.isDeleted) {
 | 
					 | 
				
			||||||
            this.isDeleted = false;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!this.dateCreated) {
 | 
					 | 
				
			||||||
            this.dateCreated = dateUtils.nowDate();
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        this.dateModified = dateUtils.nowDate();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
module.exports = Label;
 | 
					 | 
				
			||||||
@@ -1,6 +1,7 @@
 | 
				
			|||||||
"use strict";
 | 
					"use strict";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const Entity = require('./entity');
 | 
					const Entity = require('./entity');
 | 
				
			||||||
 | 
					const Attribute = require('./attribute');
 | 
				
			||||||
const protectedSessionService = require('../services/protected_session');
 | 
					const protectedSessionService = require('../services/protected_session');
 | 
				
			||||||
const repository = require('../services/repository');
 | 
					const repository = require('../services/repository');
 | 
				
			||||||
const dateUtils = require('../services/date_utils');
 | 
					const dateUtils = require('../services/date_utils');
 | 
				
			||||||
@@ -8,11 +9,13 @@ const dateUtils = require('../services/date_utils');
 | 
				
			|||||||
class Note extends Entity {
 | 
					class Note extends Entity {
 | 
				
			||||||
    static get tableName() { return "notes"; }
 | 
					    static get tableName() { return "notes"; }
 | 
				
			||||||
    static get primaryKeyName() { return "noteId"; }
 | 
					    static get primaryKeyName() { return "noteId"; }
 | 
				
			||||||
    static get hashedProperties() { return ["noteId", "title", "content", "type", "dateModified", "isProtected", "isDeleted"]; }
 | 
					    static get hashedProperties() { return ["noteId", "title", "content", "type", "isProtected", "isDeleted"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    constructor(row) {
 | 
					    constructor(row) {
 | 
				
			||||||
        super(row);
 | 
					        super(row);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.isProtected = !!this.isProtected;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // check if there's noteId, otherwise this is a new entity which wasn't encrypted yet
 | 
					        // check if there's noteId, otherwise this is a new entity which wasn't encrypted yet
 | 
				
			||||||
        if (this.isProtected && this.noteId) {
 | 
					        if (this.isProtected && this.noteId) {
 | 
				
			||||||
            protectedSessionService.decryptNote(this);
 | 
					            protectedSessionService.decryptNote(this);
 | 
				
			||||||
@@ -30,6 +33,10 @@ class Note extends Entity {
 | 
				
			|||||||
        catch(e) {}
 | 
					        catch(e) {}
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    isRoot() {
 | 
				
			||||||
 | 
					        return this.noteId === 'root';
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    isJson() {
 | 
					    isJson() {
 | 
				
			||||||
        return this.mime === "application/json";
 | 
					        return this.mime === "application/json";
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -40,7 +47,7 @@ class Note extends Entity {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    isHtml() {
 | 
					    isHtml() {
 | 
				
			||||||
        return (this.type === "code" || this.type === "file") && this.mime === "text/html";
 | 
					        return (this.type === "code" || this.type === "file" || this.type === "render") && this.mime === "text/html";
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    getScriptEnv() {
 | 
					    getScriptEnv() {
 | 
				
			||||||
@@ -59,30 +66,140 @@ class Note extends Entity {
 | 
				
			|||||||
        return null;
 | 
					        return null;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async getLabels() {
 | 
					    async getOwnedAttributes() {
 | 
				
			||||||
        return await repository.getEntities("SELECT * FROM labels WHERE noteId = ? AND isDeleted = 0", [this.noteId]);
 | 
					        return await repository.getEntities(`SELECT * FROM attributes WHERE isDeleted = 0 AND noteId = ?`, [this.noteId]);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // WARNING: this doesn't take into account the possibility to have multi-valued labels!
 | 
					    async getAttributes() {
 | 
				
			||||||
    async getLabelMap() {
 | 
					        if (!this.__attributeCache) {
 | 
				
			||||||
        const map = {};
 | 
					            await this.loadAttributesToCache();
 | 
				
			||||||
 | 
					 | 
				
			||||||
        for (const label of await this.getLabels()) {
 | 
					 | 
				
			||||||
            map[label.name] = label.value;
 | 
					 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return map;
 | 
					        return this.__attributeCache;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    invalidateAttributeCache() {
 | 
				
			||||||
 | 
					        this.__attributeCache = null;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async loadAttributesToCache() {
 | 
				
			||||||
 | 
					        const attributes = await repository.getEntities(`
 | 
				
			||||||
 | 
					            WITH RECURSIVE
 | 
				
			||||||
 | 
					            tree(noteId, level) AS (
 | 
				
			||||||
 | 
					                SELECT ?, 0
 | 
				
			||||||
 | 
					                UNION
 | 
				
			||||||
 | 
					                SELECT branches.parentNoteId, tree.level + 1 FROM branches
 | 
				
			||||||
 | 
					                    JOIN tree ON branches.noteId = tree.noteId
 | 
				
			||||||
 | 
					                    JOIN notes ON notes.noteId = branches.parentNoteId
 | 
				
			||||||
 | 
					                WHERE notes.isDeleted = 0
 | 
				
			||||||
 | 
					                  AND branches.isDeleted = 0
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					            treeWithAttrs(noteId, level) AS (
 | 
				
			||||||
 | 
					                SELECT * FROM tree
 | 
				
			||||||
 | 
					                UNION
 | 
				
			||||||
 | 
					                SELECT attributes.value, treeWithAttrs.level + 1 FROM attributes
 | 
				
			||||||
 | 
					                     JOIN treeWithAttrs ON treeWithAttrs.noteId = attributes.noteId
 | 
				
			||||||
 | 
					                WHERE attributes.isDeleted = 0
 | 
				
			||||||
 | 
					                  AND attributes.type = 'relation'
 | 
				
			||||||
 | 
					                  AND attributes.name = 'inheritAttributes'
 | 
				
			||||||
 | 
					                  AND (attributes.noteId = ? OR attributes.isInheritable = 1)
 | 
				
			||||||
 | 
					                )
 | 
				
			||||||
 | 
					            SELECT attributes.* FROM attributes JOIN treeWithAttrs ON attributes.noteId = treeWithAttrs.noteId
 | 
				
			||||||
 | 
					            WHERE attributes.isDeleted = 0 AND (attributes.isInheritable = 1 OR attributes.noteId = ?)
 | 
				
			||||||
 | 
					            ORDER BY level, noteId, position`, [this.noteId, this.noteId, this.noteId]);
 | 
				
			||||||
 | 
					        // attributes are ordered so that "closest" attributes are first
 | 
				
			||||||
 | 
					        // we order by noteId so that attributes from same note stay together. Actual noteId ordering doesn't matter.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const filteredAttributes = attributes.filter((attr, index) => {
 | 
				
			||||||
 | 
					            if (attr.isDefinition()) {
 | 
				
			||||||
 | 
					                const firstDefinitionIndex = attributes.findIndex(el => el.type === attr.type && el.name === attr.name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                // keep only if this element is the first definition for this type & name
 | 
				
			||||||
 | 
					                return firstDefinitionIndex === index;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else {
 | 
				
			||||||
 | 
					                const definitionAttr = attributes.find(el => el.type === attr.type + '-definition' && el.name === attr.name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if (!definitionAttr) {
 | 
				
			||||||
 | 
					                    return true;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                const definition = definitionAttr.value;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if (definition.multiplicityType === 'multivalue') {
 | 
				
			||||||
 | 
					                    return true;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else {
 | 
				
			||||||
 | 
					                    const firstAttrIndex = attributes.findIndex(el => el.type === attr.type && el.name === attr.name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    // in case of single-valued attribute we'll keep it only if it's first (closest)
 | 
				
			||||||
 | 
					                    return firstAttrIndex === index;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (const attr of filteredAttributes) {
 | 
				
			||||||
 | 
					            attr.isOwned = attr.noteId === this.noteId;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.__attributeCache = filteredAttributes;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async hasLabel(name) {
 | 
					    async hasLabel(name) {
 | 
				
			||||||
        const map = await this.getLabelMap();
 | 
					        return !!await this.getLabel(name);
 | 
				
			||||||
 | 
					 | 
				
			||||||
        return map.hasOwnProperty(name);
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // WARNING: this doesn't take into account the possibility to have multi-valued labels!
 | 
					    // WARNING: this doesn't take into account the possibility to have multi-valued labels!
 | 
				
			||||||
    async getLabel(name) {
 | 
					    async getLabel(name) {
 | 
				
			||||||
        return await repository.getEntity("SELECT * FROM labels WHERE noteId = ? AND name = ?", [this.noteId, name]);
 | 
					        const attributes = await this.getAttributes();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return attributes.find(attr => attr.type === 'label' && attr.name === name);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async getLabelValue(name) {
 | 
				
			||||||
 | 
					        const label = await this.getLabel(name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return label ? label.value : null;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async toggleLabel(enabled, name, value = "") {
 | 
				
			||||||
 | 
					        if (enabled) {
 | 
				
			||||||
 | 
					            await this.setLabel(name, value);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else {
 | 
				
			||||||
 | 
					            await this.removeLabel(name, value);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async setLabel(name, value = "") {
 | 
				
			||||||
 | 
					        const attributes = await this.getOwnedAttributes();
 | 
				
			||||||
 | 
					        let label = attributes.find(attr => attr.type === 'label' && attr.value === value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!label) {
 | 
				
			||||||
 | 
					            label = new Attribute({
 | 
				
			||||||
 | 
					                noteId: this.noteId,
 | 
				
			||||||
 | 
					                type: 'label',
 | 
				
			||||||
 | 
					                name: name,
 | 
				
			||||||
 | 
					                value: value
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            await label.save();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            this.invalidateAttributeCache();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async removeLabel(name, value = "") {
 | 
				
			||||||
 | 
					        const attributes = await this.getOwnedAttributes();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (const attribute of attributes) {
 | 
				
			||||||
 | 
					            if (attribute.type === 'label' && (!value || value === attribute.value)) {
 | 
				
			||||||
 | 
					                attribute.isDeleted = true;
 | 
				
			||||||
 | 
					                await attribute.save();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                this.invalidateAttributeCache();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async getRevisions() {
 | 
					    async getRevisions() {
 | 
				
			||||||
@@ -140,8 +257,6 @@ class Note extends Entity {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    beforeSaving() {
 | 
					    beforeSaving() {
 | 
				
			||||||
        super.beforeSaving();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (this.isJson() && this.jsonContent) {
 | 
					        if (this.isJson() && this.jsonContent) {
 | 
				
			||||||
            this.content = JSON.stringify(this.jsonContent, null, '\t');
 | 
					            this.content = JSON.stringify(this.jsonContent, null, '\t');
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -158,8 +273,12 @@ class Note extends Entity {
 | 
				
			|||||||
            this.dateCreated = dateUtils.nowDate();
 | 
					            this.dateCreated = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.isChanged) {
 | 
				
			||||||
            this.dateModified = dateUtils.nowDate();
 | 
					            this.dateModified = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
module.exports = Note;
 | 
					module.exports = Note;
 | 
				
			||||||
@@ -7,7 +7,7 @@ const dateUtils = require('../services/date_utils');
 | 
				
			|||||||
class NoteImage extends Entity {
 | 
					class NoteImage extends Entity {
 | 
				
			||||||
    static get tableName() { return "note_images"; }
 | 
					    static get tableName() { return "note_images"; }
 | 
				
			||||||
    static get primaryKeyName() { return "noteImageId"; }
 | 
					    static get primaryKeyName() { return "noteImageId"; }
 | 
				
			||||||
    static get hashedProperties() { return ["noteImageId", "noteId", "imageId", "isDeleted", "dateModified", "dateCreated"]; }
 | 
					    static get hashedProperties() { return ["noteImageId", "noteId", "imageId", "isDeleted", "dateCreated"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async getNote() {
 | 
					    async getNote() {
 | 
				
			||||||
        return await repository.getEntity("SELECT * FROM notes WHERE noteId = ?", [this.noteId]);
 | 
					        return await repository.getEntity("SELECT * FROM notes WHERE noteId = ?", [this.noteId]);
 | 
				
			||||||
@@ -18,8 +18,6 @@ class NoteImage extends Entity {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    beforeSaving() {
 | 
					    beforeSaving() {
 | 
				
			||||||
        super.beforeSaving();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!this.isDeleted) {
 | 
					        if (!this.isDeleted) {
 | 
				
			||||||
            this.isDeleted = false;
 | 
					            this.isDeleted = false;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -28,8 +26,12 @@ class NoteImage extends Entity {
 | 
				
			|||||||
            this.dateCreated = dateUtils.nowDate();
 | 
					            this.dateCreated = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.isChanged) {
 | 
				
			||||||
            this.dateModified = dateUtils.nowDate();
 | 
					            this.dateModified = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
module.exports = NoteImage;
 | 
					module.exports = NoteImage;
 | 
				
			||||||
@@ -7,11 +7,13 @@ const repository = require('../services/repository');
 | 
				
			|||||||
class NoteRevision extends Entity {
 | 
					class NoteRevision extends Entity {
 | 
				
			||||||
    static get tableName() { return "note_revisions"; }
 | 
					    static get tableName() { return "note_revisions"; }
 | 
				
			||||||
    static get primaryKeyName() { return "noteRevisionId"; }
 | 
					    static get primaryKeyName() { return "noteRevisionId"; }
 | 
				
			||||||
    static get hashedProperties() { return ["noteRevisionId", "noteId", "title", "content", "dateModifiedFrom", "dateModifiedTo"]; }
 | 
					    static get hashedProperties() { return ["noteRevisionId", "noteId", "title", "content", "isProtected", "dateModifiedFrom", "dateModifiedTo"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    constructor(row) {
 | 
					    constructor(row) {
 | 
				
			||||||
        super(row);
 | 
					        super(row);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.isProtected = !!this.isProtected;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (this.isProtected) {
 | 
					        if (this.isProtected) {
 | 
				
			||||||
            protectedSessionService.decryptNoteRevision(this);
 | 
					            protectedSessionService.decryptNoteRevision(this);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -22,11 +24,11 @@ class NoteRevision extends Entity {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    beforeSaving() {
 | 
					    beforeSaving() {
 | 
				
			||||||
        super.beforeSaving();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (this.isProtected) {
 | 
					        if (this.isProtected) {
 | 
				
			||||||
            protectedSessionService.encryptNoteRevision(this);
 | 
					            protectedSessionService.encryptNoteRevision(this);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -8,11 +8,19 @@ class Option extends Entity {
 | 
				
			|||||||
    static get primaryKeyName() { return "name"; }
 | 
					    static get primaryKeyName() { return "name"; }
 | 
				
			||||||
    static get hashedProperties() { return ["name", "value"]; }
 | 
					    static get hashedProperties() { return ["name", "value"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(row) {
 | 
				
			||||||
 | 
					        super(row);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.isSynced = !!this.isSynced;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    beforeSaving() {
 | 
					    beforeSaving() {
 | 
				
			||||||
        super.beforeSaving();
 | 
					        super.beforeSaving();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (this.isChanged) {
 | 
				
			||||||
            this.dateModified = dateUtils.nowDate();
 | 
					            this.dateModified = dateUtils.nowDate();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
module.exports = Option;
 | 
					module.exports = Option;
 | 
				
			||||||
@@ -1,11 +1,24 @@
 | 
				
			|||||||
"use strict";
 | 
					"use strict";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const Entity = require('./entity');
 | 
					const Entity = require('./entity');
 | 
				
			||||||
 | 
					const dateUtils = require('../services/date_utils');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class RecentNote extends Entity {
 | 
					class RecentNote extends Entity {
 | 
				
			||||||
    static get tableName() { return "recent_notes"; }
 | 
					    static get tableName() { return "recent_notes"; }
 | 
				
			||||||
    static get primaryKeyName() { return "branchId"; }
 | 
					    static get primaryKeyName() { return "branchId"; }
 | 
				
			||||||
    static get hashedProperties() { return ["branchId", "notePath", "dateAccessed", "isDeleted"]; }
 | 
					    static get hashedProperties() { return ["branchId", "notePath", "dateCreated", "isDeleted"]; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    beforeSaving() {
 | 
				
			||||||
 | 
					        if (!this.isDeleted) {
 | 
				
			||||||
 | 
					            this.isDeleted = false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!this.dateCreated) {
 | 
				
			||||||
 | 
					            this.dateCreated = dateUtils.nowDate();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        super.beforeSaving();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
module.exports = RecentNote;
 | 
					module.exports = RecentNote;
 | 
				
			||||||
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/back-24.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 511 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/clock-16.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 381 B  | 
| 
		 Before Width: | Height: | Size: 245 B After Width: | Height: | Size: 245 B  | 
| 
		 Before Width: | Height: | Size: 339 B After Width: | Height: | Size: 339 B  | 
| 
		 Before Width: | Height: | Size: 463 B After Width: | Height: | Size: 463 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/edit-20.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 312 B  | 
| 
		 Before Width: | Height: | Size: 288 B After Width: | Height: | Size: 288 B  | 
| 
		 Before Width: | Height: | Size: 284 B After Width: | Height: | Size: 284 B  | 
| 
		 Before Width: | Height: | Size: 292 B After Width: | Height: | Size: 292 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/forward-24.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 511 B  | 
| 
		 Before Width: | Height: | Size: 155 B After Width: | Height: | Size: 155 B  | 
| 
		 Before Width: | Height: | Size: 323 B  | 
| 
		 Before Width: | Height: | Size: 358 B After Width: | Height: | Size: 358 B  | 
| 
		 Before Width: | Height: | Size: 252 B After Width: | Height: | Size: 252 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/play-20.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 288 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/save-20.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 388 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/search-20.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 431 B  | 
| 
		 Before Width: | Height: | Size: 419 B After Width: | Height: | Size: 419 B  | 
| 
		 Before Width: | Height: | Size: 354 B After Width: | Height: | Size: 354 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/shield-20.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 388 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/shield-off-20.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 462 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/tree-root-16.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 240 B  | 
| 
		 Before Width: | Height: | Size: 337 B  | 
							
								
								
									
										
											BIN
										
									
								
								src/public/images/icons/x-20.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 259 B  | 
							
								
								
									
										1
									
								
								src/public/images/trilium.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" x="0px" y="0px" viewBox="0 0 100 100" enable-background="new 0 0 100 100" xml:space="preserve"><path d="M63.966,45.043c0.008-0.009,0.021-0.021,0.027-0.029c0.938-1.156-0.823-13.453-5.063-20.125  c-1.389-2.186-2.239-3.423-3.219-4.719c-3.907-5.166-6-6.125-6-6.125S35.732,24.78,36.149,44.315  c-1.754,0.065-11.218,7.528-14.826,14.388c-1.206,2.291-1.856,3.645-2.493,5.141c-2.539,5.957-2.33,8.25-2.33,8.25  s16.271,6.79,33.014-3.294c0.007,0.021,0.013,0.046,0.02,0.063c0.537,1.389,12.08,5.979,19.976,5.621  c2.587-0.116,4.084-0.238,5.696-0.444c6.424-0.818,8.298-2.157,8.298-2.157S81.144,54.396,63.966,45.043z M50.787,65.343  c1.059-1.183,4.648-5.853,0.995-11.315c-0.253-0.377-0.496-0.236-0.496-0.236s0.063,10.822-5.162,12.359  c-5.225,1.537-13.886,4.4-20.427,0.455C25,66.186,26.924,53.606,38.544,47.229c0.546,1.599,2.836,6.854,9.292,6.409  c0.453-0.031,0.453-0.313,0.453-0.313s-9.422-5.328-8.156-10.625s3.089-14.236,9.766-17.948c0.714-0.397,10.746,7.593,10.417,20.94  c-1.606-0.319-7.377-1.004-10.226,4.864c-0.198,0.409,0.046,0.549,0.046,0.549s9.31-5.521,13.275-1.789  c3.965,3.733,10.813,9.763,10.71,17.4C74.111,67.533,62.197,72.258,50.787,65.343z M35.613,35.145c0,0-0.991,3.241-0.603,7.524  l-13.393-7.524C21.618,35.145,27.838,30.931,35.613,35.145z M21.193,36.03l13.344,7.612c-3.872,1.872-6.142,4.388-6.142,4.388  C20.78,43.531,21.193,36.03,21.193,36.03z M72.287,49.064c0,0-2.321-2.471-6.23-4.263l13.187-7.881  C79.243,36.92,79.808,44.413,72.287,49.064z M78.687,36.113l-13.237,7.794c0.3-4.291-0.754-7.511-0.754-7.511  C72.383,32.025,78.687,36.113,78.687,36.113z M42.076,73.778c0,0,3.309-0.737,6.845-3.185l0.056,15.361  C48.977,85.955,42.244,82.621,42.076,73.778z M49.956,85.888L50,70.526c3.539,2.445,6.846,3.181,6.846,3.181  C56.686,82.551,49.956,85.888,49.956,85.888z"></path></svg>
 | 
				
			||||||
| 
		 After Width: | Height: | Size: 1.8 KiB  | 
@@ -2,7 +2,8 @@ import cloningService from '../services/cloning.js';
 | 
				
			|||||||
import linkService from '../services/link.js';
 | 
					import linkService from '../services/link.js';
 | 
				
			||||||
import noteDetailService from '../services/note_detail.js';
 | 
					import noteDetailService from '../services/note_detail.js';
 | 
				
			||||||
import treeUtils from '../services/tree_utils.js';
 | 
					import treeUtils from '../services/tree_utils.js';
 | 
				
			||||||
import autocompleteService from '../services/autocomplete.js';
 | 
					import server from "../services/server.js";
 | 
				
			||||||
 | 
					import noteDetailText from "../services/note_detail_text.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const $dialog = $("#add-link-dialog");
 | 
					const $dialog = $("#add-link-dialog");
 | 
				
			||||||
const $form = $("#add-link-form");
 | 
					const $form = $("#add-link-form");
 | 
				
			||||||
@@ -11,8 +12,10 @@ const $linkTitle = $("#link-title");
 | 
				
			|||||||
const $clonePrefix = $("#clone-prefix");
 | 
					const $clonePrefix = $("#clone-prefix");
 | 
				
			||||||
const $linkTitleFormGroup = $("#add-link-title-form-group");
 | 
					const $linkTitleFormGroup = $("#add-link-title-form-group");
 | 
				
			||||||
const $prefixFormGroup = $("#add-link-prefix-form-group");
 | 
					const $prefixFormGroup = $("#add-link-prefix-form-group");
 | 
				
			||||||
 | 
					const $linkTypeDiv = $("#add-link-type-div");
 | 
				
			||||||
const $linkTypes = $("input[name='add-link-type']");
 | 
					const $linkTypes = $("input[name='add-link-type']");
 | 
				
			||||||
const $linkTypeHtml = $linkTypes.filter('input[value="html"]');
 | 
					const $linkTypeHtml = $linkTypes.filter('input[value="html"]');
 | 
				
			||||||
 | 
					const $showRecentNotesButton = $dialog.find(".show-recent-notes-button");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function setLinkType(linkType) {
 | 
					function setLinkType(linkType) {
 | 
				
			||||||
    $linkTypes.each(function () {
 | 
					    $linkTypes.each(function () {
 | 
				
			||||||
@@ -51,12 +54,33 @@ async function showDialog() {
 | 
				
			|||||||
        $linkTitle.val(noteTitle);
 | 
					        $linkTitle.val(noteTitle);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $autoComplete.autocomplete({
 | 
					    await $autoComplete.autocomplete({
 | 
				
			||||||
        source: await autocompleteService.getAutocompleteItems(),
 | 
					        source: async function(request, response) {
 | 
				
			||||||
 | 
					            const result = await server.get('autocomplete?query=' + encodeURIComponent(request.term));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (result.length > 0) {
 | 
				
			||||||
 | 
					                response(result.map(row => {
 | 
				
			||||||
 | 
					                    return {
 | 
				
			||||||
 | 
					                        label: row.label,
 | 
				
			||||||
 | 
					                        value: row.label + ' (' + row.value + ')'
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }));
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else {
 | 
				
			||||||
 | 
					                response([{
 | 
				
			||||||
 | 
					                    label: "No results",
 | 
				
			||||||
 | 
					                    value: "No results"
 | 
				
			||||||
 | 
					                }]);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
        minLength: 0,
 | 
					        minLength: 0,
 | 
				
			||||||
        change: async () => {
 | 
					        change: async (event, ui) => {
 | 
				
			||||||
            const val = $autoComplete.val();
 | 
					            if (!ui.item) {
 | 
				
			||||||
            const notePath = linkService.getNodePathFromLabel(val);
 | 
					                return;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            const notePath = linkService.getNotePathFromLabel(ui.item.value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if (!notePath) {
 | 
					            if (!notePath) {
 | 
				
			||||||
                return;
 | 
					                return;
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
@@ -67,21 +91,30 @@ async function showDialog() {
 | 
				
			|||||||
                await setDefaultLinkTitle(noteId);
 | 
					                await setDefaultLinkTitle(noteId);
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
 | 
					        select: function (event, ui) {
 | 
				
			||||||
 | 
					            if (ui.item.value === 'No results') {
 | 
				
			||||||
 | 
					                return false;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
        // this is called when user goes through autocomplete list with keyboard
 | 
					        // this is called when user goes through autocomplete list with keyboard
 | 
				
			||||||
        // at this point the item isn't selected yet so we use supplied ui.item to see WHERE the cursor is
 | 
					        // at this point the item isn't selected yet so we use supplied ui.item to see WHERE the cursor is
 | 
				
			||||||
        focus: async (event, ui) => {
 | 
					        focus: async (event, ui) => {
 | 
				
			||||||
            const notePath = linkService.getNodePathFromLabel(ui.item.value);
 | 
					            const notePath = linkService.getNotePathFromLabel(ui.item.value);
 | 
				
			||||||
            const noteId = treeUtils.getNoteIdFromNotePath(notePath);
 | 
					            const noteId = treeUtils.getNoteIdFromNotePath(notePath);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            await setDefaultLinkTitle(noteId);
 | 
					            await setDefaultLinkTitle(noteId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            event.preventDefault();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    showRecentNotes();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
$form.submit(() => {
 | 
					$form.submit(() => {
 | 
				
			||||||
    const value = $autoComplete.val();
 | 
					    const value = $autoComplete.val();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const notePath = linkService.getNodePathFromLabel(value);
 | 
					    const notePath = linkService.getNotePathFromLabel(value);
 | 
				
			||||||
    const noteId = treeUtils.getNoteIdFromNotePath(notePath);
 | 
					    const noteId = treeUtils.getNoteIdFromNotePath(notePath);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (notePath) {
 | 
					    if (notePath) {
 | 
				
			||||||
@@ -92,7 +125,16 @@ $form.submit(() => {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            $dialog.dialog("close");
 | 
					            $dialog.dialog("close");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            linkService.addLinkToEditor(linkTitle, '#' + notePath);
 | 
					            const linkHref = '#' + notePath;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (hasSelection()) {
 | 
				
			||||||
 | 
					                const editor = noteDetailText.getEditor();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                editor.execute('link', linkHref);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else {
 | 
				
			||||||
 | 
					                linkService.addLinkToEditor(linkTitle, linkHref);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        else if (linkType === 'selected-to-current') {
 | 
					        else if (linkType === 'selected-to-current') {
 | 
				
			||||||
            const prefix = $clonePrefix.val();
 | 
					            const prefix = $clonePrefix.val();
 | 
				
			||||||
@@ -113,21 +155,31 @@ $form.submit(() => {
 | 
				
			|||||||
    return false;
 | 
					    return false;
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// returns true if user selected some text, false if there's no selection
 | 
				
			||||||
 | 
					function hasSelection() {
 | 
				
			||||||
 | 
					    const model = noteDetailText.getEditor().model;
 | 
				
			||||||
 | 
					    const selection = model.document.selection;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return !selection.isCollapsed;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function linkTypeChanged() {
 | 
					function linkTypeChanged() {
 | 
				
			||||||
    const value = $linkTypes.filter(":checked").val();
 | 
					    const value = $linkTypes.filter(":checked").val();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (value === 'html') {
 | 
					    $linkTitleFormGroup.toggle(!hasSelection() && value === 'html');
 | 
				
			||||||
        $linkTitleFormGroup.show();
 | 
					    $prefixFormGroup.toggle(!hasSelection() && value !== 'html');
 | 
				
			||||||
        $prefixFormGroup.hide();
 | 
					
 | 
				
			||||||
    }
 | 
					    $linkTypeDiv.toggle(!hasSelection());
 | 
				
			||||||
    else {
 | 
					}
 | 
				
			||||||
        $linkTitleFormGroup.hide();
 | 
					
 | 
				
			||||||
        $prefixFormGroup.show();
 | 
					function showRecentNotes() {
 | 
				
			||||||
    }
 | 
					    $autoComplete.autocomplete("search", "");
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
$linkTypes.change(linkTypeChanged);
 | 
					$linkTypes.change(linkTypeChanged);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$showRecentNotesButton.click(showRecentNotes);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					export default {
 | 
				
			||||||
    showDialog
 | 
					    showDialog
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
							
								
								
									
										318
									
								
								src/public/javascripts/dialogs/attributes.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,318 @@
 | 
				
			|||||||
 | 
					import noteDetailService from '../services/note_detail.js';
 | 
				
			||||||
 | 
					import server from '../services/server.js';
 | 
				
			||||||
 | 
					import infoService from "../services/info.js";
 | 
				
			||||||
 | 
					import treeUtils from "../services/tree_utils.js";
 | 
				
			||||||
 | 
					import linkService from "../services/link.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const $dialog = $("#attributes-dialog");
 | 
				
			||||||
 | 
					const $saveAttributesButton = $("#save-attributes-button");
 | 
				
			||||||
 | 
					const $ownedAttributesBody = $('#owned-attributes-table tbody');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const attributesModel = new AttributesModel();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function AttributesModel() {
 | 
				
			||||||
 | 
					    const self = this;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.ownedAttributes = ko.observableArray();
 | 
				
			||||||
 | 
					    this.inheritedAttributes = ko.observableArray();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.availableTypes = [
 | 
				
			||||||
 | 
					        { text: "Label", value: "label" },
 | 
				
			||||||
 | 
					        { text: "Label definition", value: "label-definition" },
 | 
				
			||||||
 | 
					        { text: "Relation", value: "relation" },
 | 
				
			||||||
 | 
					        { text: "Relation definition", value: "relation-definition" }
 | 
				
			||||||
 | 
					    ];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.availableLabelTypes = [
 | 
				
			||||||
 | 
					        { text: "Text", value: "text" },
 | 
				
			||||||
 | 
					        { text: "Number", value: "number" },
 | 
				
			||||||
 | 
					        { text: "Boolean", value: "boolean" },
 | 
				
			||||||
 | 
					        { text: "Date", value: "date" }
 | 
				
			||||||
 | 
					    ];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.multiplicityTypes = [
 | 
				
			||||||
 | 
					        { text: "Single value", value: "singlevalue" },
 | 
				
			||||||
 | 
					        { text: "Multi value", value: "multivalue" }
 | 
				
			||||||
 | 
					    ];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.typeChanged = function(data, event) {
 | 
				
			||||||
 | 
					        self.getTargetAttribute(event.target).valueHasMutated();
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.labelTypeChanged = function(data, event) {
 | 
				
			||||||
 | 
					        self.getTargetAttribute(event.target).valueHasMutated();
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.updateAttributePositions = function() {
 | 
				
			||||||
 | 
					        let position = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // we need to update positions by searching in the DOM, because order of the
 | 
				
			||||||
 | 
					        // attributes in the viewmodel (self.ownedAttributes()) stays the same
 | 
				
			||||||
 | 
					        $ownedAttributesBody.find('input[name="position"]').each(function() {
 | 
				
			||||||
 | 
					            const attribute = self.getTargetAttribute(this);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            attribute().position = position++;
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async function showAttributes(attributes) {
 | 
				
			||||||
 | 
					        const ownedAttributes = attributes.filter(attr => attr.isOwned);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (const attr of ownedAttributes) {
 | 
				
			||||||
 | 
					            attr.labelValue = attr.type === 'label' ? attr.value : '';
 | 
				
			||||||
 | 
					            attr.relationValue = attr.type === 'relation' ? (await treeUtils.getNoteTitle(attr.value) + " (" + attr.value + ")") : '';
 | 
				
			||||||
 | 
					            attr.labelDefinition = (attr.type === 'label-definition' && attr.value) ? attr.value : {
 | 
				
			||||||
 | 
					                labelType: "text",
 | 
				
			||||||
 | 
					                multiplicityType: "singlevalue",
 | 
				
			||||||
 | 
					                isPromoted: true
 | 
				
			||||||
 | 
					            };
 | 
				
			||||||
 | 
					            attr.relationDefinition = attr.type === ('relation-definition' && attr.value) ? attr.value : {
 | 
				
			||||||
 | 
					                multiplicityType: "singlevalue",
 | 
				
			||||||
 | 
					                isPromoted: true
 | 
				
			||||||
 | 
					            };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            delete attr.value;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.ownedAttributes(ownedAttributes.map(ko.observable));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        addLastEmptyRow();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const inheritedAttributes = attributes.filter(attr => !attr.isOwned);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.inheritedAttributes(inheritedAttributes);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.loadAttributes = async function() {
 | 
				
			||||||
 | 
					        const noteId = noteDetailService.getCurrentNoteId();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const attributes = await server.get('notes/' + noteId + '/attributes');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        await showAttributes(attributes);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // attribute might not be rendered immediatelly so could not focus
 | 
				
			||||||
 | 
					        setTimeout(() => $(".attribute-name:last").focus(), 100);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $ownedAttributesBody.sortable({
 | 
				
			||||||
 | 
					            handle: '.handle',
 | 
				
			||||||
 | 
					            containment: $ownedAttributesBody,
 | 
				
			||||||
 | 
					            update: this.updateAttributePositions
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.deleteAttribute = function(data, event) {
 | 
				
			||||||
 | 
					        const attribute = self.getTargetAttribute(event.target);
 | 
				
			||||||
 | 
					        const attributeData = attribute();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (attributeData) {
 | 
				
			||||||
 | 
					            attributeData.isDeleted = true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            attribute(attributeData);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            addLastEmptyRow();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function isValid() {
 | 
				
			||||||
 | 
					        for (let attributes = self.ownedAttributes(), i = 0; i < attributes.length; i++) {
 | 
				
			||||||
 | 
					            if (self.isEmptyName(i)) {
 | 
				
			||||||
 | 
					                return false;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return true;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.save = async function() {
 | 
				
			||||||
 | 
					        // we need to defocus from input (in case of enter-triggered save) because value is updated
 | 
				
			||||||
 | 
					        // on blur event (because of conflict with jQuery UI Autocomplete). Without this, input would
 | 
				
			||||||
 | 
					        // stay in focus, blur wouldn't be triggered and change wouldn't be updated in the viewmodel.
 | 
				
			||||||
 | 
					        $saveAttributesButton.focus();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!isValid()) {
 | 
				
			||||||
 | 
					            alert("Please fix all validation errors and try saving again.");
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.updateAttributePositions();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const noteId = noteDetailService.getCurrentNoteId();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const attributesToSave = self.ownedAttributes()
 | 
				
			||||||
 | 
					            .map(attribute => attribute())
 | 
				
			||||||
 | 
					            .filter(attribute => attribute.attributeId !== "" || attribute.name !== "");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (const attr of attributesToSave) {
 | 
				
			||||||
 | 
					            if (attr.type === 'label') {
 | 
				
			||||||
 | 
					                attr.value = attr.labelValue;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else if (attr.type === 'relation') {
 | 
				
			||||||
 | 
					                attr.value = treeUtils.getNoteIdFromNotePath(linkService.getNotePathFromLabel(attr.relationValue));
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else if (attr.type === 'label-definition') {
 | 
				
			||||||
 | 
					                attr.value = attr.labelDefinition;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else if (attr.type === 'relation-definition') {
 | 
				
			||||||
 | 
					                attr.value = attr.relationDefinition;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            delete attr.labelValue;
 | 
				
			||||||
 | 
					            delete attr.relationValue;
 | 
				
			||||||
 | 
					            delete attr.labelDefinition;
 | 
				
			||||||
 | 
					            delete attr.relationDefinition;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const attributes = await server.put('notes/' + noteId + '/attributes', attributesToSave);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        await showAttributes(attributes);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        infoService.showMessage("Attributes have been saved.");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        noteDetailService.loadAttributes();
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function addLastEmptyRow() {
 | 
				
			||||||
 | 
					        const attributes = self.ownedAttributes().filter(attr => !attr().isDeleted);
 | 
				
			||||||
 | 
					        const last = attributes.length === 0 ? null : attributes[attributes.length - 1]();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (!last || last.name.trim() !== "") {
 | 
				
			||||||
 | 
					            self.ownedAttributes.push(ko.observable({
 | 
				
			||||||
 | 
					                attributeId: '',
 | 
				
			||||||
 | 
					                type: 'label',
 | 
				
			||||||
 | 
					                name: '',
 | 
				
			||||||
 | 
					                labelValue: '',
 | 
				
			||||||
 | 
					                relationValue: '',
 | 
				
			||||||
 | 
					                isInheritable: false,
 | 
				
			||||||
 | 
					                isDeleted: false,
 | 
				
			||||||
 | 
					                position: 0,
 | 
				
			||||||
 | 
					                labelDefinition: {
 | 
				
			||||||
 | 
					                    labelType: "text",
 | 
				
			||||||
 | 
					                    multiplicityType: "singlevalue",
 | 
				
			||||||
 | 
					                    isPromoted: true
 | 
				
			||||||
 | 
					                },
 | 
				
			||||||
 | 
					                relationDefinition: {
 | 
				
			||||||
 | 
					                    multiplicityType: "singlevalue",
 | 
				
			||||||
 | 
					                    isPromoted: true
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.attributeChanged = function (data, event) {
 | 
				
			||||||
 | 
					        addLastEmptyRow();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const attribute = self.getTargetAttribute(event.target);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        attribute.valueHasMutated();
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.isNotUnique = function(index) {
 | 
				
			||||||
 | 
					        const cur = self.ownedAttributes()[index]();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (cur.name.trim() === "") {
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (let attributes = self.ownedAttributes(), i = 0; i < attributes.length; i++) {
 | 
				
			||||||
 | 
					            const attribute = attributes[i]();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (index !== i && cur.name === attribute.name && cur.type === attribute.type) {
 | 
				
			||||||
 | 
					                return true;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.isEmptyName = function(index) {
 | 
				
			||||||
 | 
					        const cur = self.ownedAttributes()[index]();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return cur.name.trim() === "" && !cur.isDeleted && (cur.attributeId !== "" || cur.labelValue !== "" || cur.relationValue);
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.getTargetAttribute = function(target) {
 | 
				
			||||||
 | 
					        const context = ko.contextFor(target);
 | 
				
			||||||
 | 
					        const index = context.$index();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return self.ownedAttributes()[index];
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function showDialog() {
 | 
				
			||||||
 | 
					    glob.activeDialog = $dialog;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    await attributesModel.loadAttributes();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $dialog.dialog({
 | 
				
			||||||
 | 
					        modal: true,
 | 
				
			||||||
 | 
					        width: 950,
 | 
				
			||||||
 | 
					        height: 500
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ko.applyBindings(attributesModel, $dialog[0]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$dialog.on('focus', '.attribute-name', function (e) {
 | 
				
			||||||
 | 
					    if (!$(this).hasClass("ui-autocomplete-input")) {
 | 
				
			||||||
 | 
					        $(this).autocomplete({
 | 
				
			||||||
 | 
					            source: async (request, response) => {
 | 
				
			||||||
 | 
					                const attribute = attributesModel.getTargetAttribute(this);
 | 
				
			||||||
 | 
					                const type = (attribute().type === 'relation' || attribute().type === 'relation-definition') ? 'relation' : 'label';
 | 
				
			||||||
 | 
					                const names = await server.get('attributes/names/?type=' + type + '&query=' + encodeURIComponent(request.term));
 | 
				
			||||||
 | 
					                const result = names.map(name => {
 | 
				
			||||||
 | 
					                    return {
 | 
				
			||||||
 | 
					                        label: name,
 | 
				
			||||||
 | 
					                        value: name
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if (result.length > 0) {
 | 
				
			||||||
 | 
					                    response(result);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else {
 | 
				
			||||||
 | 
					                    response([{
 | 
				
			||||||
 | 
					                        label: "No results",
 | 
				
			||||||
 | 
					                        value: "No results"
 | 
				
			||||||
 | 
					                    }]);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            minLength: 0
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $(this).autocomplete("search", $(this).val());
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$dialog.on('focus', '.label-value', async function (e) {
 | 
				
			||||||
 | 
					    if (!$(this).hasClass("ui-autocomplete-input")) {
 | 
				
			||||||
 | 
					        const attributeName = $(this).parent().parent().find('.attribute-name').val();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (attributeName.trim() === "") {
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const attributeValues = await server.get('attributes/values/' + encodeURIComponent(attributeName));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (attributeValues.length === 0) {
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $(this).autocomplete({
 | 
				
			||||||
 | 
					            // shouldn't be required and autocomplete should just accept array of strings, but that fails
 | 
				
			||||||
 | 
					            // because we have overriden filter() function in autocomplete.js
 | 
				
			||||||
 | 
					            source: attributeValues.map(attribute => {
 | 
				
			||||||
 | 
					                return {
 | 
				
			||||||
 | 
					                    attribute: attribute,
 | 
				
			||||||
 | 
					                    value: attribute
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }),
 | 
				
			||||||
 | 
					            minLength: 0
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $(this).autocomplete("search", $(this).val());
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default {
 | 
				
			||||||
 | 
					    showDialog
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
@@ -19,7 +19,7 @@ async function showDialog() {
 | 
				
			|||||||
    $list.html('');
 | 
					    $list.html('');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for (const event of result) {
 | 
					    for (const event of result) {
 | 
				
			||||||
        const dateTime = utils.formatDateTime(utils.parseDate(event.dateAdded));
 | 
					        const dateTime = utils.formatDateTime(utils.parseDate(event.dateCreated));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (event.noteId) {
 | 
					        if (event.noteId) {
 | 
				
			||||||
            const noteLink = await linkService.createNoteLink(event.noteId).prop('outerHTML');
 | 
					            const noteLink = await linkService.createNoteLink(event.noteId).prop('outerHTML');
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,10 +1,11 @@
 | 
				
			|||||||
import treeService from '../services/tree.js';
 | 
					import treeService from '../services/tree.js';
 | 
				
			||||||
import linkService from '../services/link.js';
 | 
					 | 
				
			||||||
import server from '../services/server.js';
 | 
					import server from '../services/server.js';
 | 
				
			||||||
 | 
					import searchNotesService from '../services/search_notes.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const $dialog = $("#jump-to-note-dialog");
 | 
					const $dialog = $("#jump-to-note-dialog");
 | 
				
			||||||
const $autoComplete = $("#jump-to-note-autocomplete");
 | 
					const $autoComplete = $("#jump-to-note-autocomplete");
 | 
				
			||||||
const $form = $("#jump-to-note-form");
 | 
					const $showInFullTextButton = $("#show-in-full-text-button");
 | 
				
			||||||
 | 
					const $showRecentNotesButton = $dialog.find(".show-recent-notes-button");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function showDialog() {
 | 
					async function showDialog() {
 | 
				
			||||||
    glob.activeDialog = $dialog;
 | 
					    glob.activeDialog = $dialog;
 | 
				
			||||||
@@ -13,39 +14,66 @@ async function showDialog() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    $dialog.dialog({
 | 
					    $dialog.dialog({
 | 
				
			||||||
        modal: true,
 | 
					        modal: true,
 | 
				
			||||||
        width: 800
 | 
					        width: 800,
 | 
				
			||||||
 | 
					        position: { my: "center top+100", at: "top", of: window }
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await $autoComplete.autocomplete({
 | 
					    await $autoComplete.autocomplete({
 | 
				
			||||||
        source: async function(request, response) {
 | 
					        source: async function(request, response) {
 | 
				
			||||||
            const result = await server.get('autocomplete?query=' + encodeURIComponent(request.term));
 | 
					            const result = await server.get('autocomplete?query=' + encodeURIComponent(request.term));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (result.length > 0) {
 | 
				
			||||||
                response(result);
 | 
					                response(result);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else {
 | 
				
			||||||
 | 
					                response([{
 | 
				
			||||||
 | 
					                    label: "No results",
 | 
				
			||||||
 | 
					                    value: "No results"
 | 
				
			||||||
 | 
					                }]);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        minLength: 2
 | 
					        focus: function(event, ui) {
 | 
				
			||||||
    });
 | 
					            event.preventDefault();
 | 
				
			||||||
}
 | 
					        },
 | 
				
			||||||
 | 
					        minLength: 0,
 | 
				
			||||||
 | 
					        autoFocus: true,
 | 
				
			||||||
 | 
					        select: function (event, ui) {
 | 
				
			||||||
 | 
					            if (ui.item.value === 'No results') {
 | 
				
			||||||
 | 
					                return false;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function getSelectedNotePath() {
 | 
					            treeService.activateNode(ui.item.value);
 | 
				
			||||||
    const val = $autoComplete.val();
 | 
					 | 
				
			||||||
    return linkService.getNodePathFromLabel(val);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function goToNote() {
 | 
					 | 
				
			||||||
    const notePath = getSelectedNotePath();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (notePath) {
 | 
					 | 
				
			||||||
        treeService.activateNode(notePath);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
            $dialog.dialog('close');
 | 
					            $dialog.dialog('close');
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    showRecentNotes();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
$form.submit(() => {
 | 
					function showInFullText(e) {
 | 
				
			||||||
    goToNote();
 | 
					    // stop from propagating upwards (dangerous especially with ctrl+enter executable javascript notes)
 | 
				
			||||||
 | 
					    e.preventDefault();
 | 
				
			||||||
 | 
					    e.stopPropagation();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return false;
 | 
					    const searchText = $autoComplete.val();
 | 
				
			||||||
});
 | 
					
 | 
				
			||||||
 | 
					    searchNotesService.resetSearch();
 | 
				
			||||||
 | 
					    searchNotesService.showSearch();
 | 
				
			||||||
 | 
					    searchNotesService.doSearch(searchText);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $dialog.dialog('close');
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function showRecentNotes() {
 | 
				
			||||||
 | 
					    $autoComplete.autocomplete("search", "");
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$showInFullTextButton.click(showInFullText);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$showRecentNotesButton.click(showRecentNotes);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$dialog.bind('keydown', 'ctrl+return', showInFullText);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					export default {
 | 
				
			||||||
    showDialog
 | 
					    showDialog
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,223 +0,0 @@
 | 
				
			|||||||
import noteDetailService from '../services/note_detail.js';
 | 
					 | 
				
			||||||
import utils from '../services/utils.js';
 | 
					 | 
				
			||||||
import server from '../services/server.js';
 | 
					 | 
				
			||||||
import infoService from "../services/info.js";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const $dialog = $("#labels-dialog");
 | 
					 | 
				
			||||||
const $saveLabelsButton = $("#save-labels-button");
 | 
					 | 
				
			||||||
const $labelsBody = $('#labels-table tbody');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const labelsModel = new LabelsModel();
 | 
					 | 
				
			||||||
let labelNames = [];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function LabelsModel() {
 | 
					 | 
				
			||||||
    const self = this;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.labels = ko.observableArray();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.updateLabelPositions = function() {
 | 
					 | 
				
			||||||
        let position = 0;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        // we need to update positions by searching in the DOM, because order of the
 | 
					 | 
				
			||||||
        // labels in the viewmodel (self.labels()) stays the same
 | 
					 | 
				
			||||||
        $labelsBody.find('input[name="position"]').each(function() {
 | 
					 | 
				
			||||||
            const label = self.getTargetLabel(this);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            label().position = position++;
 | 
					 | 
				
			||||||
        });
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.loadLabels = async function() {
 | 
					 | 
				
			||||||
        const noteId = noteDetailService.getCurrentNoteId();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const labels = await server.get('notes/' + noteId + '/labels');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        self.labels(labels.map(ko.observable));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        addLastEmptyRow();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        labelNames = await server.get('labels/names');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        // label might not be rendered immediatelly so could not focus
 | 
					 | 
				
			||||||
        setTimeout(() => $(".label-name:last").focus(), 100);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        $labelsBody.sortable({
 | 
					 | 
				
			||||||
            handle: '.handle',
 | 
					 | 
				
			||||||
            containment: $labelsBody,
 | 
					 | 
				
			||||||
            update: this.updateLabelPositions
 | 
					 | 
				
			||||||
        });
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.deleteLabel = function(data, event) {
 | 
					 | 
				
			||||||
        const label = self.getTargetLabel(event.target);
 | 
					 | 
				
			||||||
        const labelData = label();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (labelData) {
 | 
					 | 
				
			||||||
            labelData.isDeleted = 1;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            label(labelData);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            addLastEmptyRow();
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    function isValid() {
 | 
					 | 
				
			||||||
        for (let labels = self.labels(), i = 0; i < labels.length; i++) {
 | 
					 | 
				
			||||||
            if (self.isEmptyName(i)) {
 | 
					 | 
				
			||||||
                return false;
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.save = async function() {
 | 
					 | 
				
			||||||
        // we need to defocus from input (in case of enter-triggered save) because value is updated
 | 
					 | 
				
			||||||
        // on blur event (because of conflict with jQuery UI Autocomplete). Without this, input would
 | 
					 | 
				
			||||||
        // stay in focus, blur wouldn't be triggered and change wouldn't be updated in the viewmodel.
 | 
					 | 
				
			||||||
        $saveLabelsButton.focus();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!isValid()) {
 | 
					 | 
				
			||||||
            alert("Please fix all validation errors and try saving again.");
 | 
					 | 
				
			||||||
            return;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        self.updateLabelPositions();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const noteId = noteDetailService.getCurrentNoteId();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const labelsToSave = self.labels()
 | 
					 | 
				
			||||||
            .map(label => label())
 | 
					 | 
				
			||||||
            .filter(label => label.labelId !== "" || label.name !== "");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const labels = await server.put('notes/' + noteId + '/labels', labelsToSave);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        self.labels(labels.map(ko.observable));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        addLastEmptyRow();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        infoService.showMessage("Labels have been saved.");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        noteDetailService.loadLabelList();
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    function addLastEmptyRow() {
 | 
					 | 
				
			||||||
        const labels = self.labels().filter(attr => attr().isDeleted === 0);
 | 
					 | 
				
			||||||
        const last = labels.length === 0 ? null : labels[labels.length - 1]();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!last || last.name.trim() !== "" || last.value !== "") {
 | 
					 | 
				
			||||||
            self.labels.push(ko.observable({
 | 
					 | 
				
			||||||
                labelId: '',
 | 
					 | 
				
			||||||
                name: '',
 | 
					 | 
				
			||||||
                value: '',
 | 
					 | 
				
			||||||
                isDeleted: 0,
 | 
					 | 
				
			||||||
                position: 0
 | 
					 | 
				
			||||||
            }));
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.labelChanged = function (data, event) {
 | 
					 | 
				
			||||||
        addLastEmptyRow();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const label = self.getTargetLabel(event.target);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        label.valueHasMutated();
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.isNotUnique = function(index) {
 | 
					 | 
				
			||||||
        const cur = self.labels()[index]();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (cur.name.trim() === "") {
 | 
					 | 
				
			||||||
            return false;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        for (let labels = self.labels(), i = 0; i < labels.length; i++) {
 | 
					 | 
				
			||||||
            const label = labels[i]();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            if (index !== i && cur.name === label.name) {
 | 
					 | 
				
			||||||
                return true;
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return false;
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.isEmptyName = function(index) {
 | 
					 | 
				
			||||||
        const cur = self.labels()[index]();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return cur.name.trim() === "" && (cur.labelId !== "" || cur.value !== "");
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    this.getTargetLabel = function(target) {
 | 
					 | 
				
			||||||
        const context = ko.contextFor(target);
 | 
					 | 
				
			||||||
        const index = context.$index();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return self.labels()[index];
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
async function showDialog() {
 | 
					 | 
				
			||||||
    glob.activeDialog = $dialog;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    await labelsModel.loadLabels();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $dialog.dialog({
 | 
					 | 
				
			||||||
        modal: true,
 | 
					 | 
				
			||||||
        width: 800,
 | 
					 | 
				
			||||||
        height: 500
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
ko.applyBindings(labelsModel, document.getElementById('labels-dialog'));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
$(document).on('focus', '.label-name', function (e) {
 | 
					 | 
				
			||||||
    if (!$(this).hasClass("ui-autocomplete-input")) {
 | 
					 | 
				
			||||||
        $(this).autocomplete({
 | 
					 | 
				
			||||||
            // shouldn't be required and autocomplete should just accept array of strings, but that fails
 | 
					 | 
				
			||||||
            // because we have overriden filter() function in autocomplete.js
 | 
					 | 
				
			||||||
            source: labelNames.map(label => {
 | 
					 | 
				
			||||||
                return {
 | 
					 | 
				
			||||||
                    label: label,
 | 
					 | 
				
			||||||
                    value: label
 | 
					 | 
				
			||||||
                }
 | 
					 | 
				
			||||||
            }),
 | 
					 | 
				
			||||||
            minLength: 0
 | 
					 | 
				
			||||||
        });
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $(this).autocomplete("search", $(this).val());
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
$(document).on('focus', '.label-value', async function (e) {
 | 
					 | 
				
			||||||
    if (!$(this).hasClass("ui-autocomplete-input")) {
 | 
					 | 
				
			||||||
        const labelName = $(this).parent().parent().find('.label-name').val();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (labelName.trim() === "") {
 | 
					 | 
				
			||||||
            return;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const labelValues = await server.get('labels/values/' + encodeURIComponent(labelName));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (labelValues.length === 0) {
 | 
					 | 
				
			||||||
            return;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        $(this).autocomplete({
 | 
					 | 
				
			||||||
            // shouldn't be required and autocomplete should just accept array of strings, but that fails
 | 
					 | 
				
			||||||
            // because we have overriden filter() function in autocomplete.js
 | 
					 | 
				
			||||||
            source: labelValues.map(label => {
 | 
					 | 
				
			||||||
                return {
 | 
					 | 
				
			||||||
                    label: label,
 | 
					 | 
				
			||||||
                    value: label
 | 
					 | 
				
			||||||
                }
 | 
					 | 
				
			||||||
            }),
 | 
					 | 
				
			||||||
            minLength: 0
 | 
					 | 
				
			||||||
        });
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $(this).autocomplete("search", $(this).val());
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default {
 | 
					 | 
				
			||||||
    showDialog
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
@@ -29,7 +29,7 @@ function formatNode(node, level) {
 | 
				
			|||||||
    const indentAfter  = new Array(level - 1).join('  ');
 | 
					    const indentAfter  = new Array(level - 1).join('  ');
 | 
				
			||||||
    let textNode;
 | 
					    let textNode;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for (const i = 0; i < node.children.length; i++) {
 | 
					    for (let i = 0; i < node.children.length; i++) {
 | 
				
			||||||
        textNode = document.createTextNode('\n' + indentBefore);
 | 
					        textNode = document.createTextNode('\n' + indentBefore);
 | 
				
			||||||
        node.insertBefore(textNode, node.children[i]);
 | 
					        node.insertBefore(textNode, node.children[i]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,9 +1,10 @@
 | 
				
			|||||||
"use strict";
 | 
					"use strict";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import protectedSessionHolder from '../services/protected_session_holder.js';
 | 
					import protectedSessionHolder from '../services/protected_session_holder.js';
 | 
				
			||||||
import utils from '../services/utils.js';
 | 
					 | 
				
			||||||
import server from '../services/server.js';
 | 
					import server from '../services/server.js';
 | 
				
			||||||
import infoService from "../services/info.js";
 | 
					import infoService from "../services/info.js";
 | 
				
			||||||
 | 
					import zoomService from "../services/zoom.js";
 | 
				
			||||||
 | 
					import utils from "../services/utils.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const $dialog = $("#options-dialog");
 | 
					const $dialog = $("#options-dialog");
 | 
				
			||||||
const $tabs = $("#options-tabs");
 | 
					const $tabs = $("#options-tabs");
 | 
				
			||||||
@@ -33,8 +34,8 @@ async function showDialog() {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function saveOptions(optionName, optionValue) {
 | 
					async function saveOptions(options) {
 | 
				
			||||||
    await server.put('options/' + encodeURIComponent(optionName) + '/' + encodeURIComponent(optionValue));
 | 
					    await server.put('options', options);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    infoService.showMessage("Options change have been saved.");
 | 
					    infoService.showMessage("Options change have been saved.");
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -44,6 +45,41 @@ export default {
 | 
				
			|||||||
    saveOptions
 | 
					    saveOptions
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					addTabHandler((function() {
 | 
				
			||||||
 | 
					    const $themeSelect = $("#theme-select");
 | 
				
			||||||
 | 
					    const $zoomFactorSelect = $("#zoom-factor-select");
 | 
				
			||||||
 | 
					    const $html = $("html");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function optionsLoaded(options) {
 | 
				
			||||||
 | 
					        $themeSelect.val(options.theme);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (utils.isElectron()) {
 | 
				
			||||||
 | 
					            $zoomFactorSelect.val(options.zoomFactor);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else {
 | 
				
			||||||
 | 
					            $zoomFactorSelect.prop('disabled', true);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $themeSelect.change(function() {
 | 
				
			||||||
 | 
					        const newTheme = $(this).val();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $html.attr("class", "theme-" + newTheme);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        server.put('options/theme/' + newTheme);
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $zoomFactorSelect.change(function() {
 | 
				
			||||||
 | 
					        const newZoomFactor = $(this).val();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        zoomService.setZoomFactorAndSave(newZoomFactor);
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return {
 | 
				
			||||||
 | 
					        optionsLoaded
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					})());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
addTabHandler((function() {
 | 
					addTabHandler((function() {
 | 
				
			||||||
    const $form = $("#change-password-form");
 | 
					    const $form = $("#change-password-form");
 | 
				
			||||||
    const $oldPassword = $("#old-password");
 | 
					    const $oldPassword = $("#old-password");
 | 
				
			||||||
@@ -93,16 +129,15 @@ addTabHandler((function() {
 | 
				
			|||||||
addTabHandler((function() {
 | 
					addTabHandler((function() {
 | 
				
			||||||
    const $form = $("#protected-session-timeout-form");
 | 
					    const $form = $("#protected-session-timeout-form");
 | 
				
			||||||
    const $protectedSessionTimeout = $("#protected-session-timeout-in-seconds");
 | 
					    const $protectedSessionTimeout = $("#protected-session-timeout-in-seconds");
 | 
				
			||||||
    const optionName = 'protectedSessionTimeout';
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function optionsLoaded(options) {
 | 
					    function optionsLoaded(options) {
 | 
				
			||||||
        $protectedSessionTimeout.val(options[optionName]);
 | 
					        $protectedSessionTimeout.val(options['protectedSessionTimeout']);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $form.submit(() => {
 | 
					    $form.submit(() => {
 | 
				
			||||||
        const protectedSessionTimeout = $protectedSessionTimeout.val();
 | 
					        const protectedSessionTimeout = $protectedSessionTimeout.val();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        saveOptions(optionName, protectedSessionTimeout).then(() => {
 | 
					        saveOptions({ 'protectedSessionTimeout': protectedSessionTimeout }).then(() => {
 | 
				
			||||||
            protectedSessionHolder.setProtectedSessionTimeout(protectedSessionTimeout);
 | 
					            protectedSessionHolder.setProtectedSessionTimeout(protectedSessionTimeout);
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -117,14 +152,13 @@ addTabHandler((function() {
 | 
				
			|||||||
addTabHandler((function () {
 | 
					addTabHandler((function () {
 | 
				
			||||||
    const $form = $("#note-revision-snapshot-time-interval-form");
 | 
					    const $form = $("#note-revision-snapshot-time-interval-form");
 | 
				
			||||||
    const $timeInterval = $("#note-revision-snapshot-time-interval-in-seconds");
 | 
					    const $timeInterval = $("#note-revision-snapshot-time-interval-in-seconds");
 | 
				
			||||||
    const optionName = 'noteRevisionSnapshotTimeInterval';
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function optionsLoaded(options) {
 | 
					    function optionsLoaded(options) {
 | 
				
			||||||
        $timeInterval.val(options[optionName]);
 | 
					        $timeInterval.val(options['noteRevisionSnapshotTimeInterval']);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $form.submit(() => {
 | 
					    $form.submit(() => {
 | 
				
			||||||
        saveOptions(optionName, $timeInterval.val());
 | 
					        saveOptions({ 'noteRevisionSnapshotTimeInterval': $timeInterval.val() });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return false;
 | 
					        return false;
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
@@ -137,6 +171,7 @@ addTabHandler((function () {
 | 
				
			|||||||
addTabHandler((async function () {
 | 
					addTabHandler((async function () {
 | 
				
			||||||
    const $appVersion = $("#app-version");
 | 
					    const $appVersion = $("#app-version");
 | 
				
			||||||
    const $dbVersion = $("#db-version");
 | 
					    const $dbVersion = $("#db-version");
 | 
				
			||||||
 | 
					    const $syncVersion = $("#sync-version");
 | 
				
			||||||
    const $buildDate = $("#build-date");
 | 
					    const $buildDate = $("#build-date");
 | 
				
			||||||
    const $buildRevision = $("#build-revision");
 | 
					    const $buildRevision = $("#build-revision");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -144,6 +179,7 @@ addTabHandler((async function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    $appVersion.html(appInfo.appVersion);
 | 
					    $appVersion.html(appInfo.appVersion);
 | 
				
			||||||
    $dbVersion.html(appInfo.dbVersion);
 | 
					    $dbVersion.html(appInfo.dbVersion);
 | 
				
			||||||
 | 
					    $syncVersion.html(appInfo.syncVersion);
 | 
				
			||||||
    $buildDate.html(appInfo.buildDate);
 | 
					    $buildDate.html(appInfo.buildDate);
 | 
				
			||||||
    $buildRevision.html(appInfo.buildRevision);
 | 
					    $buildRevision.html(appInfo.buildRevision);
 | 
				
			||||||
    $buildRevision.attr('href', 'https://github.com/zadam/trilium/commit/' + appInfo.buildRevision);
 | 
					    $buildRevision.attr('href', 'https://github.com/zadam/trilium/commit/' + appInfo.buildRevision);
 | 
				
			||||||
@@ -151,6 +187,57 @@ addTabHandler((async function () {
 | 
				
			|||||||
    return {};
 | 
					    return {};
 | 
				
			||||||
})());
 | 
					})());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					addTabHandler((function() {
 | 
				
			||||||
 | 
					    const $form = $("#sync-setup-form");
 | 
				
			||||||
 | 
					    const $syncServerHost = $("#sync-server-host");
 | 
				
			||||||
 | 
					    const $syncServerTimeout = $("#sync-server-timeout");
 | 
				
			||||||
 | 
					    const $syncProxy = $("#sync-proxy");
 | 
				
			||||||
 | 
					    const $testSyncButton = $("#test-sync-button");
 | 
				
			||||||
 | 
					    const $syncToServerButton = $("#sync-to-server-button");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function optionsLoaded(options) {
 | 
				
			||||||
 | 
					        $syncServerHost.val(options['syncServerHost']);
 | 
				
			||||||
 | 
					        $syncServerTimeout.val(options['syncServerTimeout']);
 | 
				
			||||||
 | 
					        $syncProxy.val(options['syncProxy']);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $form.submit(() => {
 | 
				
			||||||
 | 
					        saveOptions({
 | 
				
			||||||
 | 
					            'syncServerHost': $syncServerHost.val(),
 | 
				
			||||||
 | 
					            'syncServerTimeout': $syncServerTimeout.val(),
 | 
				
			||||||
 | 
					            'syncProxy': $syncProxy.val()
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $testSyncButton.click(async () => {
 | 
				
			||||||
 | 
					        const result = await server.post('sync/test');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (result.connection === "Success") {
 | 
				
			||||||
 | 
					            infoService.showMessage("Sync server handshake has been successful");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else {
 | 
				
			||||||
 | 
					            infoService.showError("Sync server handshake failed, error: " + result.error);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $syncToServerButton.click(async () => {
 | 
				
			||||||
 | 
					        const resp = await server.post("setup/sync-to-server");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (resp.success) {
 | 
				
			||||||
 | 
					            infoService.showMessage("Sync has been established to the server instance. It will take some time to finish.");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else {
 | 
				
			||||||
 | 
					            infoService.showError('Sync setup failed: ' + resp.error);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return {
 | 
				
			||||||
 | 
					        optionsLoaded
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					})());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
addTabHandler((async function () {
 | 
					addTabHandler((async function () {
 | 
				
			||||||
    const $forceFullSyncButton = $("#force-full-sync-button");
 | 
					    const $forceFullSyncButton = $("#force-full-sync-button");
 | 
				
			||||||
    const $fillSyncRowsButton = $("#fill-sync-rows-button");
 | 
					    const $fillSyncRowsButton = $("#fill-sync-rows-button");
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,113 +0,0 @@
 | 
				
			|||||||
import treeService from '../services/tree.js';
 | 
					 | 
				
			||||||
import messagingService from '../services/messaging.js';
 | 
					 | 
				
			||||||
import server from '../services/server.js';
 | 
					 | 
				
			||||||
import utils from "../services/utils.js";
 | 
					 | 
				
			||||||
import treeUtils from "../services/tree_utils.js";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const $dialog = $("#recent-notes-dialog");
 | 
					 | 
				
			||||||
const $searchInput = $('#recent-notes-search-input');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// list of recent note paths
 | 
					 | 
				
			||||||
let list = [];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
async function reload() {
 | 
					 | 
				
			||||||
    const result = await server.get('recent-notes');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    list = result.map(r => r.notePath);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function addRecentNote(branchId, notePath) {
 | 
					 | 
				
			||||||
    setTimeout(async () => {
 | 
					 | 
				
			||||||
        // we include the note into recent list only if the user stayed on the note at least 5 seconds
 | 
					 | 
				
			||||||
        if (notePath && notePath === treeService.getCurrentNotePath()) {
 | 
					 | 
				
			||||||
            const result = await server.put('recent-notes/' + branchId + '/' + encodeURIComponent(notePath));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            list = result.map(r => r.notePath);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }, 1500);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
async function getNoteTitle(notePath) {
 | 
					 | 
				
			||||||
    let noteTitle;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
        noteTitle = await treeUtils.getNotePathTitle(notePath);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    catch (e) {
 | 
					 | 
				
			||||||
        noteTitle = "[error - can't find note title]";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        messagingService.logError("Could not find title for notePath=" + notePath + ", stack=" + e.stack);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return noteTitle;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
async function showDialog() {
 | 
					 | 
				
			||||||
    glob.activeDialog = $dialog;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $dialog.dialog({
 | 
					 | 
				
			||||||
        modal: true,
 | 
					 | 
				
			||||||
        width: 800,
 | 
					 | 
				
			||||||
        height: 100,
 | 
					 | 
				
			||||||
        position: { my: "center top+100", at: "top", of: window }
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $searchInput.val('');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    // remove the current note
 | 
					 | 
				
			||||||
    const recNotes = list.filter(note => note !== treeService.getCurrentNotePath());
 | 
					 | 
				
			||||||
    const items = [];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for (const notePath of recNotes) {
 | 
					 | 
				
			||||||
        items.push({
 | 
					 | 
				
			||||||
            label: await getNoteTitle(notePath),
 | 
					 | 
				
			||||||
            value: notePath
 | 
					 | 
				
			||||||
        });
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $searchInput.autocomplete({
 | 
					 | 
				
			||||||
        source: items,
 | 
					 | 
				
			||||||
        minLength: 0,
 | 
					 | 
				
			||||||
        autoFocus: true,
 | 
					 | 
				
			||||||
        select: function (event, ui) {
 | 
					 | 
				
			||||||
            treeService.activateNode(ui.item.value);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            $searchInput.autocomplete('destroy');
 | 
					 | 
				
			||||||
            $dialog.dialog('close');
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        focus: function (event, ui) {
 | 
					 | 
				
			||||||
            event.preventDefault();
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        close: function (event, ui) {
 | 
					 | 
				
			||||||
            if (event.keyCode === 27) { // escape closes dialog
 | 
					 | 
				
			||||||
                $searchInput.autocomplete('destroy');
 | 
					 | 
				
			||||||
                $dialog.dialog('close');
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
            else {
 | 
					 | 
				
			||||||
                // keep autocomplete open
 | 
					 | 
				
			||||||
                // we're kind of abusing autocomplete to work in a way which it's not designed for
 | 
					 | 
				
			||||||
                $searchInput.autocomplete("search", "");
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        create: () => $searchInput.autocomplete("search", ""),
 | 
					 | 
				
			||||||
        classes: {
 | 
					 | 
				
			||||||
            "ui-autocomplete": "recent-notes-autocomplete"
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
setTimeout(reload, 100);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
messagingService.subscribeToMessages(syncData => {
 | 
					 | 
				
			||||||
    if (syncData.some(sync => sync.entityName === 'recent_notes')) {
 | 
					 | 
				
			||||||
        console.log(utils.now(), "Reloading recent notes because of background changes");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        reload();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default {
 | 
					 | 
				
			||||||
    showDialog,
 | 
					 | 
				
			||||||
    addRecentNote,
 | 
					 | 
				
			||||||
    reload
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
@@ -6,7 +6,8 @@ class NoteShort {
 | 
				
			|||||||
        this.isProtected = row.isProtected;
 | 
					        this.isProtected = row.isProtected;
 | 
				
			||||||
        this.type = row.type;
 | 
					        this.type = row.type;
 | 
				
			||||||
        this.mime = row.mime;
 | 
					        this.mime = row.mime;
 | 
				
			||||||
        this.hideInAutocomplete = row.hideInAutocomplete;
 | 
					        this.archived = row.archived;
 | 
				
			||||||
 | 
					        this.cssClass = row.cssClass;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    isJson() {
 | 
					    isJson() {
 | 
				
			||||||
@@ -59,7 +60,7 @@ class NoteShort {
 | 
				
			|||||||
    get dto() {
 | 
					    get dto() {
 | 
				
			||||||
        const dto = Object.assign({}, this);
 | 
					        const dto = Object.assign({}, this);
 | 
				
			||||||
        delete dto.treeCache;
 | 
					        delete dto.treeCache;
 | 
				
			||||||
        delete dto.hideInAutocomplete;
 | 
					        delete dto.archived;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return dto;
 | 
					        return dto;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,46 +0,0 @@
 | 
				
			|||||||
import server from './services/server.js';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
$(document).ready(async () => {
 | 
					 | 
				
			||||||
    const {appDbVersion, dbVersion} = await server.get('migration');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    console.log("HI", {appDbVersion, dbVersion});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (appDbVersion === dbVersion) {
 | 
					 | 
				
			||||||
        $("#up-to-date").show();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    else {
 | 
					 | 
				
			||||||
        $("#need-to-migrate").show();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        $("#app-db-version").html(appDbVersion);
 | 
					 | 
				
			||||||
        $("#db-version").html(dbVersion);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
$("#run-migration").click(async () => {
 | 
					 | 
				
			||||||
    $("#run-migration").prop("disabled", true);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $("#migration-result").show();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const result = await server.post('migration');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for (const migration of result.migrations) {
 | 
					 | 
				
			||||||
        const row = $('<tr>')
 | 
					 | 
				
			||||||
            .append($('<td>').html(migration.dbVersion))
 | 
					 | 
				
			||||||
            .append($('<td>').html(migration.name))
 | 
					 | 
				
			||||||
            .append($('<td>').html(migration.success ? 'Yes' : 'No'))
 | 
					 | 
				
			||||||
            .append($('<td>').html(migration.success ? 'N/A' : migration.error));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!migration.success) {
 | 
					 | 
				
			||||||
            row.addClass("danger");
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        $("#migration-table").append(row);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// copy of this shortcut to be able to debug migration problems
 | 
					 | 
				
			||||||
$(document).bind('keydown', 'ctrl+shift+i', () => {
 | 
					 | 
				
			||||||
    require('electron').remote.getCurrentWindow().toggleDevTools();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return false;
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
@@ -1,104 +0,0 @@
 | 
				
			|||||||
import treeCache from "./tree_cache.js";
 | 
					 | 
				
			||||||
import treeUtils from "./tree_utils.js";
 | 
					 | 
				
			||||||
import protectedSessionHolder from './protected_session_holder.js';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
async function getAutocompleteItems(parentNoteId, notePath, titlePath) {
 | 
					 | 
				
			||||||
    if (!parentNoteId) {
 | 
					 | 
				
			||||||
        parentNoteId = 'root';
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const parentNote = await treeCache.getNote(parentNoteId);
 | 
					 | 
				
			||||||
    const childNotes = await parentNote.getChildNotes();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (!childNotes.length) {
 | 
					 | 
				
			||||||
        return [];
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (!notePath) {
 | 
					 | 
				
			||||||
        notePath = '';
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (!titlePath) {
 | 
					 | 
				
			||||||
        titlePath = '';
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const autocompleteItems = [];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for (const childNote of childNotes) {
 | 
					 | 
				
			||||||
        if (childNote.hideInAutocomplete) {
 | 
					 | 
				
			||||||
            continue;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const childNotePath = (notePath ? (notePath + '/') : '') + childNote.noteId;
 | 
					 | 
				
			||||||
        const childTitlePath = (titlePath ? (titlePath + ' / ') : '') + await treeUtils.getNoteTitle(childNote.noteId, parentNoteId);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (!childNote.isProtected || protectedSessionHolder.isProtectedSessionAvailable()) {
 | 
					 | 
				
			||||||
            autocompleteItems.push({
 | 
					 | 
				
			||||||
                value: childTitlePath + ' (' + childNotePath + ')',
 | 
					 | 
				
			||||||
                label: childTitlePath
 | 
					 | 
				
			||||||
            });
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const childItems = await getAutocompleteItems(childNote.noteId, childNotePath, childTitlePath);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        for (const childItem of childItems) {
 | 
					 | 
				
			||||||
            autocompleteItems.push(childItem);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (parentNoteId === 'root') {
 | 
					 | 
				
			||||||
        console.log(`Generated ${autocompleteItems.length} autocomplete items`);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return autocompleteItems;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// Overrides the default autocomplete filter function to search for matched on atleast 1 word in each of the input term's words
 | 
					 | 
				
			||||||
$.ui.autocomplete.filter = (array, terms) => {
 | 
					 | 
				
			||||||
    if (!terms) {
 | 
					 | 
				
			||||||
        return array;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const startDate = new Date();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const results = [];
 | 
					 | 
				
			||||||
    const tokens = terms.toLowerCase().split(" ");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for (const item of array) {
 | 
					 | 
				
			||||||
        const lcLabel = item.label.toLowerCase();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        const found = tokens.every(token => lcLabel.indexOf(token) !== -1);
 | 
					 | 
				
			||||||
        if (!found) {
 | 
					 | 
				
			||||||
            continue;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        // this is not completely correct and might cause minor problems with note with names containing this " / "
 | 
					 | 
				
			||||||
        const lastSegmentIndex = lcLabel.lastIndexOf(" / ");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (lastSegmentIndex !== -1) {
 | 
					 | 
				
			||||||
            const lastSegment = lcLabel.substr(lastSegmentIndex + 3);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            // at least some token needs to be in the last segment (leaf note), otherwise this
 | 
					 | 
				
			||||||
            // particular note is not that interesting (query is satisfied by parent note)
 | 
					 | 
				
			||||||
            const foundInLastSegment = tokens.some(token => lastSegment.indexOf(token) !== -1);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            if (!foundInLastSegment) {
 | 
					 | 
				
			||||||
                continue;
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        results.push(item);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (results.length > 100) {
 | 
					 | 
				
			||||||
            break;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    console.log("Search took " + (new Date().getTime() - startDate.getTime()) + "ms");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return results;
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default {
 | 
					 | 
				
			||||||
    getAutocompleteItems
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
							
								
								
									
										15
									
								
								src/public/javascripts/services/bootstrap.js
									
									
									
									
										vendored
									
									
								
							
							
						
						@@ -1,10 +1,9 @@
 | 
				
			|||||||
import addLinkDialog from '../dialogs/add_link.js';
 | 
					import addLinkDialog from '../dialogs/add_link.js';
 | 
				
			||||||
import jumpToNoteDialog from '../dialogs/jump_to_note.js';
 | 
					import jumpToNoteDialog from '../dialogs/jump_to_note.js';
 | 
				
			||||||
import labelsDialog from '../dialogs/labels.js';
 | 
					import attributesDialog from '../dialogs/attributes.js';
 | 
				
			||||||
import noteRevisionsDialog from '../dialogs/note_revisions.js';
 | 
					import noteRevisionsDialog from '../dialogs/note_revisions.js';
 | 
				
			||||||
import noteSourceDialog from '../dialogs/note_source.js';
 | 
					import noteSourceDialog from '../dialogs/note_source.js';
 | 
				
			||||||
import recentChangesDialog from '../dialogs/recent_changes.js';
 | 
					import recentChangesDialog from '../dialogs/recent_changes.js';
 | 
				
			||||||
import recentNotesDialog from '../dialogs/recent_notes.js';
 | 
					 | 
				
			||||||
import optionsDialog from '../dialogs/options.js';
 | 
					import optionsDialog from '../dialogs/options.js';
 | 
				
			||||||
import sqlConsoleDialog from '../dialogs/sql_console.js';
 | 
					import sqlConsoleDialog from '../dialogs/sql_console.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -17,7 +16,7 @@ import messagingService from './messaging.js';
 | 
				
			|||||||
import noteDetailService from './note_detail.js';
 | 
					import noteDetailService from './note_detail.js';
 | 
				
			||||||
import noteType from './note_type.js';
 | 
					import noteType from './note_type.js';
 | 
				
			||||||
import protected_session from './protected_session.js';
 | 
					import protected_session from './protected_session.js';
 | 
				
			||||||
import searchTreeService from './search_tree.js';
 | 
					import searchNotesService from './search_notes.js';
 | 
				
			||||||
import ScriptApi from './script_api.js';
 | 
					import ScriptApi from './script_api.js';
 | 
				
			||||||
import ScriptContext from './script_context.js';
 | 
					import ScriptContext from './script_context.js';
 | 
				
			||||||
import sync from './sync.js';
 | 
					import sync from './sync.js';
 | 
				
			||||||
@@ -35,6 +34,9 @@ import libraryLoader from "./library_loader.js";
 | 
				
			|||||||
// required for CKEditor image upload plugin
 | 
					// required for CKEditor image upload plugin
 | 
				
			||||||
window.glob.getCurrentNode = treeService.getCurrentNode;
 | 
					window.glob.getCurrentNode = treeService.getCurrentNode;
 | 
				
			||||||
window.glob.getHeaders = server.getHeaders;
 | 
					window.glob.getHeaders = server.getHeaders;
 | 
				
			||||||
 | 
					window.glob.showAddLinkDialog = addLinkDialog.showDialog;
 | 
				
			||||||
 | 
					// this is required by CKEditor when uploading images
 | 
				
			||||||
 | 
					window.glob.noteChanged = noteDetailService.noteChanged;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// required for ESLint plugin
 | 
					// required for ESLint plugin
 | 
				
			||||||
window.glob.getCurrentNote = noteDetailService.getCurrentNote;
 | 
					window.glob.getCurrentNote = noteDetailService.getCurrentNote;
 | 
				
			||||||
@@ -46,7 +48,12 @@ window.onerror = function (msg, url, lineNo, columnNo, error) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    let message = "Uncaught error: ";
 | 
					    let message = "Uncaught error: ";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (string.indexOf("script error") > -1){
 | 
					    if (string.includes("Cannot read property 'defaultView' of undefined")) {
 | 
				
			||||||
 | 
					        // ignore this specific error which is very common but we don't know where it comes from
 | 
				
			||||||
 | 
					        // and it seems to be harmless
 | 
				
			||||||
 | 
					        return true;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else if (string.includes("script error")) {
 | 
				
			||||||
        message += 'No details available';
 | 
					        message += 'No details available';
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    else {
 | 
					    else {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,14 @@
 | 
				
			|||||||
import ScriptContext from "./script_context.js";
 | 
					import ScriptContext from "./script_context.js";
 | 
				
			||||||
import server from "./server.js";
 | 
					import server from "./server.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function executeBundle(bundle) {
 | 
					async function getAndExecuteBundle(noteId, originEntity = null) {
 | 
				
			||||||
    const apiContext = ScriptContext(bundle.note, bundle.allNotes);
 | 
					    const bundle = await server.get('script/bundle/' + noteId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    await executeBundle(bundle, originEntity);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function executeBundle(bundle, originEntity) {
 | 
				
			||||||
 | 
					    const apiContext = ScriptContext(bundle.note, bundle.allNotes, originEntity);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return await (function () {
 | 
					    return await (function () {
 | 
				
			||||||
        return eval(`const apiContext = this; (async function() { ${bundle.script}\r\n})()`);
 | 
					        return eval(`const apiContext = this; (async function() { ${bundle.script}\r\n})()`);
 | 
				
			||||||
@@ -17,7 +23,17 @@ async function executeStartupBundles() {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function executeRelationBundles(note, relationName) {
 | 
				
			||||||
 | 
					    const bundlesToRun = await server.get("script/relation/" + note.noteId + "/" + relationName);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (const bundle of bundlesToRun) {
 | 
				
			||||||
 | 
					        await executeBundle(bundle, note);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					export default {
 | 
				
			||||||
    executeBundle,
 | 
					    executeBundle,
 | 
				
			||||||
    executeStartupBundles
 | 
					    getAndExecuteBundle,
 | 
				
			||||||
 | 
					    executeStartupBundles,
 | 
				
			||||||
 | 
					    executeRelationBundles
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -94,27 +94,34 @@ const contextMenuOptions = {
 | 
				
			|||||||
        {title: "Paste into <kbd>Ctrl+V</kbd>", cmd: "pasteInto", uiIcon: "ui-icon-clipboard"},
 | 
					        {title: "Paste into <kbd>Ctrl+V</kbd>", cmd: "pasteInto", uiIcon: "ui-icon-clipboard"},
 | 
				
			||||||
        {title: "Paste after", cmd: "pasteAfter", uiIcon: "ui-icon-clipboard"},
 | 
					        {title: "Paste after", cmd: "pasteAfter", uiIcon: "ui-icon-clipboard"},
 | 
				
			||||||
        {title: "----"},
 | 
					        {title: "----"},
 | 
				
			||||||
        {title: "Export branch", cmd: "exportBranch", uiIcon: " ui-icon-arrowthick-1-ne"},
 | 
					        {title: "Export branch", cmd: "exportBranch", uiIcon: " ui-icon-arrowthick-1-ne", children: [
 | 
				
			||||||
        {title: "Import into branch", cmd: "importBranch", uiIcon: "ui-icon-arrowthick-1-sw"},
 | 
					            {title: "Native Tar", cmd: "exportBranchToTar"},
 | 
				
			||||||
 | 
					            {title: "OPML", cmd: "exportBranchToOpml"}
 | 
				
			||||||
 | 
					        ]},
 | 
				
			||||||
 | 
					        {title: "Import into branch (tar, opml)", cmd: "importBranch", uiIcon: "ui-icon-arrowthick-1-sw"},
 | 
				
			||||||
        {title: "----"},
 | 
					        {title: "----"},
 | 
				
			||||||
        {title: "Collapse branch <kbd>Alt+-</kbd>", cmd: "collapseBranch", uiIcon: "ui-icon-minus"},
 | 
					        {title: "Collapse branch <kbd>Alt+-</kbd>", cmd: "collapseBranch", uiIcon: "ui-icon-minus"},
 | 
				
			||||||
        {title: "Force note sync", cmd: "forceNoteSync", uiIcon: "ui-icon-refresh"},
 | 
					        {title: "Force note sync", cmd: "forceNoteSync", uiIcon: "ui-icon-refresh"},
 | 
				
			||||||
        {title: "Sort alphabetically <kbd>Alt+S</kbd>", cmd: "sortAlphabetically", uiIcon: " ui-icon-arrowthick-2-n-s"}
 | 
					        {title: "Sort alphabetically <kbd>Alt+S</kbd>", cmd: "sortAlphabetically", uiIcon: " ui-icon-arrowthick-2-n-s"}
 | 
				
			||||||
 | 
					 | 
				
			||||||
    ],
 | 
					    ],
 | 
				
			||||||
    beforeOpen: async (event, ui) => {
 | 
					    beforeOpen: async (event, ui) => {
 | 
				
			||||||
        const node = $.ui.fancytree.getNode(ui.target);
 | 
					        const node = $.ui.fancytree.getNode(ui.target);
 | 
				
			||||||
        const branch = await treeCache.getBranch(node.data.branchId);
 | 
					        const branch = await treeCache.getBranch(node.data.branchId);
 | 
				
			||||||
        const note = await treeCache.getNote(node.data.noteId);
 | 
					        const note = await treeCache.getNote(node.data.noteId);
 | 
				
			||||||
        const parentNote = await treeCache.getNote(branch.parentNoteId);
 | 
					        const parentNote = await treeCache.getNote(branch.parentNoteId);
 | 
				
			||||||
 | 
					        const isNotRoot = note.noteId !== 'root';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // Modify menu entries depending on node status
 | 
					        // Modify menu entries depending on node status
 | 
				
			||||||
        $tree.contextmenu("enableEntry", "pasteAfter", clipboardIds.length > 0 && (!parentNote || parentNote.type !== 'search'));
 | 
					        $tree.contextmenu("enableEntry", "insertNoteHere", isNotRoot && parentNote.type !== 'search');
 | 
				
			||||||
        $tree.contextmenu("enableEntry", "pasteInto", clipboardIds.length > 0 && note.type !== 'search');
 | 
					 | 
				
			||||||
        $tree.contextmenu("enableEntry", "insertNoteHere", !parentNote || parentNote.type !== 'search');
 | 
					 | 
				
			||||||
        $tree.contextmenu("enableEntry", "insertChildNote", note.type !== 'search');
 | 
					        $tree.contextmenu("enableEntry", "insertChildNote", note.type !== 'search');
 | 
				
			||||||
 | 
					        $tree.contextmenu("enableEntry", "delete", isNotRoot && parentNote.type !== 'search');
 | 
				
			||||||
 | 
					        $tree.contextmenu("enableEntry", "copy", isNotRoot);
 | 
				
			||||||
 | 
					        $tree.contextmenu("enableEntry", "cut", isNotRoot);
 | 
				
			||||||
 | 
					        $tree.contextmenu("enableEntry", "pasteAfter", clipboardIds.length > 0 && isNotRoot && parentNote.type !== 'search');
 | 
				
			||||||
 | 
					        $tree.contextmenu("enableEntry", "pasteInto", clipboardIds.length > 0 && note.type !== 'search');
 | 
				
			||||||
        $tree.contextmenu("enableEntry", "importBranch", note.type !== 'search');
 | 
					        $tree.contextmenu("enableEntry", "importBranch", note.type !== 'search');
 | 
				
			||||||
        $tree.contextmenu("enableEntry", "exportBranch", note.type !== 'search');
 | 
					        $tree.contextmenu("enableEntry", "exportBranch", note.type !== 'search');
 | 
				
			||||||
 | 
					        $tree.contextmenu("enableEntry", "editBranchPrefix", isNotRoot && parentNote.type !== 'search');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // Activate node on right-click
 | 
					        // Activate node on right-click
 | 
				
			||||||
        node.setActive();
 | 
					        node.setActive();
 | 
				
			||||||
@@ -159,8 +166,11 @@ const contextMenuOptions = {
 | 
				
			|||||||
        else if (ui.cmd === "delete") {
 | 
					        else if (ui.cmd === "delete") {
 | 
				
			||||||
            treeChangesService.deleteNodes(treeService.getSelectedNodes(true));
 | 
					            treeChangesService.deleteNodes(treeService.getSelectedNodes(true));
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        else if (ui.cmd === "exportBranch") {
 | 
					        else if (ui.cmd === "exportBranchToTar") {
 | 
				
			||||||
            exportService.exportBranch(node.data.noteId);
 | 
					            exportService.exportBranch(node.data.noteId, 'tar');
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else if (ui.cmd === "exportBranchToOpml") {
 | 
				
			||||||
 | 
					            exportService.exportBranch(node.data.noteId, 'opml');
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        else if (ui.cmd === "importBranch") {
 | 
					        else if (ui.cmd === "importBranch") {
 | 
				
			||||||
            exportService.importBranch(node.data.noteId);
 | 
					            exportService.importBranch(node.data.noteId);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -2,16 +2,17 @@ import utils from "./utils.js";
 | 
				
			|||||||
import treeService from "./tree.js";
 | 
					import treeService from "./tree.js";
 | 
				
			||||||
import linkService from "./link.js";
 | 
					import linkService from "./link.js";
 | 
				
			||||||
import fileService from "./file.js";
 | 
					import fileService from "./file.js";
 | 
				
			||||||
 | 
					import zoomService from "./zoom.js";
 | 
				
			||||||
import noteRevisionsDialog from "../dialogs/note_revisions.js";
 | 
					import noteRevisionsDialog from "../dialogs/note_revisions.js";
 | 
				
			||||||
import optionsDialog from "../dialogs/options.js";
 | 
					import optionsDialog from "../dialogs/options.js";
 | 
				
			||||||
import addLinkDialog from "../dialogs/add_link.js";
 | 
					import addLinkDialog from "../dialogs/add_link.js";
 | 
				
			||||||
import recentNotesDialog from "../dialogs/recent_notes.js";
 | 
					 | 
				
			||||||
import jumpToNoteDialog from "../dialogs/jump_to_note.js";
 | 
					import jumpToNoteDialog from "../dialogs/jump_to_note.js";
 | 
				
			||||||
import noteSourceDialog from "../dialogs/note_source.js";
 | 
					import noteSourceDialog from "../dialogs/note_source.js";
 | 
				
			||||||
import recentChangesDialog from "../dialogs/recent_changes.js";
 | 
					import recentChangesDialog from "../dialogs/recent_changes.js";
 | 
				
			||||||
import sqlConsoleDialog from "../dialogs/sql_console.js";
 | 
					import sqlConsoleDialog from "../dialogs/sql_console.js";
 | 
				
			||||||
import searchTreeService from "./search_tree.js";
 | 
					import searchNotesService from "./search_notes.js";
 | 
				
			||||||
import labelsDialog from "../dialogs/labels.js";
 | 
					import attributesDialog from "../dialogs/attributes.js";
 | 
				
			||||||
 | 
					import protectedSessionService from "./protected_session.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function registerEntrypoints() {
 | 
					function registerEntrypoints() {
 | 
				
			||||||
    // hot keys are active also inside inputs and content editables
 | 
					    // hot keys are active also inside inputs and content editables
 | 
				
			||||||
@@ -21,35 +22,44 @@ function registerEntrypoints() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    utils.bindShortcut('ctrl+l', addLinkDialog.showDialog);
 | 
					    utils.bindShortcut('ctrl+l', addLinkDialog.showDialog);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#jump-to-note-button").click(jumpToNoteDialog.showDialog);
 | 
					    $("#jump-to-note-dialog-button").click(jumpToNoteDialog.showDialog);
 | 
				
			||||||
    utils.bindShortcut('ctrl+j', jumpToNoteDialog.showDialog);
 | 
					    utils.bindShortcut('ctrl+j', jumpToNoteDialog.showDialog);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#show-note-revisions-button").click(noteRevisionsDialog.showCurrentNoteRevisions);
 | 
					    $("#show-note-revisions-button").click(noteRevisionsDialog.showCurrentNoteRevisions);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#show-source-button").click(noteSourceDialog.showDialog);
 | 
					    $("#show-source-button").click(noteSourceDialog.showDialog);
 | 
				
			||||||
    utils.bindShortcut('ctrl+u', noteSourceDialog.showDialog);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#recent-changes-button").click(recentChangesDialog.showDialog);
 | 
					    $("#recent-changes-button").click(recentChangesDialog.showDialog);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#recent-notes-button").click(recentNotesDialog.showDialog);
 | 
					    $("#protected-session-on").click(protectedSessionService.enterProtectedSession);
 | 
				
			||||||
    utils.bindShortcut('ctrl+e', recentNotesDialog.showDialog);
 | 
					    $("#protected-session-off").click(protectedSessionService.leaveProtectedSession);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#toggle-search-button").click(searchTreeService.toggleSearch);
 | 
					    $("#toggle-search-button").click(searchNotesService.toggleSearch);
 | 
				
			||||||
    utils.bindShortcut('ctrl+s', searchTreeService.toggleSearch);
 | 
					    utils.bindShortcut('ctrl+s', searchNotesService.toggleSearch);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $(".show-labels-button").click(labelsDialog.showDialog);
 | 
					    $(".show-attributes-button").click(attributesDialog.showDialog);
 | 
				
			||||||
    utils.bindShortcut('alt+l', labelsDialog.showDialog);
 | 
					    utils.bindShortcut('alt+a', attributesDialog.showDialog);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#options-button").click(optionsDialog.showDialog);
 | 
					    $("#options-button").click(optionsDialog.showDialog);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    utils.bindShortcut('alt+o', sqlConsoleDialog.showDialog);
 | 
					    utils.bindShortcut('alt+o', sqlConsoleDialog.showDialog);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (utils.isElectron()) {
 | 
					    if (utils.isElectron()) {
 | 
				
			||||||
 | 
					        $("#history-navigation").show();
 | 
				
			||||||
 | 
					        $("#history-back-button").click(window.history.back);
 | 
				
			||||||
 | 
					        $("#history-forward-button").click(window.history.forward);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        utils.bindShortcut('alt+left', window.history.back);
 | 
					        utils.bindShortcut('alt+left', window.history.back);
 | 
				
			||||||
        utils.bindShortcut('alt+right', window.history.forward);
 | 
					        utils.bindShortcut('alt+right', window.history.forward);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    utils.bindShortcut('alt+m', e => $(".hide-toggle").toggleClass("suppressed"));
 | 
					    utils.bindShortcut('alt+m', e => {
 | 
				
			||||||
 | 
					        $(".hide-toggle").toggle();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // when hiding switch display to block, otherwise grid still tries to display columns which shows
 | 
				
			||||||
 | 
					        // left empty column
 | 
				
			||||||
 | 
					        $("#container").css("display", $("#container").css("display") === "grid" ? "block" : "grid");
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // hide (toggle) everything except for the note content for distraction free writing
 | 
					    // hide (toggle) everything except for the note content for distraction free writing
 | 
				
			||||||
    utils.bindShortcut('alt+t', e => {
 | 
					    utils.bindShortcut('alt+t', e => {
 | 
				
			||||||
@@ -101,28 +111,11 @@ function registerEntrypoints() {
 | 
				
			|||||||
        $("#note-detail-text").focus();
 | 
					        $("#note-detail-text").focus();
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $(document).bind('keydown', 'ctrl+-', () => {
 | 
					 | 
				
			||||||
    if (utils.isElectron()) {
 | 
					    if (utils.isElectron()) {
 | 
				
			||||||
            const webFrame = require('electron').webFrame;
 | 
					        $(document).bind('keydown', 'ctrl+-', zoomService.decreaseZoomFactor);
 | 
				
			||||||
 | 
					        $(document).bind('keydown', 'ctrl+=', zoomService.increaseZoomFactor);
 | 
				
			||||||
            if (webFrame.getZoomFactor() > 0.2) {
 | 
					 | 
				
			||||||
                webFrame.setZoomFactor(webFrame.getZoomFactor() - 0.1);
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            return false;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $(document).bind('keydown', 'ctrl+=', () => {
 | 
					 | 
				
			||||||
        if (utils.isElectron()) {
 | 
					 | 
				
			||||||
            const webFrame = require('electron').webFrame;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            webFrame.setZoomFactor(webFrame.getZoomFactor() + 0.1);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            return false;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $("#note-title").bind('keydown', 'return', () => $("#note-detail-text").focus());
 | 
					    $("#note-title").bind('keydown', 'return', () => $("#note-detail-text").focus());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $("#upload-file-button").click(fileService.uploadFile);
 | 
					    $("#upload-file-button").click(fileService.uploadFile);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,9 +3,9 @@ import protectedSessionHolder from './protected_session_holder.js';
 | 
				
			|||||||
import utils from './utils.js';
 | 
					import utils from './utils.js';
 | 
				
			||||||
import server from './server.js';
 | 
					import server from './server.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function exportBranch(noteId) {
 | 
					function exportBranch(noteId, format) {
 | 
				
			||||||
    const url = utils.getHost() + "/api/notes/" + noteId + "/export?protectedSessionId="
 | 
					    const url = utils.getHost() + "/api/notes/" + noteId + "/export/" + format +
 | 
				
			||||||
        + encodeURIComponent(protectedSessionHolder.getProtectedSessionId());
 | 
					        "?protectedSessionId=" + encodeURIComponent(protectedSessionHolder.getProtectedSessionId());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    utils.download(url);
 | 
					    utils.download(url);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -29,7 +29,7 @@ $("#import-upload").change(async function() {
 | 
				
			|||||||
        type: 'POST',
 | 
					        type: 'POST',
 | 
				
			||||||
        contentType: false, // NEEDED, DON'T OMIT THIS
 | 
					        contentType: false, // NEEDED, DON'T OMIT THIS
 | 
				
			||||||
        processData: false, // NEEDED, DON'T OMIT THIS
 | 
					        processData: false, // NEEDED, DON'T OMIT THIS
 | 
				
			||||||
    });
 | 
					    }).fail((xhr, status, error) => alert('Import error: ' + xhr.responseText));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await treeService.reload();
 | 
					    await treeService.reload();
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,7 +3,7 @@ import noteDetailText from './note_detail_text.js';
 | 
				
			|||||||
import treeUtils from './tree_utils.js';
 | 
					import treeUtils from './tree_utils.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function getNotePathFromLink(url) {
 | 
					function getNotePathFromLink(url) {
 | 
				
			||||||
    const notePathMatch = /#([A-Za-z0-9/]+)$/.exec(url);
 | 
					    const notePathMatch = /#root([A-Za-z0-9/]*)$/.exec(url);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (notePathMatch === null) {
 | 
					    if (notePathMatch === null) {
 | 
				
			||||||
        return null;
 | 
					        return null;
 | 
				
			||||||
@@ -13,8 +13,8 @@ function getNotePathFromLink(url) {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function getNodePathFromLabel(label) {
 | 
					function getNotePathFromLabel(label) {
 | 
				
			||||||
    const notePathMatch = / \(([A-Za-z0-9/]+)\)/.exec(label);
 | 
					    const notePathMatch = / \(([#A-Za-z0-9/]+)\)/.exec(label);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (notePathMatch !== null) {
 | 
					    if (notePathMatch !== null) {
 | 
				
			||||||
        return notePathMatch[1];
 | 
					        return notePathMatch[1];
 | 
				
			||||||
@@ -23,7 +23,7 @@ function getNodePathFromLabel(label) {
 | 
				
			|||||||
    return null;
 | 
					    return null;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function createNoteLink(notePath, noteTitle) {
 | 
					async function createNoteLink(notePath, noteTitle = null) {
 | 
				
			||||||
    if (!noteTitle) {
 | 
					    if (!noteTitle) {
 | 
				
			||||||
        const noteId = treeUtils.getNoteIdFromNotePath(notePath);
 | 
					        const noteId = treeUtils.getNoteIdFromNotePath(notePath);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -90,6 +90,18 @@ function addTextToEditor(text) {
 | 
				
			|||||||
    doc.enqueueChanges(() => editor.data.insertText(text), doc.selection);
 | 
					    doc.enqueueChanges(() => editor.data.insertText(text), doc.selection);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ko.bindingHandlers.noteLink = {
 | 
				
			||||||
 | 
					    init: async function(element, valueAccessor, allBindings, viewModel, bindingContext) {
 | 
				
			||||||
 | 
					        const noteId = ko.unwrap(valueAccessor());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (noteId) {
 | 
				
			||||||
 | 
					            const link = await createNoteLink(noteId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            $(element).append(link);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// when click on link popup, in case of internal link, just go the the referenced note instead of default behavior
 | 
					// when click on link popup, in case of internal link, just go the the referenced note instead of default behavior
 | 
				
			||||||
// of opening the link in new window/tab
 | 
					// of opening the link in new window/tab
 | 
				
			||||||
$(document).on('click', "a[action='note']", goToLink);
 | 
					$(document).on('click', "a[action='note']", goToLink);
 | 
				
			||||||
@@ -97,7 +109,7 @@ $(document).on('click', 'div.popover-content a, div.ui-tooltip-content a', goToL
 | 
				
			|||||||
$(document).on('dblclick', '#note-detail-text a', goToLink);
 | 
					$(document).on('dblclick', '#note-detail-text a', goToLink);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					export default {
 | 
				
			||||||
    getNodePathFromLabel,
 | 
					    getNotePathFromLabel,
 | 
				
			||||||
    getNotePathFromLink,
 | 
					    getNotePathFromLink,
 | 
				
			||||||
    createNoteLink,
 | 
					    createNoteLink,
 | 
				
			||||||
    addLinkToEditor,
 | 
					    addLinkToEditor,
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,9 @@
 | 
				
			|||||||
import utils from './utils.js';
 | 
					import utils from './utils.js';
 | 
				
			||||||
import infoService from "./info.js";
 | 
					import infoService from "./info.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const $changesToPushCount = $("#changes-to-push-count");
 | 
					const $outstandingSyncsCount = $("#outstanding-syncs-count");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const syncMessageHandlers = [];
 | 
				
			||||||
const messageHandlers = [];
 | 
					const messageHandlers = [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
let ws;
 | 
					let ws;
 | 
				
			||||||
@@ -25,9 +26,17 @@ function subscribeToMessages(messageHandler) {
 | 
				
			|||||||
    messageHandlers.push(messageHandler);
 | 
					    messageHandlers.push(messageHandler);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function subscribeToSyncMessages(messageHandler) {
 | 
				
			||||||
 | 
					    syncMessageHandlers.push(messageHandler);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function handleMessage(event) {
 | 
					function handleMessage(event) {
 | 
				
			||||||
    const message = JSON.parse(event.data);
 | 
					    const message = JSON.parse(event.data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (const messageHandler of messageHandlers) {
 | 
				
			||||||
 | 
					        messageHandler(message);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (message.type === 'sync') {
 | 
					    if (message.type === 'sync') {
 | 
				
			||||||
        lastPingTs = new Date().getTime();
 | 
					        lastPingTs = new Date().getTime();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -39,11 +48,11 @@ function handleMessage(event) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        const syncData = message.data.filter(sync => sync.sourceId !== glob.sourceId);
 | 
					        const syncData = message.data.filter(sync => sync.sourceId !== glob.sourceId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        for (const messageHandler of messageHandlers) {
 | 
					        for (const syncMessageHandler of syncMessageHandlers) {
 | 
				
			||||||
            messageHandler(syncData);
 | 
					            syncMessageHandler(syncData);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $changesToPushCount.html(message.changesToPushCount);
 | 
					        $outstandingSyncsCount.html(message.outstandingSyncs);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    else if (message.type === 'sync-hash-check-failed') {
 | 
					    else if (message.type === 'sync-hash-check-failed') {
 | 
				
			||||||
        infoService.showError("Sync check failed!", 60000);
 | 
					        infoService.showError("Sync check failed!", 60000);
 | 
				
			||||||
@@ -73,26 +82,10 @@ setTimeout(() => {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    lastSyncId = glob.maxSyncIdAtLoad;
 | 
					    lastSyncId = glob.maxSyncIdAtLoad;
 | 
				
			||||||
    lastPingTs = new Date().getTime();
 | 
					    lastPingTs = new Date().getTime();
 | 
				
			||||||
    let connectionBrokenNotification = null;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    setInterval(async () => {
 | 
					    setInterval(async () => {
 | 
				
			||||||
        if (new Date().getTime() - lastPingTs > 30000) {
 | 
					        if (new Date().getTime() - lastPingTs > 30000) {
 | 
				
			||||||
            if (!connectionBrokenNotification) {
 | 
					            console.log("Lost connection to server");
 | 
				
			||||||
                connectionBrokenNotification = $.notify({
 | 
					 | 
				
			||||||
                    // options
 | 
					 | 
				
			||||||
                    message: "Lost connection to server"
 | 
					 | 
				
			||||||
                },{
 | 
					 | 
				
			||||||
                    // options
 | 
					 | 
				
			||||||
                    type: 'danger',
 | 
					 | 
				
			||||||
                    delay: 100000000 // keep it until we explicitly close it
 | 
					 | 
				
			||||||
                });
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
        else if (connectionBrokenNotification) {
 | 
					 | 
				
			||||||
            await connectionBrokenNotification.close();
 | 
					 | 
				
			||||||
            connectionBrokenNotification = null;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            infoService.showMessage("Re-connected to server");
 | 
					 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        ws.send(JSON.stringify({
 | 
					        ws.send(JSON.stringify({
 | 
				
			||||||
@@ -104,5 +97,6 @@ setTimeout(() => {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					export default {
 | 
				
			||||||
    logError,
 | 
					    logError,
 | 
				
			||||||
    subscribeToMessages
 | 
					    subscribeToMessages,
 | 
				
			||||||
 | 
					    subscribeToSyncMessages
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
							
								
								
									
										54
									
								
								src/public/javascripts/services/note_autocomplete.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,54 @@
 | 
				
			|||||||
 | 
					import server from "./server.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function initNoteAutocomplete($el) {
 | 
				
			||||||
 | 
					    if (!$el.hasClass("ui-autocomplete-input")) {
 | 
				
			||||||
 | 
					        const $showRecentNotesButton = $("<span>")
 | 
				
			||||||
 | 
					            .addClass("input-group-addon show-recent-notes-button")
 | 
				
			||||||
 | 
					            .prop("title", "Show recent notes");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $el.after($showRecentNotesButton);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $showRecentNotesButton.click(() => $el.autocomplete("search", ""));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        await $el.autocomplete({
 | 
				
			||||||
 | 
					            appendTo: $el.parent().parent(),
 | 
				
			||||||
 | 
					            source: async function (request, response) {
 | 
				
			||||||
 | 
					                const result = await server.get('autocomplete?query=' + encodeURIComponent(request.term));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if (result.length > 0) {
 | 
				
			||||||
 | 
					                    response(result.map(row => {
 | 
				
			||||||
 | 
					                        return {
 | 
				
			||||||
 | 
					                            label: row.label,
 | 
				
			||||||
 | 
					                            value: row.label + ' (' + row.value + ')'
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                    }));
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else {
 | 
				
			||||||
 | 
					                    response([{
 | 
				
			||||||
 | 
					                        label: "No results",
 | 
				
			||||||
 | 
					                        value: "No results"
 | 
				
			||||||
 | 
					                    }]);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            minLength: 0,
 | 
				
			||||||
 | 
					            change: function (event, ui) {
 | 
				
			||||||
 | 
					                $el.trigger("change");
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            select: function (event, ui) {
 | 
				
			||||||
 | 
					                if (ui.item.value === 'No results') {
 | 
				
			||||||
 | 
					                    return false;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ko.bindingHandlers.noteAutocomplete = {
 | 
				
			||||||
 | 
					    init: function(element, valueAccessor, allBindings, viewModel, bindingContext) {
 | 
				
			||||||
 | 
					        initNoteAutocomplete($(element));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default {
 | 
				
			||||||
 | 
					    initNoteAutocomplete
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -7,6 +7,7 @@ import utils from './utils.js';
 | 
				
			|||||||
import server from './server.js';
 | 
					import server from './server.js';
 | 
				
			||||||
import messagingService from "./messaging.js";
 | 
					import messagingService from "./messaging.js";
 | 
				
			||||||
import infoService from "./info.js";
 | 
					import infoService from "./info.js";
 | 
				
			||||||
 | 
					import linkService from "./link.js";
 | 
				
			||||||
import treeCache from "./tree_cache.js";
 | 
					import treeCache from "./tree_cache.js";
 | 
				
			||||||
import NoteFull from "../entities/note_full.js";
 | 
					import NoteFull from "../entities/note_full.js";
 | 
				
			||||||
import noteDetailCode from './note_detail_code.js';
 | 
					import noteDetailCode from './note_detail_code.js';
 | 
				
			||||||
@@ -14,6 +15,8 @@ import noteDetailText from './note_detail_text.js';
 | 
				
			|||||||
import noteDetailFile from './note_detail_file.js';
 | 
					import noteDetailFile from './note_detail_file.js';
 | 
				
			||||||
import noteDetailSearch from './note_detail_search.js';
 | 
					import noteDetailSearch from './note_detail_search.js';
 | 
				
			||||||
import noteDetailRender from './note_detail_render.js';
 | 
					import noteDetailRender from './note_detail_render.js';
 | 
				
			||||||
 | 
					import bundleService from "./bundle.js";
 | 
				
			||||||
 | 
					import noteAutocompleteService from "./note_autocomplete.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const $noteTitle = $("#note-title");
 | 
					const $noteTitle = $("#note-title");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -22,10 +25,13 @@ const $noteDetailComponents = $(".note-detail-component");
 | 
				
			|||||||
const $protectButton = $("#protect-button");
 | 
					const $protectButton = $("#protect-button");
 | 
				
			||||||
const $unprotectButton = $("#unprotect-button");
 | 
					const $unprotectButton = $("#unprotect-button");
 | 
				
			||||||
const $noteDetailWrapper = $("#note-detail-wrapper");
 | 
					const $noteDetailWrapper = $("#note-detail-wrapper");
 | 
				
			||||||
 | 
					const $noteDetailComponentWrapper = $("#note-detail-component-wrapper");
 | 
				
			||||||
const $noteIdDisplay = $("#note-id-display");
 | 
					const $noteIdDisplay = $("#note-id-display");
 | 
				
			||||||
const $labelList = $("#label-list");
 | 
					const $attributeList = $("#attribute-list");
 | 
				
			||||||
const $labelListInner = $("#label-list-inner");
 | 
					const $attributeListInner = $("#attribute-list-inner");
 | 
				
			||||||
const $childrenOverview = $("#children-overview");
 | 
					const $childrenOverview = $("#children-overview");
 | 
				
			||||||
 | 
					const $scriptArea = $("#note-detail-script-area");
 | 
				
			||||||
 | 
					const $promotedAttributesContainer = $("#note-detail-promoted-attributes");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
let currentNote = null;
 | 
					let currentNote = null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -116,9 +122,9 @@ async function saveNoteIfChanged() {
 | 
				
			|||||||
function setNoteBackgroundIfProtected(note) {
 | 
					function setNoteBackgroundIfProtected(note) {
 | 
				
			||||||
    const isProtected = !!note.isProtected;
 | 
					    const isProtected = !!note.isProtected;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $noteDetailWrapper.toggleClass("protected", isProtected);
 | 
					    $noteDetailComponentWrapper.toggleClass("protected", isProtected);
 | 
				
			||||||
    $protectButton.toggle(!isProtected);
 | 
					    $protectButton.toggleClass("active", isProtected);
 | 
				
			||||||
    $unprotectButton.toggle(isProtected);
 | 
					    $unprotectButton.toggleClass("active", !isProtected);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
let isNewNoteCreated = false;
 | 
					let isNewNoteCreated = false;
 | 
				
			||||||
@@ -150,6 +156,8 @@ async function loadNoteDetail(noteId) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    $noteIdDisplay.html(noteId);
 | 
					    $noteIdDisplay.html(noteId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    setNoteBackgroundIfProtected(currentNote);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await handleProtectedSession();
 | 
					    await handleProtectedSession();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $noteDetailWrapper.show();
 | 
					    $noteDetailWrapper.show();
 | 
				
			||||||
@@ -170,15 +178,18 @@ async function loadNoteDetail(noteId) {
 | 
				
			|||||||
        noteChangeDisabled = false;
 | 
					        noteChangeDisabled = false;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    setNoteBackgroundIfProtected(currentNote);
 | 
					 | 
				
			||||||
    treeService.setBranchBackgroundBasedOnProtectedStatus(noteId);
 | 
					    treeService.setBranchBackgroundBasedOnProtectedStatus(noteId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // after loading new note make sure editor is scrolled to the top
 | 
					    // after loading new note make sure editor is scrolled to the top
 | 
				
			||||||
    $noteDetailWrapper.scrollTop(0);
 | 
					    $noteDetailWrapper.scrollTop(0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const labels = await loadLabelList();
 | 
					    $scriptArea.html('');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const hideChildrenOverview = labels.some(label => label.name === 'hideChildrenOverview');
 | 
					    await bundleService.executeRelationBundles(getCurrentNote(), 'runOnNoteView');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const attributes = await loadAttributes();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const hideChildrenOverview = attributes.some(attr => attr.type === 'label' && attr.name === 'hideChildrenOverview');
 | 
				
			||||||
    await showChildrenOverview(hideChildrenOverview);
 | 
					    await showChildrenOverview(hideChildrenOverview);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -207,25 +218,207 @@ async function showChildrenOverview(hideChildrenOverview) {
 | 
				
			|||||||
    $childrenOverview.show();
 | 
					    $childrenOverview.show();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function loadLabelList() {
 | 
					async function loadAttributes() {
 | 
				
			||||||
 | 
					    $promotedAttributesContainer.empty();
 | 
				
			||||||
 | 
					    $attributeList.hide();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const noteId = getCurrentNoteId();
 | 
					    const noteId = getCurrentNoteId();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const labels = await server.get('notes/' + noteId + '/labels');
 | 
					    const attributes = await server.get('notes/' + noteId + '/attributes');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $labelListInner.html('');
 | 
					    const promoted = attributes.filter(attr => (attr.type === 'label-definition' || attr.type === 'relation-definition') && attr.value.isPromoted);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (labels.length > 0) {
 | 
					    let idx = 1;
 | 
				
			||||||
        for (const label of labels) {
 | 
					
 | 
				
			||||||
            $labelListInner.append(utils.formatLabel(label) + " ");
 | 
					    async function createRow(definitionAttr, valueAttr) {
 | 
				
			||||||
 | 
					        const definition = definitionAttr.value;
 | 
				
			||||||
 | 
					        const inputId = "promoted-input-" + idx;
 | 
				
			||||||
 | 
					        const $tr = $("<tr>");
 | 
				
			||||||
 | 
					        const $labelCell = $("<th>").append(valueAttr.name);
 | 
				
			||||||
 | 
					        const $input = $("<input>")
 | 
				
			||||||
 | 
					            .prop("id", inputId)
 | 
				
			||||||
 | 
					            .prop("tabindex", definitionAttr.position)
 | 
				
			||||||
 | 
					            .prop("attribute-id", valueAttr.isOwned ? valueAttr.attributeId : '') // if not owned, we'll force creation of a new attribute instead of updating the inherited one
 | 
				
			||||||
 | 
					            .prop("attribute-type", valueAttr.type)
 | 
				
			||||||
 | 
					            .prop("attribute-name", valueAttr.name)
 | 
				
			||||||
 | 
					            .prop("value", valueAttr.value)
 | 
				
			||||||
 | 
					            .addClass("form-control")
 | 
				
			||||||
 | 
					            .addClass("promoted-attribute-input");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        idx++;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const $inputCell = $("<td>").append($("<div>").addClass("input-group").append($input));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const $actionCell = $("<td>");
 | 
				
			||||||
 | 
					        const $multiplicityCell = $("<td>");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $tr
 | 
				
			||||||
 | 
					            .append($labelCell)
 | 
				
			||||||
 | 
					            .append($inputCell)
 | 
				
			||||||
 | 
					            .append($actionCell)
 | 
				
			||||||
 | 
					            .append($multiplicityCell);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (valueAttr.type === 'label') {
 | 
				
			||||||
 | 
					            if (definition.labelType === 'text') {
 | 
				
			||||||
 | 
					                $input.prop("type", "text");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                // no need to await for this, can be done asynchronously
 | 
				
			||||||
 | 
					                server.get('attributes/values/' + encodeURIComponent(valueAttr.name)).then(attributeValues => {
 | 
				
			||||||
 | 
					                    if (attributeValues.length === 0) {
 | 
				
			||||||
 | 
					                        return;
 | 
				
			||||||
                    }
 | 
					                    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $labelList.show();
 | 
					                    $input.autocomplete({
 | 
				
			||||||
 | 
					                        // shouldn't be required and autocomplete should just accept array of strings, but that fails
 | 
				
			||||||
 | 
					                        // because we have overriden filter() function in autocomplete.js
 | 
				
			||||||
 | 
					                        source: attributeValues.map(attribute => {
 | 
				
			||||||
 | 
					                            return {
 | 
				
			||||||
 | 
					                                attribute: attribute,
 | 
				
			||||||
 | 
					                                value: attribute
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        }),
 | 
				
			||||||
 | 
					                        minLength: 0
 | 
				
			||||||
 | 
					                    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    $input.focus(() => $input.autocomplete("search", ""));
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else if (definition.labelType === 'number') {
 | 
				
			||||||
 | 
					                $input.prop("type", "number");
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else if (definition.labelType === 'boolean') {
 | 
				
			||||||
 | 
					                $input.prop("type", "checkbox");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                if (valueAttr.value === "true") {
 | 
				
			||||||
 | 
					                    $input.prop("checked", "checked");
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            else if (definition.labelType === 'date') {
 | 
				
			||||||
 | 
					                $input.prop("type", "text");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                $input.datepicker({
 | 
				
			||||||
 | 
					                    changeMonth: true,
 | 
				
			||||||
 | 
					                    changeYear: true,
 | 
				
			||||||
 | 
					                    dateFormat: "yy-mm-dd"
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                const $todayButton = $("<button>").addClass("btn btn-small").text("Today").click(() => {
 | 
				
			||||||
 | 
					                    $input.val(utils.formatDateISO(new Date()));
 | 
				
			||||||
 | 
					                    $input.trigger("change");
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                $actionCell.append($todayButton);
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            else {
 | 
					            else {
 | 
				
			||||||
        $labelList.hide();
 | 
					                messagingService.logError("Unknown labelType=" + definitionAttr.labelType);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else if (valueAttr.type === 'relation') {
 | 
				
			||||||
 | 
					            if (valueAttr.value) {
 | 
				
			||||||
 | 
					                $input.val((await treeUtils.getNoteTitle(valueAttr.value) + " (" + valueAttr.value + ")"));
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return labels;
 | 
					            // no need to wait for this
 | 
				
			||||||
 | 
					            noteAutocompleteService.initNoteAutocomplete($input);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else {
 | 
				
			||||||
 | 
					            messagingService.logError("Unknown attribute type=" + valueAttr.type);
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (definition.multiplicityType === "multivalue") {
 | 
				
			||||||
 | 
					            const addButton = $("<span>")
 | 
				
			||||||
 | 
					                .addClass("glyphicon glyphicon-plus pointer")
 | 
				
			||||||
 | 
					                .prop("title", "Add new attribute")
 | 
				
			||||||
 | 
					                .click(async () => {
 | 
				
			||||||
 | 
					                const $new = await createRow(definitionAttr, {
 | 
				
			||||||
 | 
					                    attributeId: "",
 | 
				
			||||||
 | 
					                    type: valueAttr.type,
 | 
				
			||||||
 | 
					                    name: definitionAttr.name,
 | 
				
			||||||
 | 
					                    value: ""
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                $tr.after($new);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                $new.find('input').focus();
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            const removeButton = $("<span>")
 | 
				
			||||||
 | 
					                .addClass("glyphicon glyphicon-trash pointer")
 | 
				
			||||||
 | 
					                .prop("title", "Remove this attribute")
 | 
				
			||||||
 | 
					                .click(async () => {
 | 
				
			||||||
 | 
					                if (valueAttr.attributeId) {
 | 
				
			||||||
 | 
					                    await server.remove("notes/" + noteId + "/attributes/" + valueAttr.attributeId);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                $tr.remove();
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            $multiplicityCell.append(addButton).append("   ").append(removeButton);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return $tr;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (promoted.length > 0) {
 | 
				
			||||||
 | 
					        const $tbody = $("<tbody>");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (const definitionAttr of promoted) {
 | 
				
			||||||
 | 
					            const definitionType = definitionAttr.type;
 | 
				
			||||||
 | 
					            const valueType = definitionType.substr(0, definitionType.length - 11);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            let valueAttrs = attributes.filter(el => el.name === definitionAttr.name && el.type === valueType);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (valueAttrs.length === 0) {
 | 
				
			||||||
 | 
					                valueAttrs.push({
 | 
				
			||||||
 | 
					                    attributeId: "",
 | 
				
			||||||
 | 
					                    type: valueType,
 | 
				
			||||||
 | 
					                    name: definitionAttr.name,
 | 
				
			||||||
 | 
					                    value: ""
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (definitionAttr.value.multiplicityType === 'singlevalue') {
 | 
				
			||||||
 | 
					                valueAttrs = valueAttrs.slice(0, 1);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            for (const valueAttr of valueAttrs) {
 | 
				
			||||||
 | 
					                const $tr = await createRow(definitionAttr, valueAttr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                $tbody.append($tr);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // we replace the whole content in one step so there can't be any race conditions
 | 
				
			||||||
 | 
					        // (previously we saw promoted attributes doubling)
 | 
				
			||||||
 | 
					        $promotedAttributesContainer.empty().append($tbody);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else {
 | 
				
			||||||
 | 
					        $attributeListInner.html('');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (attributes.length > 0) {
 | 
				
			||||||
 | 
					            for (const attribute of attributes) {
 | 
				
			||||||
 | 
					                if (attribute.type === 'label') {
 | 
				
			||||||
 | 
					                    $attributeListInner.append(utils.formatLabel(attribute) + " ");
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else if (attribute.type === 'relation') {
 | 
				
			||||||
 | 
					                    $attributeListInner.append(attribute.name + "=");
 | 
				
			||||||
 | 
					                    $attributeListInner.append(await linkService.createNoteLink(attribute.value));
 | 
				
			||||||
 | 
					                    $attributeListInner.append(" ");
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else if (attribute.type === 'label-definition' || attribute.type === 'relation-definition') {
 | 
				
			||||||
 | 
					                    $attributeListInner.append(attribute.name + " definition ");
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else {
 | 
				
			||||||
 | 
					                    messagingService.logError("Unknown attr type: " + attribute.type);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            $attributeList.show();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return attributes;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function loadNote(noteId) {
 | 
					async function loadNote(noteId) {
 | 
				
			||||||
@@ -240,7 +433,7 @@ function focus() {
 | 
				
			|||||||
    getComponent(note.type).focus();
 | 
					    getComponent(note.type).focus();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
messagingService.subscribeToMessages(syncData => {
 | 
					messagingService.subscribeToSyncMessages(syncData => {
 | 
				
			||||||
    if (syncData.some(sync => sync.entityName === 'notes' && sync.entityId === getCurrentNoteId())) {
 | 
					    if (syncData.some(sync => sync.entityName === 'notes' && sync.entityId === getCurrentNoteId())) {
 | 
				
			||||||
        infoService.showMessage('Reloading note because of background changes');
 | 
					        infoService.showMessage('Reloading note because of background changes');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -248,6 +441,35 @@ messagingService.subscribeToMessages(syncData => {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$promotedAttributesContainer.on('change', '.promoted-attribute-input', async event => {
 | 
				
			||||||
 | 
					    const $attr = $(event.target);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let value;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if ($attr.prop("type") === "checkbox") {
 | 
				
			||||||
 | 
					        value = $attr.is(':checked') ? "true" : "false";
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else if ($attr.prop("attribute-type") === "relation") {
 | 
				
			||||||
 | 
					        if ($attr.val()) {
 | 
				
			||||||
 | 
					            value = treeUtils.getNoteIdFromNotePath(linkService.getNotePathFromLabel($attr.val()));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else {
 | 
				
			||||||
 | 
					        value = $attr.val();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const result = await server.put("notes/" + getCurrentNoteId() + "/attribute", {
 | 
				
			||||||
 | 
					        attributeId: $attr.prop("attribute-id"),
 | 
				
			||||||
 | 
					        type: $attr.prop("attribute-type"),
 | 
				
			||||||
 | 
					        name: $attr.prop("attribute-name"),
 | 
				
			||||||
 | 
					        value: value
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $attr.prop("attribute-id", result.attributeId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    infoService.showMessage("Attribute has been saved.");
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
$(document).ready(() => {
 | 
					$(document).ready(() => {
 | 
				
			||||||
    $noteTitle.on('input', () => {
 | 
					    $noteTitle.on('input', () => {
 | 
				
			||||||
        noteChanged();
 | 
					        noteChanged();
 | 
				
			||||||
@@ -276,7 +498,7 @@ export default {
 | 
				
			|||||||
    getCurrentNoteId,
 | 
					    getCurrentNoteId,
 | 
				
			||||||
    newNoteCreated,
 | 
					    newNoteCreated,
 | 
				
			||||||
    focus,
 | 
					    focus,
 | 
				
			||||||
    loadLabelList,
 | 
					    loadAttributes,
 | 
				
			||||||
    saveNote,
 | 
					    saveNote,
 | 
				
			||||||
    saveNoteIfChanged,
 | 
					    saveNoteIfChanged,
 | 
				
			||||||
    noteChanged
 | 
					    noteChanged
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -32,7 +32,7 @@ async function show() {
 | 
				
			|||||||
            lint: true,
 | 
					            lint: true,
 | 
				
			||||||
            gutters: ["CodeMirror-lint-markers"],
 | 
					            gutters: ["CodeMirror-lint-markers"],
 | 
				
			||||||
            lineNumbers: true,
 | 
					            lineNumbers: true,
 | 
				
			||||||
            tabindex: 2 // so that tab from title will lead to code editor focus
 | 
					            tabindex: 100
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        codeEditor.on('change', noteDetailService.noteChanged);
 | 
					        codeEditor.on('change', noteDetailService.noteChanged);
 | 
				
			||||||
@@ -64,16 +64,18 @@ function focus() {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function executeCurrentNote() {
 | 
					async function executeCurrentNote() {
 | 
				
			||||||
    if (noteDetailService.getCurrentNoteType() === 'code') {
 | 
					    // ctrl+enter is also used elsewhere so make sure we're running only when appropriate
 | 
				
			||||||
 | 
					    if (noteDetailService.getCurrentNoteType() !== 'code') {
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // make sure note is saved so we load latest changes
 | 
					    // make sure note is saved so we load latest changes
 | 
				
			||||||
    await noteDetailService.saveNoteIfChanged();
 | 
					    await noteDetailService.saveNoteIfChanged();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const currentNote = noteDetailService.getCurrentNote();
 | 
					    const currentNote = noteDetailService.getCurrentNote();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (currentNote.mime.endsWith("env=frontend")) {
 | 
					    if (currentNote.mime.endsWith("env=frontend")) {
 | 
				
			||||||
            const bundle = await server.get('script/bundle/' + noteDetailService.getCurrentNoteId());
 | 
					        await bundleService.getAndExecuteBundle(noteDetailService.getCurrentNoteId());
 | 
				
			||||||
 | 
					 | 
				
			||||||
            bundleService.executeBundle(bundle);
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (currentNote.mime.endsWith("env=backend")) {
 | 
					    if (currentNote.mime.endsWith("env=backend")) {
 | 
				
			||||||
@@ -81,7 +83,6 @@ async function executeCurrentNote() {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    infoService.showMessage("Note executed");
 | 
					    infoService.showMessage("Note executed");
 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
$(document).bind('keydown', "ctrl+return", executeCurrentNote);
 | 
					$(document).bind('keydown', "ctrl+return", executeCurrentNote);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -14,13 +14,13 @@ const $fileOpen = $("#file-open");
 | 
				
			|||||||
async function show() {
 | 
					async function show() {
 | 
				
			||||||
    const currentNote = noteDetailService.getCurrentNote();
 | 
					    const currentNote = noteDetailService.getCurrentNote();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const labels = await server.get('notes/' + currentNote.noteId + '/labels');
 | 
					    const attributes = await server.get('notes/' + currentNote.noteId + '/attributes');
 | 
				
			||||||
    const labelMap = utils.toObject(labels, l => [l.name, l.value]);
 | 
					    const attributeMap = utils.toObject(attributes, l => [l.name, l.value]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $noteDetailFile.show();
 | 
					    $noteDetailFile.show();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $fileFileName.text(labelMap.original_file_name);
 | 
					    $fileFileName.text(attributeMap.original_file_name);
 | 
				
			||||||
    $fileFileSize.text(labelMap.file_size + " bytes");
 | 
					    $fileFileSize.text(attributeMap.file_size + " bytes");
 | 
				
			||||||
    $fileFileType.text(currentNote.mime);
 | 
					    $fileFileType.text(currentNote.mime);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,21 +1,73 @@
 | 
				
			|||||||
import bundleService from "./bundle.js";
 | 
					import bundleService from "./bundle.js";
 | 
				
			||||||
import server from "./server.js";
 | 
					import server from "./server.js";
 | 
				
			||||||
import noteDetailService from "./note_detail.js";
 | 
					import noteDetailService from "./note_detail.js";
 | 
				
			||||||
 | 
					import noteDetailCodeService from "./note_detail_code.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const $noteDetailCode = $('#note-detail-code');
 | 
				
			||||||
const $noteDetailRender = $('#note-detail-render');
 | 
					const $noteDetailRender = $('#note-detail-render');
 | 
				
			||||||
 | 
					const $toggleEditButton = $('#toggle-edit-button');
 | 
				
			||||||
 | 
					const $renderButton = $('#render-button');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					let codeEditorInitialized;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function show() {
 | 
					async function show() {
 | 
				
			||||||
 | 
					    codeEditorInitialized = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $noteDetailRender.show();
 | 
					    $noteDetailRender.show();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    await render();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function toggleEdit() {
 | 
				
			||||||
 | 
					    if ($noteDetailCode.is(":visible")) {
 | 
				
			||||||
 | 
					        $noteDetailCode.hide();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else {
 | 
				
			||||||
 | 
					        if (!codeEditorInitialized) {
 | 
				
			||||||
 | 
					            await noteDetailCodeService.show();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            // because we can't properly scroll only the editor without scrolling the rendering
 | 
				
			||||||
 | 
					            // we limit its height
 | 
				
			||||||
 | 
					            $noteDetailCode.find('.CodeMirror').css('height', '300');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            codeEditorInitialized = true;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else {
 | 
				
			||||||
 | 
					            $noteDetailCode.show();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$toggleEditButton.click(toggleEdit);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$renderButton.click(render);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function render() {
 | 
				
			||||||
 | 
					    // ctrl+enter is also used elsewhere so make sure we're running only when appropriate
 | 
				
			||||||
 | 
					    if (noteDetailService.getCurrentNoteType() !== 'render') {
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (codeEditorInitialized) {
 | 
				
			||||||
 | 
					        await noteDetailService.saveNoteIfChanged();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const bundle = await server.get('script/bundle/' + noteDetailService.getCurrentNoteId());
 | 
					    const bundle = await server.get('script/bundle/' + noteDetailService.getCurrentNoteId());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $noteDetailRender.html(bundle.html);
 | 
					    $noteDetailRender.html(bundle.html);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // if the note is empty, it doesn't make sense to do render-only since nothing will be rendered
 | 
				
			||||||
 | 
					    if (!bundle.html.trim()) {
 | 
				
			||||||
 | 
					        toggleEdit();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await bundleService.executeBundle(bundle);
 | 
					    await bundleService.executeBundle(bundle);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$(document).bind('keydown', "ctrl+return", render);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					export default {
 | 
				
			||||||
    show,
 | 
					    show,
 | 
				
			||||||
    getContent: () => null,
 | 
					    getContent: noteDetailCodeService.getContent,
 | 
				
			||||||
    focus: () => null
 | 
					    focus: () => null
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -11,7 +11,13 @@ async function show() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        textEditor = await BalloonEditor.create($noteDetailText[0], {});
 | 
					        textEditor = await BalloonEditor.create($noteDetailText[0], {});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        textEditor.model.document.on('change', noteDetailService.noteChanged);
 | 
					        textEditor.model.document.on('change', () => {
 | 
				
			||||||
 | 
					                // change is triggered on just marker/selection changes which is not interesting for us
 | 
				
			||||||
 | 
					                if (textEditor.model.document.differ.getChanges().length > 0) {
 | 
				
			||||||
 | 
					                    noteDetailService.noteChanged();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    textEditor.setData(noteDetailService.getCurrentNote().content);
 | 
					    textEditor.setData(noteDetailService.getCurrentNote().content);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -4,6 +4,9 @@ import server from './server.js';
 | 
				
			|||||||
import infoService from "./info.js";
 | 
					import infoService from "./info.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const $executeScriptButton = $("#execute-script-button");
 | 
					const $executeScriptButton = $("#execute-script-button");
 | 
				
			||||||
 | 
					const $toggleEditButton = $('#toggle-edit-button');
 | 
				
			||||||
 | 
					const $renderButton = $('#render-button');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const noteTypeModel = new NoteTypeModel();
 | 
					const noteTypeModel = new NoteTypeModel();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function NoteTypeModel() {
 | 
					function NoteTypeModel() {
 | 
				
			||||||
@@ -107,7 +110,7 @@ function NoteTypeModel() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    this.selectRender = function() {
 | 
					    this.selectRender = function() {
 | 
				
			||||||
        self.type('render');
 | 
					        self.type('render');
 | 
				
			||||||
        self.mime('');
 | 
					        self.mime('text/html');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        save();
 | 
					        save();
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
@@ -128,6 +131,9 @@ function NoteTypeModel() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    this.updateExecuteScriptButtonVisibility = function() {
 | 
					    this.updateExecuteScriptButtonVisibility = function() {
 | 
				
			||||||
        $executeScriptButton.toggle(self.mime().startsWith('application/javascript'));
 | 
					        $executeScriptButton.toggle(self.mime().startsWith('application/javascript'));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $toggleEditButton.toggle(self.type() === 'render');
 | 
				
			||||||
 | 
					        $renderButton.toggle(self.type() === 'render');
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										9
									
								
								src/public/javascripts/services/options_init.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,9 @@
 | 
				
			|||||||
 | 
					import server from "./server.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const optionsReady = new Promise((resolve, reject) => {
 | 
				
			||||||
 | 
					    $(document).ready(() => server.get('options').then(resolve));
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default {
 | 
				
			||||||
 | 
					    optionsReady
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -11,9 +11,23 @@ const $password = $("#protected-session-password");
 | 
				
			|||||||
const $noteDetailWrapper = $("#note-detail-wrapper");
 | 
					const $noteDetailWrapper = $("#note-detail-wrapper");
 | 
				
			||||||
const $protectButton = $("#protect-button");
 | 
					const $protectButton = $("#protect-button");
 | 
				
			||||||
const $unprotectButton = $("#unprotect-button");
 | 
					const $unprotectButton = $("#unprotect-button");
 | 
				
			||||||
 | 
					const $protectedSessionOnButton = $("#protected-session-on");
 | 
				
			||||||
 | 
					const $protectedSessionOffButton = $("#protected-session-off");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
let protectedSessionDeferred = null;
 | 
					let protectedSessionDeferred = null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function enterProtectedSession() {
 | 
				
			||||||
 | 
					    if (!protectedSessionHolder.isProtectedSessionAvailable()) {
 | 
				
			||||||
 | 
					        await ensureProtectedSession(true, true);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function leaveProtectedSession() {
 | 
				
			||||||
 | 
					    if (protectedSessionHolder.isProtectedSessionAvailable()) {
 | 
				
			||||||
 | 
					        utils.reloadApp();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function ensureProtectedSession(requireProtectedSession, modal) {
 | 
					function ensureProtectedSession(requireProtectedSession, modal) {
 | 
				
			||||||
    const dfd = $.Deferred();
 | 
					    const dfd = $.Deferred();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -25,7 +39,10 @@ function ensureProtectedSession(requireProtectedSession, modal) {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $dialog.dialog({
 | 
					        $dialog.dialog({
 | 
				
			||||||
            modal: modal,
 | 
					            // modal: modal,
 | 
				
			||||||
 | 
					            // everything is now non-modal, because modal dialog caused weird high CPU usage on opening
 | 
				
			||||||
 | 
					            // and tearing of text input
 | 
				
			||||||
 | 
					            modal: false,
 | 
				
			||||||
            width: 400,
 | 
					            width: 400,
 | 
				
			||||||
            open: () => {
 | 
					            open: () => {
 | 
				
			||||||
                if (!modal) {
 | 
					                if (!modal) {
 | 
				
			||||||
@@ -46,7 +63,7 @@ async function setupProtectedSession() {
 | 
				
			|||||||
    const password = $password.val();
 | 
					    const password = $password.val();
 | 
				
			||||||
    $password.val("");
 | 
					    $password.val("");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const response = await enterProtectedSession(password);
 | 
					    const response = await enterProtectedSessionOnServer(password);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!response.success) {
 | 
					    if (!response.success) {
 | 
				
			||||||
        infoService.showError("Wrong password.");
 | 
					        infoService.showError("Wrong password.");
 | 
				
			||||||
@@ -66,8 +83,10 @@ async function setupProtectedSession() {
 | 
				
			|||||||
        $noteDetailWrapper.show();
 | 
					        $noteDetailWrapper.show();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        protectedSessionDeferred.resolve();
 | 
					        protectedSessionDeferred.resolve();
 | 
				
			||||||
 | 
					 | 
				
			||||||
        protectedSessionDeferred = null;
 | 
					        protectedSessionDeferred = null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $protectedSessionOnButton.addClass('active');
 | 
				
			||||||
 | 
					        $protectedSessionOffButton.removeClass('active');
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -81,13 +100,17 @@ function ensureDialogIsClosed() {
 | 
				
			|||||||
    $password.val('');
 | 
					    $password.val('');
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function enterProtectedSession(password) {
 | 
					async function enterProtectedSessionOnServer(password) {
 | 
				
			||||||
    return await server.post('login/protected', {
 | 
					    return await server.post('login/protected', {
 | 
				
			||||||
        password: password
 | 
					        password: password
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function protectNoteAndSendToServer() {
 | 
					async function protectNoteAndSendToServer() {
 | 
				
			||||||
 | 
					    if (noteDetailService.getCurrentNote().isProtected) {
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await ensureProtectedSession(true, true);
 | 
					    await ensureProtectedSession(true, true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const note = noteDetailService.getCurrentNote();
 | 
					    const note = noteDetailService.getCurrentNote();
 | 
				
			||||||
@@ -101,6 +124,10 @@ async function protectNoteAndSendToServer() {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function unprotectNoteAndSendToServer() {
 | 
					async function unprotectNoteAndSendToServer() {
 | 
				
			||||||
 | 
					    if (!noteDetailService.getCurrentNote().isProtected) {
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await ensureProtectedSession(true, true);
 | 
					    await ensureProtectedSession(true, true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const note = noteDetailService.getCurrentNote();
 | 
					    const note = noteDetailService.getCurrentNote();
 | 
				
			||||||
@@ -138,5 +165,7 @@ export default {
 | 
				
			|||||||
    protectNoteAndSendToServer,
 | 
					    protectNoteAndSendToServer,
 | 
				
			||||||
    unprotectNoteAndSendToServer,
 | 
					    unprotectNoteAndSendToServer,
 | 
				
			||||||
    protectBranch,
 | 
					    protectBranch,
 | 
				
			||||||
    ensureDialogIsClosed
 | 
					    ensureDialogIsClosed,
 | 
				
			||||||
 | 
					    enterProtectedSession,
 | 
				
			||||||
 | 
					    leaveProtectedSession
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
@@ -1,13 +1,11 @@
 | 
				
			|||||||
import utils from "./utils.js";
 | 
					import utils from "./utils.js";
 | 
				
			||||||
import server from "./server.js";
 | 
					import optionsInitService from './options_init.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
let lastProtectedSessionOperationDate = null;
 | 
					let lastProtectedSessionOperationDate = null;
 | 
				
			||||||
let protectedSessionTimeout = null;
 | 
					let protectedSessionTimeout = null;
 | 
				
			||||||
let protectedSessionId = null;
 | 
					let protectedSessionId = null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
$(document).ready(() => {
 | 
					optionsInitService.optionsReady.then(options => protectedSessionTimeout = options.protectedSessionTimeout);
 | 
				
			||||||
    server.get('options').then(options => protectedSessionTimeout = options.protectedSessionTimeout);
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
setInterval(() => {
 | 
					setInterval(() => {
 | 
				
			||||||
    if (lastProtectedSessionOperationDate !== null && new Date().getTime() - lastProtectedSessionOperationDate.getTime() > protectedSessionTimeout * 1000) {
 | 
					    if (lastProtectedSessionOperationDate !== null && new Date().getTime() - lastProtectedSessionOperationDate.getTime() > protectedSessionTimeout * 1000) {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -2,14 +2,21 @@ import treeService from './tree.js';
 | 
				
			|||||||
import server from './server.js';
 | 
					import server from './server.js';
 | 
				
			||||||
import utils from './utils.js';
 | 
					import utils from './utils.js';
 | 
				
			||||||
import infoService from './info.js';
 | 
					import infoService from './info.js';
 | 
				
			||||||
 | 
					import linkService from './link.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function ScriptApi(startNote, currentNote) {
 | 
					function ScriptApi(startNote, currentNote, originEntity = null) {
 | 
				
			||||||
    const $pluginButtons = $("#plugin-buttons");
 | 
					    const $pluginButtons = $("#plugin-buttons");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async function activateNote(notePath) {
 | 
					    async function activateNote(notePath) {
 | 
				
			||||||
        await treeService.activateNode(notePath);
 | 
					        await treeService.activateNode(notePath);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async function activateNewNote(notePath) {
 | 
				
			||||||
 | 
					        await treeService.reload();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        await treeService.activateNode(notePath, true);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    function addButtonToToolbar(buttonId, button) {
 | 
					    function addButtonToToolbar(buttonId, button) {
 | 
				
			||||||
        $("#" + buttonId).remove();
 | 
					        $("#" + buttonId).remove();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -42,7 +49,9 @@ function ScriptApi(startNote, currentNote) {
 | 
				
			|||||||
            script: script,
 | 
					            script: script,
 | 
				
			||||||
            params: prepareParams(params),
 | 
					            params: prepareParams(params),
 | 
				
			||||||
            startNoteId: startNote.noteId,
 | 
					            startNoteId: startNote.noteId,
 | 
				
			||||||
            currentNoteId: currentNote.noteId
 | 
					            currentNoteId: currentNote.noteId,
 | 
				
			||||||
 | 
					            originEntityName: originEntity ? originEntity.constructor.tableName : null,
 | 
				
			||||||
 | 
					            originEntityId: originEntity ? originEntity.noteId : null
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return ret.executionResult;
 | 
					        return ret.executionResult;
 | 
				
			||||||
@@ -51,15 +60,19 @@ function ScriptApi(startNote, currentNote) {
 | 
				
			|||||||
    return {
 | 
					    return {
 | 
				
			||||||
        startNote: startNote,
 | 
					        startNote: startNote,
 | 
				
			||||||
        currentNote: currentNote,
 | 
					        currentNote: currentNote,
 | 
				
			||||||
 | 
					        originEntity: originEntity,
 | 
				
			||||||
        addButtonToToolbar,
 | 
					        addButtonToToolbar,
 | 
				
			||||||
        activateNote,
 | 
					        activateNote,
 | 
				
			||||||
 | 
					        activateNewNote,
 | 
				
			||||||
        getInstanceName: () => window.glob.instanceName,
 | 
					        getInstanceName: () => window.glob.instanceName,
 | 
				
			||||||
        runOnServer,
 | 
					        runOnServer,
 | 
				
			||||||
        formatDateISO: utils.formatDateISO,
 | 
					        formatDateISO: utils.formatDateISO,
 | 
				
			||||||
        parseDate: utils.parseDate,
 | 
					        parseDate: utils.parseDate,
 | 
				
			||||||
        showMessage: infoService.showMessage,
 | 
					        showMessage: infoService.showMessage,
 | 
				
			||||||
        showError: infoService.showError,
 | 
					        showError: infoService.showError,
 | 
				
			||||||
        reloadTree: treeService.reload
 | 
					        reloadTree: treeService.reload, // deprecated
 | 
				
			||||||
 | 
					        refreshTree: treeService.reload,
 | 
				
			||||||
 | 
					        createNoteLink: linkService.createNoteLink
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,13 +1,13 @@
 | 
				
			|||||||
import ScriptApi from './script_api.js';
 | 
					import ScriptApi from './script_api.js';
 | 
				
			||||||
import utils from './utils.js';
 | 
					import utils from './utils.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function ScriptContext(startNote, allNotes) {
 | 
					function ScriptContext(startNote, allNotes, originEntity = null) {
 | 
				
			||||||
    const modules = {};
 | 
					    const modules = {};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return {
 | 
					    return {
 | 
				
			||||||
        modules: modules,
 | 
					        modules: modules,
 | 
				
			||||||
        notes: utils.toObject(allNotes, note => [note.noteId, note]),
 | 
					        notes: utils.toObject(allNotes, note => [note.noteId, note]),
 | 
				
			||||||
        apis: utils.toObject(allNotes, note => [note.noteId, ScriptApi(startNote, note)]),
 | 
					        apis: utils.toObject(allNotes, note => [note.noteId, ScriptApi(startNote, note, originEntity)]),
 | 
				
			||||||
        require: moduleNoteIds => {
 | 
					        require: moduleNoteIds => {
 | 
				
			||||||
            return moduleName => {
 | 
					            return moduleName => {
 | 
				
			||||||
                const candidates = allNotes.filter(note => moduleNoteIds.includes(note.noteId));
 | 
					                const candidates = allNotes.filter(note => moduleNoteIds.includes(note.noteId));
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,5 +1,6 @@
 | 
				
			|||||||
import treeService from './tree.js';
 | 
					import treeService from './tree.js';
 | 
				
			||||||
import server from './server.js';
 | 
					import server from './server.js';
 | 
				
			||||||
 | 
					import treeUtils from "./tree_utils.js";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const $tree = $("#tree");
 | 
					const $tree = $("#tree");
 | 
				
			||||||
const $searchInput = $("input[name='search-text']");
 | 
					const $searchInput = $("input[name='search-text']");
 | 
				
			||||||
@@ -7,40 +8,62 @@ const $resetSearchButton = $("#reset-search-button");
 | 
				
			|||||||
const $doSearchButton = $("#do-search-button");
 | 
					const $doSearchButton = $("#do-search-button");
 | 
				
			||||||
const $saveSearchButton = $("#save-search-button");
 | 
					const $saveSearchButton = $("#save-search-button");
 | 
				
			||||||
const $searchBox = $("#search-box");
 | 
					const $searchBox = $("#search-box");
 | 
				
			||||||
 | 
					const $searchResults = $("#search-results");
 | 
				
			||||||
 | 
					const $searchResultsInner = $("#search-results-inner");
 | 
				
			||||||
 | 
					const $closeSearchButton = $("#close-search-button");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function showSearch() {
 | 
				
			||||||
 | 
					    $searchBox.show();
 | 
				
			||||||
 | 
					    $searchInput.focus();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function hideSearch() {
 | 
				
			||||||
 | 
					    resetSearch();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    $searchResults.hide();
 | 
				
			||||||
 | 
					    $searchBox.hide();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function toggleSearch() {
 | 
					function toggleSearch() {
 | 
				
			||||||
    if ($searchBox.is(":hidden")) {
 | 
					    if ($searchBox.is(":hidden")) {
 | 
				
			||||||
        $searchBox.show();
 | 
					        showSearch();
 | 
				
			||||||
        $searchInput.focus();
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    else {
 | 
					    else {
 | 
				
			||||||
        resetSearch();
 | 
					        hideSearch();
 | 
				
			||||||
 | 
					 | 
				
			||||||
        $searchBox.hide();
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function resetSearch() {
 | 
					function resetSearch() {
 | 
				
			||||||
    $searchInput.val("");
 | 
					    $searchInput.val("");
 | 
				
			||||||
 | 
					 | 
				
			||||||
    getTree().clearFilter();
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function getTree() {
 | 
					function getTree() {
 | 
				
			||||||
    return $tree.fancytree('getTree');
 | 
					    return $tree.fancytree('getTree');
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function doSearch() {
 | 
					async function doSearch(searchText) {
 | 
				
			||||||
    const searchText = $searchInput.val();
 | 
					    if (searchText) {
 | 
				
			||||||
 | 
					        $searchInput.val(searchText);
 | 
				
			||||||
    const noteIds = await server.get('search/' + encodeURIComponent(searchText));
 | 
					    }
 | 
				
			||||||
 | 
					    else {
 | 
				
			||||||
    for (const noteId of noteIds) {
 | 
					        searchText = $searchInput.val();
 | 
				
			||||||
        await treeService.expandToNote(noteId, {noAnimation: true, noEvents: true});
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Pass a string to perform case insensitive matching
 | 
					    const results = await server.get('search/' + encodeURIComponent(searchText));
 | 
				
			||||||
    getTree().filterBranches(node => noteIds.includes(node.data.noteId));
 | 
					
 | 
				
			||||||
 | 
					    $searchResultsInner.empty();
 | 
				
			||||||
 | 
					    $searchResults.show();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (const result of results) {
 | 
				
			||||||
 | 
					        const link = $('<a>', {
 | 
				
			||||||
 | 
					            href: 'javascript:',
 | 
				
			||||||
 | 
					            text: result.title
 | 
				
			||||||
 | 
					        }).attr('action', 'note').attr('note-path', result.path);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        const $result = $('<li>').append(link);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        $searchResultsInner.append($result);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function saveSearch() {
 | 
					async function saveSearch() {
 | 
				
			||||||
@@ -71,6 +94,11 @@ $resetSearchButton.click(resetSearch);
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
$saveSearchButton.click(saveSearch);
 | 
					$saveSearchButton.click(saveSearch);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$closeSearchButton.click(hideSearch);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					export default {
 | 
				
			||||||
    toggleSearch
 | 
					    toggleSearch,
 | 
				
			||||||
 | 
					    resetSearch,
 | 
				
			||||||
 | 
					    showSearch,
 | 
				
			||||||
 | 
					    doSearch
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||