Squashed '.obsidian/plugins/obsidian-desmos/' content from commit d760d65
git-subtree-dir: .obsidian/plugins/obsidian-desmos git-subtree-split: d760d65452fdf4629227e7945b90a004a4c57b60
This commit is contained in:
commit
0af3fc8ba6
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
@ -0,0 +1 @@
|
||||
/main.js linguist-generated
|
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
/data.json
|
||||
/node_modules/
|
3
.prettierrc
Normal file
3
.prettierrc
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"tabWidth": 4
|
||||
}
|
95
README.md
Normal file
95
README.md
Normal file
@ -0,0 +1,95 @@
|
||||
# Obsidian Desmos
|
||||
|
||||
Render [Desmos](https://www.desmos.com/calculator) graphs right inside your notes.
|
||||
|
||||
# Installation
|
||||
|
||||
## Using Git
|
||||
|
||||
If your vault is hosted using git then congratulations, you can use the easy method. Run
|
||||
`git subtree add --prefix .obsidian/plugins/obsidian-desmos https://github.com/Nigecat/obsidian-desmos master --squash`
|
||||
to add the plugin to the vault in the current working directory of your terminal.
|
||||
You can then run
|
||||
`git subtree pull --prefix .obsidian/plugins/obsidian-desmos https://github.com/Nigecat/obsidian-desmos master --squash`
|
||||
to update the plugin to the latest version (from the same working directory).
|
||||
|
||||
## Using anything else
|
||||
|
||||
Alternatively, if you do not use git or are not comfortable using the terminal, you can manually install the plugin. Download [manifest.json](manifest.json), [versions.json](versions.json), and [main.js](main.js) and place them in `<vault>/.obsidian/plugins/obsidian-desmos` (you may have to create any missing folders).
|
||||
This process must be repeated to update the application.
|
||||
|
||||
# Usage
|
||||
|
||||
The most basic usage of this plugin involves creating a codeblock with the tag `desmos-graph` and placing the equations you wish to graph in the body:
|
||||
|
||||
````
|
||||
```desmos-graph
|
||||
y=x
|
||||
```
|
||||
````
|
||||
|
||||
Equations use the [LaTeX math](https://en.wikibooks.org/wiki/LaTeX/Mathematics) format and you can graph multiple equations by placing each one on a seperate line:
|
||||
|
||||
````
|
||||
```desmos-graph
|
||||
y=\sin(x)
|
||||
y=\frac{1}{x}
|
||||
```
|
||||
````
|
||||
|
||||
You can restrict the bounds of the graph and apply other settings by placing a `---` seperator before your equations. The content before it must be a set of `key=value` pairs seperated by either **newlines or semicolons** (or both):
|
||||
|
||||
````
|
||||
```desmos-graph
|
||||
boundary_left=0; boundary_right=100;
|
||||
boundary_top=10; boundary_bottom=-10;
|
||||
---
|
||||
y=\sin(x)
|
||||
```
|
||||
````
|
||||
|
||||
You can set the dimensions of the rendered image by using the `height` and `width` fields.
|
||||
|
||||
#### Restrictions
|
||||
|
||||
Note that graph restrictions follow the same format as desmos itself (except we use a `|` to denote the beginning of the restrictions):
|
||||
|
||||
````
|
||||
```desmos-graph
|
||||
y=\sin(x)|{y > 0}
|
||||
```
|
||||
````
|
||||
|
||||
### Style
|
||||
|
||||
We support six different types of (case-insensitive) styles:
|
||||
Line: `SOLID` `DASHED` `DOTTED`
|
||||
Point: `POINT` `OPEN` `CROSS`
|
||||
|
||||
These are placed after the graph restrictions, following another `|`:
|
||||
|
||||
````
|
||||
```desmos-graph
|
||||
y=\sin(x)|{y > 0}|DASHED
|
||||
```
|
||||
````
|
||||
|
||||
If you do not wish to apply any restrictions, the center field can be left blank:
|
||||
|
||||
````
|
||||
```desmos-graph
|
||||
y=\sin(x)||DASHED
|
||||
(1,2)||OPEN
|
||||
```
|
||||
````
|
||||
|
||||
## Important
|
||||
|
||||
Note that to be able to render these graphs into a PDF the following conditions must be fulfilled
|
||||
|
||||
1. Memory caching **must** be enabled
|
||||
2. Obsidian **must** have been restarted since you initially created the graph
|
||||
3. You **must** have viewed the rendered graph in the preview since the restart
|
||||
|
||||
After these are complete, a standard PDF export should work fine.
|
||||
In the future these steps will be removed and you will be able to directly export them.
|
406
main.js
generated
Normal file
406
main.js
generated
Normal file
File diff suppressed because one or more lines are too long
9
manifest.json
Normal file
9
manifest.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"id": "obsidian-desmos",
|
||||
"name": "Desmos",
|
||||
"version": "0.0.1",
|
||||
"minAppVersion": "0.9.12",
|
||||
"description": "Embed Desmos graphs into your notes",
|
||||
"author": "Nigecat",
|
||||
"isDesktopOnly": true
|
||||
}
|
343
package-lock.json
generated
Normal file
343
package-lock.json
generated
Normal file
@ -0,0 +1,343 @@
|
||||
{
|
||||
"name": "obsidian-desmos",
|
||||
"version": "0.0.1",
|
||||
"lockfileVersion": 1,
|
||||
"requires": true,
|
||||
"dependencies": {
|
||||
"@rollup/plugin-commonjs": {
|
||||
"version": "19.0.0",
|
||||
"resolved": "https://registry.npmjs.org/@rollup/plugin-commonjs/-/plugin-commonjs-19.0.0.tgz",
|
||||
"integrity": "sha512-adTpD6ATGbehdaQoZQ6ipDFhdjqsTgpOAhFiPwl+dzre4pPshsecptDPyEFb61JMJ1+mGljktaC4jI8ARMSNyw==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@rollup/pluginutils": "^3.1.0",
|
||||
"commondir": "^1.0.1",
|
||||
"estree-walker": "^2.0.1",
|
||||
"glob": "^7.1.6",
|
||||
"is-reference": "^1.2.1",
|
||||
"magic-string": "^0.25.7",
|
||||
"resolve": "^1.17.0"
|
||||
}
|
||||
},
|
||||
"@rollup/plugin-node-resolve": {
|
||||
"version": "13.0.0",
|
||||
"resolved": "https://registry.npmjs.org/@rollup/plugin-node-resolve/-/plugin-node-resolve-13.0.0.tgz",
|
||||
"integrity": "sha512-41X411HJ3oikIDivT5OKe9EZ6ud6DXudtfNrGbC4nniaxx2esiWjkLOzgnZsWq1IM8YIeL2rzRGLZLBjlhnZtQ==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@rollup/pluginutils": "^3.1.0",
|
||||
"@types/resolve": "1.17.1",
|
||||
"builtin-modules": "^3.1.0",
|
||||
"deepmerge": "^4.2.2",
|
||||
"is-module": "^1.0.0",
|
||||
"resolve": "^1.19.0"
|
||||
}
|
||||
},
|
||||
"@rollup/plugin-typescript": {
|
||||
"version": "8.2.1",
|
||||
"resolved": "https://registry.npmjs.org/@rollup/plugin-typescript/-/plugin-typescript-8.2.1.tgz",
|
||||
"integrity": "sha512-Qd2E1pleDR4bwyFxqbjt4eJf+wB0UKVMLc7/BAFDGVdAXQMCsD4DUv5/7/ww47BZCYxWtJqe1Lo0KVNswBJlRw==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@rollup/pluginutils": "^3.1.0",
|
||||
"resolve": "^1.17.0"
|
||||
}
|
||||
},
|
||||
"@rollup/pluginutils": {
|
||||
"version": "3.1.0",
|
||||
"resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-3.1.0.tgz",
|
||||
"integrity": "sha512-GksZ6pr6TpIjHm8h9lSQ8pi8BE9VeubNT0OMJ3B5uZJ8pz73NPiqOtCog/x2/QzM1ENChPKxMDhiQuRHsqc+lg==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@types/estree": "0.0.39",
|
||||
"estree-walker": "^1.0.1",
|
||||
"picomatch": "^2.2.2"
|
||||
},
|
||||
"dependencies": {
|
||||
"estree-walker": {
|
||||
"version": "1.0.1",
|
||||
"resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-1.0.1.tgz",
|
||||
"integrity": "sha512-1fMXF3YP4pZZVozF8j/ZLfvnR8NSIljt56UhbZ5PeeDmmGHpgpdwQt7ITlGvYaQukCvuBRMLEiKiYC+oeIg4cg==",
|
||||
"dev": true
|
||||
}
|
||||
}
|
||||
},
|
||||
"@types/codemirror": {
|
||||
"version": "0.0.108",
|
||||
"resolved": "https://registry.npmjs.org/@types/codemirror/-/codemirror-0.0.108.tgz",
|
||||
"integrity": "sha512-3FGFcus0P7C2UOGCNUVENqObEb4SFk+S8Dnxq7K6aIsLVs/vDtlangl3PEO0ykaKXyK56swVF6Nho7VsA44uhw==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@types/tern": "*"
|
||||
}
|
||||
},
|
||||
"@types/debounce": {
|
||||
"version": "1.2.0",
|
||||
"resolved": "https://registry.npmjs.org/@types/debounce/-/debounce-1.2.0.tgz",
|
||||
"integrity": "sha512-bWG5wapaWgbss9E238T0R6bfo5Fh3OkeoSt245CM7JJwVwpw6MEBCbIxLq5z8KzsE3uJhzcIuQkyiZmzV3M/Dw==",
|
||||
"dev": true
|
||||
},
|
||||
"@types/estree": {
|
||||
"version": "0.0.39",
|
||||
"resolved": "https://registry.npmjs.org/@types/estree/-/estree-0.0.39.tgz",
|
||||
"integrity": "sha512-EYNwp3bU+98cpU4lAWYYL7Zz+2gryWH1qbdDTidVd6hkiR6weksdbMadyXKXNPEkQFhXM+hVO9ZygomHXp+AIw==",
|
||||
"dev": true
|
||||
},
|
||||
"@types/node": {
|
||||
"version": "15.12.2",
|
||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-15.12.2.tgz",
|
||||
"integrity": "sha512-zjQ69G564OCIWIOHSXyQEEDpdpGl+G348RAKY0XXy9Z5kU9Vzv1GMNnkar/ZJ8dzXB3COzD9Mo9NtRZ4xfgUww==",
|
||||
"dev": true
|
||||
},
|
||||
"@types/resolve": {
|
||||
"version": "1.17.1",
|
||||
"resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-1.17.1.tgz",
|
||||
"integrity": "sha512-yy7HuzQhj0dhGpD8RLXSZWEkLsV9ibvxvi6EiJ3bkqLAO1RGo0WbkWQiwpRlSFymTJRz0d3k5LM3kkx8ArDbLw==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@types/node": "*"
|
||||
}
|
||||
},
|
||||
"@types/tern": {
|
||||
"version": "0.23.3",
|
||||
"resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.3.tgz",
|
||||
"integrity": "sha512-imDtS4TAoTcXk0g7u4kkWqedB3E4qpjXzCpD2LU5M5NAXHzCDsypyvXSaG7mM8DKYkCRa7tFp4tS/lp/Wo7Q3w==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@types/estree": "*"
|
||||
}
|
||||
},
|
||||
"balanced-match": {
|
||||
"version": "1.0.2",
|
||||
"resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
|
||||
"integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
|
||||
"dev": true
|
||||
},
|
||||
"brace-expansion": {
|
||||
"version": "1.1.11",
|
||||
"resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
|
||||
"integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"balanced-match": "^1.0.0",
|
||||
"concat-map": "0.0.1"
|
||||
}
|
||||
},
|
||||
"builtin-modules": {
|
||||
"version": "3.2.0",
|
||||
"resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-3.2.0.tgz",
|
||||
"integrity": "sha512-lGzLKcioL90C7wMczpkY0n/oART3MbBa8R9OFGE1rJxoVI86u4WAGfEk8Wjv10eKSyTHVGkSo3bvBylCEtk7LA==",
|
||||
"dev": true
|
||||
},
|
||||
"commondir": {
|
||||
"version": "1.0.1",
|
||||
"resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz",
|
||||
"integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=",
|
||||
"dev": true
|
||||
},
|
||||
"concat-map": {
|
||||
"version": "0.0.1",
|
||||
"resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
|
||||
"integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
|
||||
"dev": true
|
||||
},
|
||||
"deepmerge": {
|
||||
"version": "4.2.2",
|
||||
"resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz",
|
||||
"integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==",
|
||||
"dev": true
|
||||
},
|
||||
"estree-walker": {
|
||||
"version": "2.0.2",
|
||||
"resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz",
|
||||
"integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==",
|
||||
"dev": true
|
||||
},
|
||||
"fs.realpath": {
|
||||
"version": "1.0.0",
|
||||
"resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
|
||||
"integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=",
|
||||
"dev": true
|
||||
},
|
||||
"fsevents": {
|
||||
"version": "2.3.2",
|
||||
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
|
||||
"integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
|
||||
"dev": true,
|
||||
"optional": true
|
||||
},
|
||||
"function-bind": {
|
||||
"version": "1.1.1",
|
||||
"resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
|
||||
"integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==",
|
||||
"dev": true
|
||||
},
|
||||
"glob": {
|
||||
"version": "7.1.7",
|
||||
"resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz",
|
||||
"integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"fs.realpath": "^1.0.0",
|
||||
"inflight": "^1.0.4",
|
||||
"inherits": "2",
|
||||
"minimatch": "^3.0.4",
|
||||
"once": "^1.3.0",
|
||||
"path-is-absolute": "^1.0.0"
|
||||
}
|
||||
},
|
||||
"has": {
|
||||
"version": "1.0.3",
|
||||
"resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
|
||||
"integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"function-bind": "^1.1.1"
|
||||
}
|
||||
},
|
||||
"inflight": {
|
||||
"version": "1.0.6",
|
||||
"resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
|
||||
"integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"once": "^1.3.0",
|
||||
"wrappy": "1"
|
||||
}
|
||||
},
|
||||
"inherits": {
|
||||
"version": "2.0.4",
|
||||
"resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
|
||||
"integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
|
||||
"dev": true
|
||||
},
|
||||
"is-core-module": {
|
||||
"version": "2.4.0",
|
||||
"resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.4.0.tgz",
|
||||
"integrity": "sha512-6A2fkfq1rfeQZjxrZJGerpLCTHRNEBiSgnu0+obeJpEPZRUooHgsizvzv0ZjJwOz3iWIHdJtVWJ/tmPr3D21/A==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"has": "^1.0.3"
|
||||
}
|
||||
},
|
||||
"is-module": {
|
||||
"version": "1.0.0",
|
||||
"resolved": "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz",
|
||||
"integrity": "sha1-Mlj7afeMFNW4FdZkM2tM/7ZEFZE=",
|
||||
"dev": true
|
||||
},
|
||||
"is-reference": {
|
||||
"version": "1.2.1",
|
||||
"resolved": "https://registry.npmjs.org/is-reference/-/is-reference-1.2.1.tgz",
|
||||
"integrity": "sha512-U82MsXXiFIrjCK4otLT+o2NA2Cd2g5MLoOVXUZjIOhLurrRxpEXzI8O0KZHr3IjLvlAH1kTPYSuqer5T9ZVBKQ==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@types/estree": "*"
|
||||
}
|
||||
},
|
||||
"magic-string": {
|
||||
"version": "0.25.7",
|
||||
"resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.7.tgz",
|
||||
"integrity": "sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"sourcemap-codec": "^1.4.4"
|
||||
}
|
||||
},
|
||||
"minimatch": {
|
||||
"version": "3.0.4",
|
||||
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
|
||||
"integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"brace-expansion": "^1.1.7"
|
||||
}
|
||||
},
|
||||
"moment": {
|
||||
"version": "2.29.1",
|
||||
"resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz",
|
||||
"integrity": "sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==",
|
||||
"dev": true
|
||||
},
|
||||
"obsidian": {
|
||||
"version": "0.12.5",
|
||||
"resolved": "https://registry.npmjs.org/obsidian/-/obsidian-0.12.5.tgz",
|
||||
"integrity": "sha512-dQkcHWVgPzVlCxvkeu02Co8P1t4Ii95dC2NsFJV5bKK04J4//YxEdTTI/TFKr77CtYcH78LlUmOFeY5rHwyU/Q==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@types/codemirror": "0.0.108",
|
||||
"moment": "2.29.1"
|
||||
}
|
||||
},
|
||||
"once": {
|
||||
"version": "1.4.0",
|
||||
"resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
|
||||
"integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"wrappy": "1"
|
||||
}
|
||||
},
|
||||
"path-is-absolute": {
|
||||
"version": "1.0.1",
|
||||
"resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
|
||||
"integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=",
|
||||
"dev": true
|
||||
},
|
||||
"path-parse": {
|
||||
"version": "1.0.7",
|
||||
"resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
|
||||
"integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
|
||||
"dev": true
|
||||
},
|
||||
"picomatch": {
|
||||
"version": "2.3.0",
|
||||
"resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz",
|
||||
"integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==",
|
||||
"dev": true
|
||||
},
|
||||
"resolve": {
|
||||
"version": "1.20.0",
|
||||
"resolved": "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz",
|
||||
"integrity": "sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"is-core-module": "^2.2.0",
|
||||
"path-parse": "^1.0.6"
|
||||
}
|
||||
},
|
||||
"rollup": {
|
||||
"version": "2.51.2",
|
||||
"resolved": "https://registry.npmjs.org/rollup/-/rollup-2.51.2.tgz",
|
||||
"integrity": "sha512-ReV2eGEadA7hmXSzjxdDKs10neqH2QURf2RxJ6ayAlq93ugy6qIvXMmbc5cWMGCDh1h5T4thuWO1e2VNbMq8FA==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"fsevents": "~2.3.1"
|
||||
}
|
||||
},
|
||||
"sourcemap-codec": {
|
||||
"version": "1.4.8",
|
||||
"resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz",
|
||||
"integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==",
|
||||
"dev": true
|
||||
},
|
||||
"tslib": {
|
||||
"version": "2.3.0",
|
||||
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.3.0.tgz",
|
||||
"integrity": "sha512-N82ooyxVNm6h1riLCoyS9e3fuJ3AMG2zIZs2Gd1ATcSFjSA23Q0fzjjZeh0jbJvWVDZ0cJT8yaNNaaXHzueNjg==",
|
||||
"dev": true
|
||||
},
|
||||
"typescript": {
|
||||
"version": "4.3.2",
|
||||
"resolved": "https://registry.npmjs.org/typescript/-/typescript-4.3.2.tgz",
|
||||
"integrity": "sha512-zZ4hShnmnoVnAHpVHWpTcxdv7dWP60S2FsydQLV8V5PbS3FifjWFFRiHSWpDJahly88PRyV5teTSLoq4eG7mKw==",
|
||||
"dev": true
|
||||
},
|
||||
"wrappy": {
|
||||
"version": "1.0.2",
|
||||
"resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
|
||||
"integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
|
||||
"dev": true
|
||||
}
|
||||
}
|
||||
}
|
22
package.json
Normal file
22
package.json
Normal file
@ -0,0 +1,22 @@
|
||||
{
|
||||
"name": "obsidian-desmos",
|
||||
"version": "0.0.1",
|
||||
"description": "Embed Desmos graphs into your notes",
|
||||
"main": "main.js",
|
||||
"author": "Nigecat",
|
||||
"scripts": {
|
||||
"build": "rollup --config rollup.config.js --environment BUILD:production",
|
||||
"build:dev": "rollup --config rollup.config.js -w"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@rollup/plugin-commonjs": "^19.0.0",
|
||||
"@rollup/plugin-node-resolve": "^13.0.0",
|
||||
"@rollup/plugin-typescript": "^8.2.1",
|
||||
"@types/debounce": "^1.2.0",
|
||||
"@types/node": "^15.12.2",
|
||||
"obsidian": "^0.12.5",
|
||||
"rollup": "^2.51.2",
|
||||
"tslib": "^2.3.0",
|
||||
"typescript": "^4.3.2"
|
||||
}
|
||||
}
|
18
rollup.config.js
Normal file
18
rollup.config.js
Normal file
@ -0,0 +1,18 @@
|
||||
import typescript from "@rollup/plugin-typescript";
|
||||
import { nodeResolve } from "@rollup/plugin-node-resolve";
|
||||
import commonjs from "@rollup/plugin-commonjs";
|
||||
|
||||
const isProd = process.env.BUILD === "production";
|
||||
|
||||
export default {
|
||||
input: "src/main.ts",
|
||||
output: {
|
||||
dir: ".",
|
||||
sourcemap: "inline",
|
||||
sourcemapExcludeSources: isProd,
|
||||
format: "cjs",
|
||||
exports: "default",
|
||||
},
|
||||
external: ["obsidian"],
|
||||
plugins: [typescript(), nodeResolve({ browser: true }), commonjs()],
|
||||
};
|
141
src/dsl.ts
Normal file
141
src/dsl.ts
Normal file
@ -0,0 +1,141 @@
|
||||
import { createHash } from "crypto";
|
||||
|
||||
export interface Fields {
|
||||
width: number;
|
||||
height: number;
|
||||
boundary_left: number;
|
||||
boundary_right: number;
|
||||
boundary_bottom: number;
|
||||
boundary_top: number;
|
||||
}
|
||||
|
||||
const FIELD_DEFAULTS: Fields = {
|
||||
width: 600,
|
||||
height: 400,
|
||||
boundary_left: -10,
|
||||
boundary_right: 10,
|
||||
boundary_bottom: -7,
|
||||
boundary_top: 7,
|
||||
};
|
||||
|
||||
export class Dsl {
|
||||
/** A (hex) SHA-256 hash of the fields of this object */
|
||||
public readonly hash: string;
|
||||
public readonly equations: string[];
|
||||
public readonly fields: Fields;
|
||||
|
||||
private constructor(equations: string[], fields: Partial<Fields>) {
|
||||
this.equations = equations;
|
||||
this.fields = { ...FIELD_DEFAULTS, ...fields };
|
||||
Dsl.assert_sanity(this.fields);
|
||||
this.hash = createHash("sha256")
|
||||
.update(JSON.stringify(this))
|
||||
.digest("hex");
|
||||
}
|
||||
|
||||
/** Check if the fields are sane, throws a `SyntaxError` if they aren't */
|
||||
private static assert_sanity(fields: Fields) {
|
||||
// Ensure boundaries are complete and in order
|
||||
if (fields.boundary_left >= fields.boundary_right) {
|
||||
throw new SyntaxError(
|
||||
`Right boundary (${fields.boundary_right}) must be greater than left boundary (${fields.boundary_left})`
|
||||
);
|
||||
}
|
||||
|
||||
if (fields.boundary_bottom >= fields.boundary_top) {
|
||||
throw new SyntaxError(`
|
||||
Top boundary (${fields.boundary_top}) must be greater than bottom boundary (${fields.boundary_bottom})
|
||||
`);
|
||||
}
|
||||
}
|
||||
|
||||
public static parse(source: string): Dsl {
|
||||
const split = source.split("---");
|
||||
|
||||
let equations: string[];
|
||||
let fields: Partial<Fields>;
|
||||
switch (split.length) {
|
||||
case 0: {
|
||||
equations = [];
|
||||
break;
|
||||
}
|
||||
|
||||
case 1: {
|
||||
equations = split[0].split("\n").filter(Boolean);
|
||||
break;
|
||||
}
|
||||
|
||||
case 2: {
|
||||
// If there are two segments then we know the first one must contain the settings
|
||||
fields = split[0]
|
||||
// Allow either a newline or semicolon as a delimiter
|
||||
.split(/[;\n]+/)
|
||||
.map((setting) => setting.trim())
|
||||
// Remove any empty elements
|
||||
.filter(Boolean)
|
||||
// Split each field on the first equals sign to create the key=value pair
|
||||
.map((setting) => {
|
||||
const [key, ...value] = setting.split("=");
|
||||
return [key, value.join("=")];
|
||||
})
|
||||
.reduce((settings, [key, value]) => {
|
||||
if (FIELD_DEFAULTS.hasOwnProperty(key)) {
|
||||
if (!value) {
|
||||
throw new SyntaxError(
|
||||
`Field '${key}' must have a value`
|
||||
);
|
||||
}
|
||||
|
||||
// We can use the defaults to determine the type of each field
|
||||
const field_v = (FIELD_DEFAULTS as any)[key];
|
||||
const field_t = typeof field_v;
|
||||
|
||||
switch (field_t) {
|
||||
case "number": {
|
||||
const s = parseInt(value);
|
||||
if (Number.isNaN(s)) {
|
||||
throw new SyntaxError(
|
||||
`Field '${key}' must have an integer value`
|
||||
);
|
||||
}
|
||||
(settings as any)[key] = s;
|
||||
break;
|
||||
}
|
||||
|
||||
case "string": {
|
||||
(settings as any)[key] = value;
|
||||
break;
|
||||
}
|
||||
|
||||
case "object": {
|
||||
const val = JSON.parse(value);
|
||||
if (
|
||||
val.constructor === field_v.constructor
|
||||
) {
|
||||
(settings as any)[key] = val;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
throw new SyntaxError(`Unrecognised field: ${key}`);
|
||||
}
|
||||
|
||||
return settings;
|
||||
}, {} as Partial<Fields>);
|
||||
|
||||
equations = split[1].split("\n").filter(Boolean);
|
||||
break;
|
||||
}
|
||||
|
||||
default: {
|
||||
fields = {};
|
||||
}
|
||||
}
|
||||
if (!equations) {
|
||||
throw new SyntaxError("Too many segments");
|
||||
}
|
||||
|
||||
return new Dsl(equations, fields);
|
||||
}
|
||||
}
|
6
src/error.ts
Normal file
6
src/error.ts
Normal file
@ -0,0 +1,6 @@
|
||||
export function renderError(err: string, el: HTMLElement) {
|
||||
el.innerHTML = `
|
||||
<div style="padding: 20px; background-color: #f44336; color: white;">
|
||||
<strong>Desmos Graph Error:</strong> ${err}
|
||||
</div>`;
|
||||
}
|
64
src/main.ts
Normal file
64
src/main.ts
Normal file
@ -0,0 +1,64 @@
|
||||
import { Dsl } from "./dsl";
|
||||
import { Renderer } from "./renderer";
|
||||
import { renderError } from "./error";
|
||||
import { debounce, Plugin } from "obsidian";
|
||||
import { Settings, SettingsTab, DEFAULT_SETTINGS } from "./settings";
|
||||
|
||||
export default class Desmos extends Plugin {
|
||||
settings: Settings;
|
||||
/** Helper for in-memory graph caching */
|
||||
graph_cache: Record<string, string>;
|
||||
|
||||
async onload() {
|
||||
this.graph_cache = {};
|
||||
await this.loadSettings();
|
||||
this.addSettingTab(new SettingsTab(this.app, this));
|
||||
|
||||
// Keep track of the total number of graphs in each file
|
||||
// This allows us to skip the debounce on recently opened files to make it feel snappier to use
|
||||
let total = 0;
|
||||
this.app.workspace.on("file-open", async (file) => {
|
||||
const contents = await this.app.vault.cachedRead(file);
|
||||
|
||||
// Attempt to figure out the number of graphs there are in this file
|
||||
// In this case it is fine if we overestimate because we only need a general idea since this just makes it skip the debounce
|
||||
total = (contents.match(/```desmos-graph/g) || []).length;
|
||||
});
|
||||
|
||||
const render = (source: string, el: HTMLElement) => {
|
||||
try {
|
||||
Renderer.render(Dsl.parse(source), this.settings, el, this);
|
||||
} catch (err) {
|
||||
renderError(err.message, el);
|
||||
}
|
||||
};
|
||||
const debounce_render = debounce(
|
||||
(source: string, el: HTMLElement) => render(source, el),
|
||||
this.settings.debounce
|
||||
);
|
||||
this.registerMarkdownCodeBlockProcessor(
|
||||
"desmos-graph",
|
||||
(source, el) => {
|
||||
if (total > 0) {
|
||||
total--;
|
||||
// Skip the debounce on initial render
|
||||
render(source, el);
|
||||
} else {
|
||||
debounce_render(source, el);
|
||||
}
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
async loadSettings() {
|
||||
this.settings = Object.assign(
|
||||
{},
|
||||
DEFAULT_SETTINGS,
|
||||
await this.loadData()
|
||||
);
|
||||
}
|
||||
|
||||
async saveSettings() {
|
||||
await this.saveData(this.settings);
|
||||
}
|
||||
}
|
205
src/renderer.ts
Normal file
205
src/renderer.ts
Normal file
@ -0,0 +1,205 @@
|
||||
import path from "path";
|
||||
import Desmos from "./main";
|
||||
import { Dsl } from "./dsl";
|
||||
import { tmpdir } from "os";
|
||||
import { Notice } from "obsidian";
|
||||
import { Settings } from "./settings";
|
||||
import { renderError } from "./error";
|
||||
import { existsSync, promises as fs } from "fs";
|
||||
|
||||
export class Renderer {
|
||||
static render(
|
||||
args: Dsl,
|
||||
settings: Settings,
|
||||
el: HTMLElement,
|
||||
plugin: Desmos
|
||||
) {
|
||||
const { fields, equations, hash } = args;
|
||||
|
||||
// Calculate cache info for filesystem caching
|
||||
const vault_root = (plugin.app.vault.adapter as any).basePath;
|
||||
const cache_dir = settings.cache_directory
|
||||
? path.isAbsolute(settings.cache_directory)
|
||||
? settings.cache_directory
|
||||
: path.join(vault_root, settings.cache_directory)
|
||||
: tmpdir();
|
||||
const cache_target = path.join(cache_dir, `desmos-graph-${hash}.png`);
|
||||
|
||||
// If this graph is in the cache then fetch it
|
||||
if (settings.cache) {
|
||||
if (
|
||||
settings.cache_location == "memory" &&
|
||||
hash in plugin.graph_cache
|
||||
) {
|
||||
const data = plugin.graph_cache[hash];
|
||||
const img = document.createElement("img");
|
||||
img.src = data;
|
||||
el.appendChild(img);
|
||||
return;
|
||||
} else if (
|
||||
settings.cache_location == "filesystem" &&
|
||||
existsSync(cache_target)
|
||||
) {
|
||||
fs.readFile(cache_target).then((data) => {
|
||||
const b64 =
|
||||
"data:image/png;base64," +
|
||||
Buffer.from(data).toString("base64");
|
||||
const img = document.createElement("img");
|
||||
img.src = b64;
|
||||
el.appendChild(img);
|
||||
});
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
const expressions = equations.map(
|
||||
(equation) =>
|
||||
`calculator.setExpression({
|
||||
latex: "${equation.split("|")[0].replace("\\", "\\\\")}${(
|
||||
equation.split("|")[1] ?? ""
|
||||
)
|
||||
.replace("{", "\\\\{")
|
||||
.replace("}", "\\\\}")
|
||||
.replace("<=", "\\\\leq ")
|
||||
.replace(">=", "\\\\geq ")
|
||||
.replace("<", "\\\\le ")
|
||||
.replace(">", "\\\\ge ")}",
|
||||
|
||||
${(() => {
|
||||
const mode = equation.split("|")[2];
|
||||
|
||||
if (mode) {
|
||||
if (
|
||||
["solid", "dashed", "dotted"].contains(
|
||||
mode.toLowerCase()
|
||||
)
|
||||
) {
|
||||
return `lineStyle: Desmos.Styles.${mode.toUpperCase()}`;
|
||||
} else if (
|
||||
["point", "open", "cross"].contains(
|
||||
mode.toLowerCase()
|
||||
)
|
||||
) {
|
||||
return `pointStyle: Desmos.Styles.${mode.toUpperCase()}`;
|
||||
}
|
||||
}
|
||||
|
||||
return "";
|
||||
})()}
|
||||
});`
|
||||
);
|
||||
|
||||
// Because of the electron sandboxing we have to do this inside an iframe,
|
||||
// otherwise we can't include the desmos API (although it would be nice if they had a REST API of some sort)
|
||||
const html_src_head = `<script src="https://www.desmos.com/api/v1.6/calculator.js?apiKey=dcb31709b452b1cf9dc26972add0fda6"></script>`;
|
||||
const html_src_body = `
|
||||
<div id="calculator" style="width: ${fields.width}px; height: ${
|
||||
fields.height
|
||||
}px;"></div>
|
||||
<script>
|
||||
const options = {
|
||||
settingsMenu: false,
|
||||
expressions: false,
|
||||
lockViewPort: true,
|
||||
zoomButtons: false,
|
||||
trace: false,
|
||||
};
|
||||
|
||||
const calculator = Desmos.GraphingCalculator(document.getElementById("calculator"), options);
|
||||
calculator.setMathBounds({
|
||||
left: ${fields.boundary_left},
|
||||
right: ${fields.boundary_right},
|
||||
top: ${fields.boundary_top},
|
||||
bottom: ${fields.boundary_bottom},
|
||||
});
|
||||
|
||||
${expressions.join("")}
|
||||
|
||||
calculator.observe("expressionAnalysis", () => {
|
||||
for (const id in calculator.expressionAnalysis) {
|
||||
const analysis = calculator.expressionAnalysis[id];
|
||||
if (analysis.isError) {
|
||||
parent.postMessage({ t: "desmos-graph", d: "error", data: analysis.errorMessage, hash: "${hash}" });
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
calculator.asyncScreenshot({ showLabels: true, format: "png" }, (data) => {
|
||||
document.body.innerHTML = "";
|
||||
parent.postMessage({ t: "desmos-graph", d: "render", data, hash: "${hash}" }, "app://obsidian.md");
|
||||
});
|
||||
</script>
|
||||
`;
|
||||
const html_src = `<html><head>${html_src_head}</head><body>${html_src_body}</body>`;
|
||||
|
||||
const iframe = document.createElement("iframe");
|
||||
iframe.width = fields.width.toString();
|
||||
iframe.height = fields.height.toString();
|
||||
iframe.style.border = "none";
|
||||
iframe.scrolling = "no"; // fixme use a non-depreciated function
|
||||
iframe.srcdoc = html_src;
|
||||
// iframe.style.display = "none"; //fixme hiding the iframe breaks the positioning
|
||||
|
||||
el.appendChild(iframe);
|
||||
|
||||
const handler = (
|
||||
message: MessageEvent<{
|
||||
t: string;
|
||||
d: string;
|
||||
data: string;
|
||||
hash: string;
|
||||
}>
|
||||
) => {
|
||||
if (
|
||||
message.origin === "app://obsidian.md" &&
|
||||
message.data.t === "desmos-graph" &&
|
||||
message.data.hash === hash
|
||||
) {
|
||||
el.empty();
|
||||
|
||||
if (message.data.d === "error") {
|
||||
renderError(message.data.data, el);
|
||||
}
|
||||
|
||||
if (message.data.d === "render") {
|
||||
const { data } = message.data;
|
||||
window.removeEventListener("message", handler);
|
||||
|
||||
const img = document.createElement("img");
|
||||
img.src = data;
|
||||
el.appendChild(img);
|
||||
|
||||
if (settings.cache) {
|
||||
if (settings.cache_location == "memory") {
|
||||
plugin.graph_cache[hash] = data;
|
||||
} else if (settings.cache_location == "filesystem") {
|
||||
if (existsSync(cache_dir)) {
|
||||
fs.writeFile(
|
||||
cache_target,
|
||||
data.replace(
|
||||
/^data:image\/png;base64,/,
|
||||
""
|
||||
),
|
||||
"base64"
|
||||
).catch(
|
||||
(err) =>
|
||||
new Notice(
|
||||
`desmos-graph: unexpected error when trying to cache graph: ${err}`,
|
||||
10000
|
||||
)
|
||||
);
|
||||
} else {
|
||||
new Notice(
|
||||
`desmos-graph: cache directory not found: '${cache_dir}'`,
|
||||
10000
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
window.addEventListener("message", handler);
|
||||
}
|
||||
}
|
105
src/settings.ts
Normal file
105
src/settings.ts
Normal file
@ -0,0 +1,105 @@
|
||||
import { tmpdir } from "os";
|
||||
import Desmos from "./main";
|
||||
import { PluginSettingTab, App, Setting } from "obsidian";
|
||||
|
||||
export interface Settings {
|
||||
debounce: number;
|
||||
cache: boolean;
|
||||
cache_location: "memory" | "filesystem";
|
||||
cache_directory: string | null;
|
||||
}
|
||||
|
||||
export const DEFAULT_SETTINGS: Settings = {
|
||||
debounce: 500,
|
||||
cache: true,
|
||||
cache_location: "memory",
|
||||
cache_directory: null,
|
||||
};
|
||||
|
||||
export class SettingsTab extends PluginSettingTab {
|
||||
plugin: Desmos;
|
||||
|
||||
constructor(app: App, plugin: Desmos) {
|
||||
super(app, plugin);
|
||||
this.plugin = plugin;
|
||||
}
|
||||
|
||||
display() {
|
||||
let { containerEl } = this;
|
||||
|
||||
containerEl.empty();
|
||||
|
||||
new Setting(containerEl)
|
||||
.setName("Debounce Time (ms)")
|
||||
.setDesc(
|
||||
"How long to wait after a keypress to render the graph (requires restart to take effect)"
|
||||
)
|
||||
.addText((text) =>
|
||||
text
|
||||
.setValue(this.plugin.settings.debounce.toString())
|
||||
.onChange(async (value) => {
|
||||
const val = parseInt(value);
|
||||
this.plugin.settings.debounce =
|
||||
val === NaN ? DEFAULT_SETTINGS.debounce : val;
|
||||
await this.plugin.saveSettings();
|
||||
})
|
||||
);
|
||||
|
||||
new Setting(containerEl)
|
||||
.setName("Cache")
|
||||
.setDesc("Whether to cache the rendered graphs")
|
||||
.addToggle((toggle) =>
|
||||
toggle
|
||||
.setValue(this.plugin.settings.cache)
|
||||
.onChange(async (value) => {
|
||||
this.plugin.settings.cache = value;
|
||||
await this.plugin.saveSettings();
|
||||
|
||||
// Reset the display so the new state can render
|
||||
this.display();
|
||||
})
|
||||
);
|
||||
|
||||
if (this.plugin.settings.cache) {
|
||||
new Setting(containerEl)
|
||||
.setName("Cache in memory (alternate: filesystem)")
|
||||
.setDesc(
|
||||
"Cache rendered graphs in memory or on the filesystem (note that memory caching is not persistent)."
|
||||
)
|
||||
.addToggle((toggle) =>
|
||||
toggle
|
||||
.setValue(
|
||||
this.plugin.settings.cache_location === "memory"
|
||||
? true
|
||||
: false
|
||||
)
|
||||
.onChange(async (value) => {
|
||||
this.plugin.settings.cache_location = value
|
||||
? "memory"
|
||||
: "filesystem";
|
||||
await this.plugin.saveSettings();
|
||||
|
||||
// Reset the display so the new state can render
|
||||
this.display();
|
||||
})
|
||||
);
|
||||
|
||||
if (this.plugin.settings.cache_location == "filesystem") {
|
||||
new Setting(containerEl)
|
||||
.setName("Cache Directory")
|
||||
.setDesc(
|
||||
"The directory to save cached graphs in (technical note: the graphs will be saved as `desmos-graph-<hash>.png` where the name is a SHA-256 hash of the graph source). The default directory is the system tempdir for your current operating system, and this value may be either a path relative to the root of your vault or an absolute path. Also note that a lot of junk will be saved to this folder, you have been warned."
|
||||
)
|
||||
.addText((text) =>
|
||||
text
|
||||
.setPlaceholder(tmpdir())
|
||||
.setValue(this.plugin.settings.cache_directory)
|
||||
.onChange(async (value) => {
|
||||
this.plugin.settings.cache_directory = value;
|
||||
await this.plugin.saveSettings();
|
||||
})
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
3
test/.obsidian/.gitignore
vendored
Normal file
3
test/.obsidian/.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
*
|
||||
!.gitignore
|
||||
!/plugins/**/*
|
1
test/.obsidian/plugins/obsidian-desmos/main.js
vendored
Symbolic link
1
test/.obsidian/plugins/obsidian-desmos/main.js
vendored
Symbolic link
@ -0,0 +1 @@
|
||||
../../../../main.js
|
1
test/.obsidian/plugins/obsidian-desmos/manifest.json
vendored
Symbolic link
1
test/.obsidian/plugins/obsidian-desmos/manifest.json
vendored
Symbolic link
@ -0,0 +1 @@
|
||||
../../../../manifest.json
|
1
test/.obsidian/plugins/obsidian-desmos/versions.json
vendored
Symbolic link
1
test/.obsidian/plugins/obsidian-desmos/versions.json
vendored
Symbolic link
@ -0,0 +1 @@
|
||||
../../../../versions.json
|
9
test/test.md
Normal file
9
test/test.md
Normal file
@ -0,0 +1,9 @@
|
||||
```desmos-graph
|
||||
boundary_left=-2; boundary_right=2;
|
||||
boundary_bottom=-1; boundary_top=3;
|
||||
---
|
||||
y=x^2|{x<0}|DASHED
|
||||
y=x||DOTTED
|
||||
(1,2)||OPEN
|
||||
(-1,2)||CROSS
|
||||
```
|
16
tsconfig.json
Normal file
16
tsconfig.json
Normal file
@ -0,0 +1,16 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"baseUrl": ".",
|
||||
"inlineSourceMap": true,
|
||||
"inlineSources": true,
|
||||
"module": "ESNext",
|
||||
"target": "es6",
|
||||
"allowJs": true,
|
||||
"noImplicitAny": true,
|
||||
"moduleResolution": "node",
|
||||
"allowSyntheticDefaultImports": true,
|
||||
"importHelpers": true,
|
||||
"lib": ["dom", "es5", "scripthost", "es2015"]
|
||||
},
|
||||
"include": ["**/*.ts"]
|
||||
}
|
3
versions.json
Normal file
3
versions.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"0.0.1": "0.9.12"
|
||||
}
|
Loading…
Reference in New Issue
Block a user