# Methoden en objecten die QOwnNotes biedt
# Het starten van een extern programma op de achtergrond
# Methodeaanroep en parameters
/**
* QML wrapper om een losstaand proces te starten
*
* @param executablePath het pad van het uitvoerbare
* @param parameters een lijst met parameterreeksen
* @param callbackIdentifier an identifier to be used in the onDetachedProcessCallback() function (optional)
* @param callbackParameter een extra parameter voor loops of iets dergelijks (optioneel)
* @param processData gegevens die naar het proces worden geschreven als de callback wordt gebruikt (optioneel)
* @param workingDirectory de werkdirectory om het proces in uit te voeren (optioneel, werkt alleen zonder callback)
* @return true op succes, anders vals
*/
bool startDetachedProcess(QString executablePath, QStringList parameters,
QString callbackIdentifier, QVariant callbackParameter,
QByteArray processData, QString workingDirectory);
2
3
4
5
6
7
8
9
10
11
12
13
14
# Voorbeeld
Eenvoudig voorbeeld:
script.startDetachedProcess("/path/to/my/program", ["my parameter"]);
Veel processen uitvoeren:
for (var i = 0; i < 100; i++) {
var dur = Math.floor(Math.random() * 10) + 1;
script.startDetachedProcess("sleep", [`${dur}s`], "my-callback", i);
}
function onDetachedProcessCallback(callbackIdentifier, resultSet, cmd, thread) {
if (callbackIdentifier == "my-callback") {
script.log(`#${thread[1]} i[${thread[0]}] t${cmd[1]}`);
}
}
2
3
4
5
6
7
8
9
10
Misschien wilt u het voorbeeld bekijken custom-actions.qml (opens new window), callback.qml (opens new window) of execute-command-after-note-update.qml (opens new window).
Misschien wil je ook eens kijken naar de onDetachedProcessCallback hook.
# Start een extern programma en wacht op de uitvoer
# Methodeaanroep en parameters
/**
* QML wrapper to start a synchronous process
*
* @param executablePath the path of the executable
* @param parameters a list of parameter strings
* @param data the data that will be written to the process (optional)
* @param workingDirectory the working directory to execute the process in (optional)
* @return the text that was returned by the process
QByteArray startSynchronousProcess(QString executablePath, QStringList parameters, QByteArray data, QString workingDirectory);
2
3
4
5
6
7
8
9
# Voorbeeld
var result = script.startSynchronousProcess("/path/to/my/program", ["my parameter"], "data", "/path/to/execute/in");
Misschien wilt u het voorbeeld bekijken encryption-keybase.qml (opens new window).
# Het pad ophalen van de huidige notitiemap
# Methodeaanroep en parameters
/**
* QML-wrapper om het huidige pad naar de notitiemap op te halen
*
* @return het pad van de huidige notitiemap
*/
QString currentNoteFolderPath ();
2
3
4
5
6
# Voorbeeld
var path = script.currentNoteFolderPath();
Misschien wilt u het voorbeeld bekijken absolute-media-links.qml (opens new window).
# De huidige notitie ophalen
# Methodeaanroep en parameters
/**
* QML-wrapper om de huidige notitie te krijgen
*
* @returns {NoteApi} het huidige notitieobject
*/
NoteApi currentNote();
2
3
4
5
6
# Voorbeeld
var note = script.currentNote();
Misschien wilt u het voorbeeld bekijken custom-actions.qml (opens new window).
# Inloggen op de log-widget
# Methodeaanroep en parameters
/**
* QML-wrapper om in te loggen op de log-widget
*
* @param tekst
*/
ongeldig logboek (QString-tekst);
2
3
4
5
6
# Voorbeeld
script.log("my text");
# Het downloaden van een url naar een string
# Methodeaanroep en parameters
/**
* QML-wrapper om een url te downloaden en als tekst terug te sturen
*
* @param url
* @return {QString} de inhoud van de gedownloade url
*/
QString downloadUrlToString (QUrl url);
2
3
4
5
6
7
# Voorbeeld
var html = script.downloadUrlToString("https://www.qownnotes.org");
Misschien wilt u het voorbeeld bekijken insert-headline-with-link-from-github-url.qml (opens new window).
# Een url downloaden naar de mediamap
# Methodeaanroep en parameters
/**
* QML-wrapper om een url naar de mediamap te downloaden en de media terug te sturen
* url of de markdown-afbeeldingstekst van de media ten opzichte van de huidige notitie
*
* @param {QString} url
* @param {bool} returnUrlAlleen als waar alleen de media-URL wordt geretourneerd (standaard false)
* @return {QString} de media-markdown of url
*/
QString downloadUrlToMedia (QUrl url, bool returnUrlOnly);
2
3
4
5
6
7
8
9
# Voorbeeld
var markdown = script.downloadUrlToMedia("http://latex.codecogs.com/gif.latex?\frac{1}{1+sin(x)}");
Misschien wilt u het voorbeeld bekijken paste-latex-image.qml (opens new window).
# Een mediabestand invoegen in de mediamap
# Methodeaanroep en parameters
/**
* QML-wrapper om een mediabestand in de mediamap in te voegen en terug te keren
* de media-url of de afgeprijsde afbeeldingstekst van de media ten opzichte van de huidige notitie
*
* @param {QString} mediaFilePath
* @param {bool} returnUrlAlleen als waar alleen de media-URL wordt geretourneerd (standaard false)
* @return {QString} de media-markdown of url
*/
QString ScriptingService :: insertMediaFile (QString mediaFilePath,
bool returnUrlOnly);
2
3
4
5
6
7
8
9
10
# Voorbeeld
var markdown = script.insertMediaFile("/path/to/your/image.png");
Misschien wilt u het voorbeeld bekijken scribble.qml (opens new window).
# Een bijlagebestand invoegen in de bijlagenmap
# Methodeaanroep en parameters
* QML-wrapper om een bijlagebestand in de map `attachments` in te voegen en
* de bijlage-URL of de afprijstekst van de bijlage retourneren
* ten opzichte van de huidige noot
*
* @param {QString} attachmentFilePath
* @param {QString} fileName to use in the markdown
* @param {bool} returnUrlOnly if true only the attachment url will be returned
* (default false)
* @return {QString} the attachment markdown or url
*/
QString ScriptingService::insertAttachmentFile(const QString &attachmentFilePath,
const QString &fileName,
bool returnUrlOnly);
2
3
4
5
6
7
8
9
10
11
12
13
# Voorbeeld
var markdown = script.insertAttachmentFile("/path/to/your/file.png");
# Het notitievoorbeeld opnieuw genereren
Vernieuwt het notitievoorbeeld.
# Methodeaanroep en parameters
/**
* Regenereert het notitievoorbeeld
*/
QString ScriptingService::regenerateNotePreview();
2
3
4
# Voorbeeld
script.regenerateNotePreview();
Misschien wilt u het voorbeeld bekijken scribble.qml (opens new window).
# Een aangepaste actie registreren
# Methodeaanroep en parameters
**
* Registreert een aangepaste actie
*
* @param identifier de identifier van de actie
* @param menu Tekst de tekst die in het menu wordt weergegeven
* @param button Sms de tekst die in de knop wordt weergegeven
* (er wordt geen knop weergegeven als deze leeg is)
* @param icon het pictogrambestandspad of de naam van een freedesktop-themapictogram
* u vindt hier een lijst met pictogrammen:
* https://specifications.freedesktop.org/icon-naming-spec/icon-naming-spec-latest.html
* @param useInNoteEditContextMenu indien true, gebruik dan de actie in de notitiebewerking
* contextmenu (standaard: false)
* @param hideButtonInToolbar indien waar zal de knop niet getoond worden in het
* aangepaste actiewerkbalk (standaard: false)
* @param useInNoteListContextMenu indien waar, gebruik dan de actie in de notitielijst
* contextmenu (standaard: false)
*/
void ScriptingService :: registerCustomAction (QString identifier,
QString menuText,
QString-knop Tekst,
QString-pictogram,
bool useInNoteEditContextMenu,
bool hideButtonInToolbar,
bool useInNoteListContextMenu);
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
TIP
U kunt ook lokale en algemene snelkoppelingen aan uw aangepaste acties toewijzen in de Sneltoets-instellingen.
WARNING
Houd er rekening mee dat freedesktop-themapictogrammen (opens new window) meestal alleen beschikbaar zijn onder Linux. Dus als je echt een pictogram onder macOS of Windows wilt gebruiken, moet je er een bij je script voegen. Om het pad van uw script te krijgen om een juist pad voor uw pictogram in te stellen, kunt u de scriptDirPath-eigenschap gebruiken.
# Voorbeeld
import QtQml 2.0
import QOwnNotesTypes 1.0
Script {
/**
* Initializes the custom actions
*/
function init() {
// add a custom action without a button
script.registerCustomAction("mycustomaction1", "Menu text");
// add a custom action with a button
script.registerCustomAction("mycustomaction2", "Menu text", "Button text");
// add a custom action with a button and freedesktop theme icon
script.registerCustomAction("mycustomaction3", "Menu text", "Button text", "task-new");
// add a custom action with a button and an icon from a file
script.registerCustomAction("mycustomaction4", "Menu text", "Button text", "/usr/share/icons/breeze/actions/24/view-calendar-tasks.svg");
}
/**
* This function is invoked when a custom action is triggered
* in the menu or via button
*
* @param identifier string the identifier defined in registerCustomAction
*/
function customActionInvoked(identifier) {
switch (identifier) {
case "mycustomaction1":
script.log("Action 1");
break;
case "mycustomaction2":
script.log("Action 2");
break;
case "mycustomaction3":
script.log("Action 3");
break;
case "mycustomaction4":
script.log("Action 4");
break;
}
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
Zie voor meer voorbeelden custom-actions.qml (opens new window).
TIP
U kunt ook een aangepaste actie activeren nadat de toepassing is gestart met de parameter --action customAction_<identifier>
. Kijk voor meer informatie op Activeer menuacties na opstarten.
# Een label registreren
# Methodeaanroep en parameters
/**
* Registreert een label om naar te schrijven
*
* @param identifier the identifier of the label
* @param text the text shown in the label (optional)
*/
void ScriptingService::registerLabel(QString identifier, QString text);
2
3
4
5
6
7
# Voorbeeld
script.registerLabel("html-label", "<strong>Strong</strong> HTML text<br />with three lines<br />and a <a href='https://www.qownnotes.org'>link to a website</a>.");
script.registerLabel("long-label", "nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst, nog een zeer lange tekst die zal omlopen");
script.registerLabel("counter-label");
2
3
4
5
De labels zijn zichtbaar in het *Scripting panel *, dat u moet inschakelen in het menu *Window / Panels *.
U kunt zowel platte tekst als html in de labels gebruiken. De tekst kan worden geselecteerd en er kan op links worden geklikt.
Misschien wilt u dan het voorbeeldscript bekijken script-label-demo.qml (opens new window).
# De tekst van een geregistreerd label instellen
# Methodeaanroep en parameters
/**
* Stelt de tekst van een geregistreerd label in
*
* @param identifier the identifier of the label
* @param text the text shown in the label
*/
void ScriptingService::setLabelText(QString identifier, QString text);
2
3
4
5
6
7
# Voorbeeld
script.setLabelText("counter-label", "counter text");
U kunt zowel platte tekst als html in de labels gebruiken. De tekst kan worden geselecteerd en er kan op links worden geklikt.
Misschien wilt u dan het voorbeeldscript bekijken script-label-demo.qml (opens new window).
TIP
Het scriptlabel ondersteunt ook externe links, evenals interne links naar notities, taken, kaarten en meer. U kunt een idee krijgen van wat wordt ondersteund door een kijkje te nemen in URL-behandeling (opens new window).
# Een nieuwe notitie maken
# Methodeaanroep en parameters
/**
* Maakt een nieuwe notitie
*
* @param sms de notitietekst
*/
void ScriptingService :: createNote (QString-tekst);
2
3
4
5
6
# Voorbeeld
script.createNote("My note headline\n===\n\nMy text");
Misschien wilt u het voorbeeld bekijken custom-actions.qml (opens new window).
TIP
Als u hebt uitgeschakeld dat de kop van de notitie de bestandsnaam van de notitie bepaalt, dan bent moet uw notitiebestand daarna zelf hernoemen, zoals:
var note = script.currentNote();
note.renameNoteFile('your-filename');
2
# Toegang tot het klembord
# Methodeaanroep en parameters
/**
* Retourneert de inhoud van het klembord als tekst of html
*
* @param asHtml geeft de inhoud van het klembord terug als html in plaats van tekst
*/
QString ScriptingService :: clipboard (bool asHtml);
2
3
4
5
6
# Voorbeeld
var clipboardText = script.clipboard();
var clipboardHtml = script.clipboard(true);
2
Misschien wilt u het voorbeeld bekijken custom-actions.qml (opens new window).
# Schrijf tekst naar de tekst van de notitie
# Methodeaanroep en parameters
/**
* Schrijft tekst naar de huidige cursorpositie in de tekstbewerking van de notitie
*
* @param tekst
*/
void ScriptingService :: noteTextEditWrite (QString-tekst);
2
3
4
5
6
# Voorbeeld
// schrijf tekst naar de notitie tekst bewerken
script.noteTextEditWrite ("Mijn aangepaste tekst");
2
Misschien wilt u kijken naar de aangepaste actie transformTextRot13
in voorbeeld custom-actions.qml (opens new window).
U kunt dit samen met noteTextEditSelectAll
gebruiken om de hele tekst van de huidige notitie te overschrijven.
# Lees de geselecteerde tekst in de tekstbewerking van de notitie
# Methodeaanroep en parameters
/**
* Leest de geselecteerde tekst in de tekstbewerking van de notitie
*
* @return
*/
QString ScriptingService :: noteTextEditSelectedText ();
2
3
4
5
6
# Voorbeeld
// lees de geselecteerde tekst uit de notitietekst bewerk
var text = script.noteTextEditSelectedText ();
2
Misschien wilt u kijken naar de aangepaste actie transformTextRot13
in voorbeeld custom-actions.qml (opens new window).
# Selecteer alle tekst in de tekstbewerking van de notitie
# Methodeaanroep en parameters
/**
* Selecteert alle tekst in de tekstbewerking van de notitie
*/
leegte ScriptingService :: noteTextEditSelectAll ();
2
3
4
# Voorbeeld
script.noteTextEditSelectAll();
U kunt dit samen met noteTextEditWrite
gebruiken om de hele tekst van de huidige notitie te overschrijven.
# Selecteer de huidige regel in de tekstbewerking van de notitie
# Methodeaanroep en parameters
/**
* Selecteert de huidige regel in de notitietekst
*/
void ScriptingService::noteTextEditSelectCurrentLine();
2
3
4
# Voorbeeld
script.noteTextEditSelectCurrentLine();
# Selecteer de huidige woord in de tekstbewerking van de notitie
# Methodeaanroep en parameters
/**
* Selecteert het huidige woord in de notitietekstbewerking
*/
void ScriptingService::noteTextEditSelectCurrentWord();
2
3
4
# Voorbeeld
script.noteTextEditSelectCurrentWord();
# Stel de momenteel geselecteerde tekst in bij het bewerken van de notitietekst
# Methodeaanroep en parameters
/**
* Stelt de momenteel geselecteerde tekst in de notitietekstbewerking in
*
* @param start
* @param einde
*/
void ScriptingService::noteTextEditSetSelection(int start, int end);
2
3
4
5
6
7
# Voorbeeld
// breidt de huidige selectie uit met één teken
script.noteTextEditSetSelection (
script.noteTextEditSelectionStart () - 1,
script.noteTextEditSelectionEnd () + 1);
2
3
4
# Verkrijg de startpositie van de huidige selectie in de notitietekst
# Methodeaanroep en parameters
/**
* Geeft als resultaat de startpositie van de huidige selectie in de notitietekst
*/
int ScriptingService::noteTextEditSelectionStart();
2
3
4
# Voorbeeld
script.log(script.noteTextEditSelectionStart());
# Haal de eindpositie van de huidige selectie op in de tekstbewerking van de notitie
# Methodeaanroep en parameters
/**
* Returns the end position of the current selection in the note text edit
*/
int ScriptingService::noteTextEditSelectionEnd();
2
3
4
# Voorbeeld
script.log(script.noteTextEditSelectionEnd());
# Plaats de tekstcursor op een bepaalde positie in de notitie-bewerking
# Methodeaanroep en parameters
/**
* Zet de tekstcursor in de notitietekstbewerking op een bepaalde positie
* 0 zou het begin van de notitie zijn
* speciaal geval: -1 zou het einde van de notitie zijn
*
* @param position
*/
void ScriptingService::noteTextEditSetCursorPosition(int position);
2
3
4
5
6
7
8
# Voorbeeld
// spring naar het 11e teken in de notitie
script.noteTextEditSetCursorPosition(10);
// spring naar het einde van de notitie
script.noteTextEditSetCursorPosition(-1);
2
3
4
5
# Haal de huidige positie van de tekstcursor op in de tekstbewerking van de notitie
# Methodeaanroep en parameters
/**
* Geeft de huidige positie van de tekstcursor terug in de notitietekst
* 0 zou het begin van de notitie zijn
*/
int ScriptingService::noteTextEditCursorPosition();
2
3
4
5
# Voorbeeld
script.log(script.noteTextEditCursorPosition());
# Lees het huidige woord uit de tekst van de notitie
# Methodeaanroep en parameters
/**
* Leest het huidige woord in de notitietekst
*
* @param withPreviousCharacters krijgen in het begin ook meer tekens
* om tekens als "@" te krijgen die dat niet zijn
* woordtekens
* @return
*/
QString ScriptingService::noteTextEditCurrentWord(bool withPreviousCharacters);
2
3
4
5
6
7
8
9
# Voorbeeld
// lees het huidige woord in de tekst van de notitie
var text = script.noteTextEditCurrentWord();
2
Misschien wilt u het voorbeeld bekijken autocompletion.qml (opens new window).
# Lees het huidige blok van de notitietekstbewerking
# Methodeaanroep en parameters
/**
* Leest het huidige blok in de notitietekstbewerking
*
* @return
*/
QString ScriptingService::noteTextEditCurrentBlock();
2
3
4
5
6
# Voorbeeld
// Lees het huidige blok in de notitie tekstbewerking
var text = script.noteTextEditCurrentBlock();
2
Bestudeer het voorbeeld eens ai-autocompletion.qml (opens new window).
# Gebruik een voltooiingsprompt op het geselecteerde AI-model
De AI-voltooiingsprompt is een functie waarmee u een voltooiingsprompt kunt gebruiken op het geselecteerde AI-model.
Om dit te laten werken moet het AI-systeem worden ingeschakeld in de AI-werkbalk of het hoofdmenu.
# Methodeaanroep en parameters
/**
* QML-wrapper om de AI-completering te gebruiken
*
* @param prompt
* @return {QString} the result of the completer
*/
QString ScriptingService::aiComplete(const QString& prompt);
2
3
4
5
6
7
# Voorbeeld
// Vraag het geselecteerde AI-model om de prompt te voltooien
var text = script.aiComplete("Tell me how do you feel today?");
2
Bestudeer het voorbeeld eens ai-autocompletion.qml (opens new window).
# Controleer of het platform Linux, OS X of Windows is
# Methodeaanroep en parameters
bool ScriptingService::platformIsLinux();
bool ScriptingService::platformIsOSX();
bool ScriptingService::platformIsWindows();
2
3
# Voorbeeld
if (script.platformIsLinux()) {
// Wordt alleen uitgevoerd onder Linux
}
2
3
# Tag de huidige notitie
# Methodeaanroep en parameters
/**
* Labelt de huidige notitie met een tag met de naam tagName
*
* @param tagName
*/
void ScriptingService::tagCurrentNote(QString tagName);
2
3
4
5
6
# Voorbeeld
// voeg een "favoriete" tag toe aan de huidige notitie
script.tagCurrentNote("favorite");
2
Misschien wilt u kijken naar de aangepaste actie favoriteNote
in voorbeeld favorite-note.qml (opens new window).
# Maak of haal een tag op met de naam van de broodkruimellijst
# Methodeaanroep en parameters
/**
* Haalt of maakt een tag door zijn "broodkruimellijst" van tagnamen
* Element nameList[0] zou het hoogst zijn in de boom (met parentId: 0)
*
* @param nameList
* @param createMissing {bool} if true (default) all missing tags will be created
* @return TagApi object van de diepste tag van de naam
*/
TagApi *ScriptingService::getTagByNameBreadcrumbList(
const QStringList &nameList, bool createMissing);
2
3
4
5
6
7
8
9
10
# Voorbeeld
// maakt alle tags tot het 3e niveau en retourneert het tag-object voor
// tag "level3", die er zo uitziet in de tagboom:
// level1 > level2 > level3
var tag = script.getTagByNameBreadcrumbList (["level1", "level2", "level3"]);
2
3
4
# Zoek naar tags op naam
# Methodeaanroep en parameters
/**
* Haalt alle tags op door een substring-zoekopdracht uit te voeren in het naamveld
*
* @param name {QString} name to search for
* @return {QStringList} list of tag names
*/
QStringList ScriptingService::searchTagsByName(QString name);
2
3
4
5
6
7
# Voorbeeld
// zoekt naar alle tags met het woordspel erin
var tags = script.searchTagsByName("game");
2
Misschien wilt u het voorbeeld bekijken autocompletion.qml (opens new window).
# Zoek naar notities op notitietekst
# Methodeaanroep en parameters
/**
* Retourneert een lijst met notitie-ID's van alle notities met een bepaalde tekst in de notitietekst
*
* Helaas is er geen gemakkelijke manier om een QList<NoteApi*> in QML te gebruiken, dus we
* kan alleen de notitie-ID's overdragen
*
* @return {QList<int>} list of note ids
*/
QList<int> ScriptingService::fetchNoteIdsByNoteTextPart(QString text);
2
3
4
5
6
7
8
9
# Voorbeeld
var noteIds = script.fetchNoteIdsByNoteTextPart("mytext");
noteIds.forEach(function (noteId){
var note = script.fetchNoteById(noteId);
// doe iets met de notitie
});
2
3
4
5
6
7
8
Misschien wilt u het voorbeeld bekijken unique-note-id.qml (opens new window).
# Voeg een aangepast stylesheet toe
# Methodeaanroep en parameters
/**
* Voegt een aangepast stylesheet toe aan de applicatie
*
* @param stylesheet
*/
void ScriptingService :: addStyleSheet (QString-stylesheet);
2
3
4
5
6
# Voorbeeld
// maak de tekst in de notitielijst groter
script.addStyleSheet("QTreeWidget # noteTreeWidget {font-size: 30px;}");
2
Misschien wilt u het voorbeeld bekijken custom-stylesheet.qml (opens new window).
U kunt de widgetnamen ophalen uit de *.ui
-bestanden, bijvoorbeeld het hoofdvenster mainwindow.ui (opens new window).
De Qt-documentatie (bijvoorbeeld QMainWindow (opens new window)) kan u helpen te zien hoe de widgets aan elkaar gerelateerd zijn (zoek naar Erfenissen
op de pagina's).
De basiswidget voor bijna alles is QWidget (opens new window). Dus gewoon QWidget
stylen met bijvoorbeeld QWidget {background-color: black; kleur: wit;}
zou betekenen dat alles een zwarte achtergrondkleur en een witte voorgrondkleur heeft.
TIP
CSS style.qss (opens new window) van qdarkstyle (opens new window) kan ook een goede referentie zijn voor stijlen die u kunt wijzigen.
Bekijk Style Sheet Reference (opens new window) voor een referentie van welke stijlen beschikbaar zijn.
Als je stijlen in html-preview wilt invoegen om de manier waarop notities worden weergegeven te wijzigen, kijk dan naar notetomarkdownhtmlhook.
TIP
Als je echt wilt zien hoe de dialogen eruitzien en hoe ze heten, kun je Qt Creator (opens new window) downloaden en de *.ui
-bestanden erin openen.
# Herladen van de script-engine
# Methodeaanroep en parameters
/**
* Herlaadt de script-engine
*/
void ScriptingService::reloadScriptingEngine();
2
3
4
# Voorbeeld
// herlaad de script-engine
script.reloadScriptingEngine();
2
# Een notitie ophalen op basis van de bestandsnaam
# Methodeaanroep en parameters
/**
* Haalt een notitie op met de bestandsnaam
*
* @param fileName string de bestandsnaam van de notitie (verplicht)
* @param noteSubFolderId geheel getal-id van de notitie-submap
* @return OpmerkingApi *
*/
NoteApi * ScriptingService::fetchNoteByFileName (QString bestandsnaam,
int noteSubFolderId);
2
3
4
5
6
7
8
9
# Voorbeeld
// haal notitie op met bestandsnaam
script.fetchNoteByFileName("my note.md");
2
# Een notitie ophalen op basis van zijn ID
# Methodeaanroep en parameters
/**
* Haalt een notitie op met zijn id
*
* @param id int the id of the note
* @return NoteApi*
*/
NoteApi* ScriptingService::fetchNoteById(int id);
2
3
4
5
6
7
# Voorbeeld
// haal notitie op met id
script.fetchNoteById(243);
2
Misschien wilt u het voorbeeld eens bekijken export-notes-as-one-html.qml (opens new window).
# Controleren of een notitie bestaat op basis van de bestandsnaam
# Methodeaanroep en parameters
/**
* Controleert of een notitiebestand bestaat op basis van de bestandsnaam
*
* @param fileName string de bestandsnaam van de notitie (verplicht)
* @param ignoreNoteId integer-id van een notitie die bij de controle moet worden genegeerd
* @param noteSubFolderId geheel getal-id van de notitie-submap
* @return bool
*/
bool ScriptingService :: noteExistsByFileName (QString bestandsnaam,
int ignoreNoteId,
int noteSubFolderId);
2
3
4
5
6
7
8
9
10
11
# Voorbeeld
// controleer of opmerking bestaat, maar negeer de id van "note"
script.noteExistsByFileName("my note.md", note.id);
2
Misschien wilt u het voorbeeld eens bekijken use-tag-names-in-filename.qml (opens new window).
# Tekst naar het klembord kopiëren
# Methodeaanroep en parameters
/**
* Kopieert tekst naar het klembord als platte tekst of html mime-gegevens
*
* @param tekst string tekst om op het klembord te plaatsen
* @param asHtml bool indien waar, wordt de tekst ingesteld als html mime-gegevens
*/
void ScriptingService::setClipboardText(QString-tekst, bool asHtml);
2
3
4
5
6
7
# Voorbeeld
// kopieer tekst naar het klembord
script.setClipboardText("text to copy");
2
Misschien wilt u het voorbeeld eens bekijken selected-markdown-to-bbcode.qml (opens new window).
# Naar een notitie springen
# Methodeaanroep en parameters
/**
* Sets the current note if the note is visible in the note list
*
* @param note NoteApi note to jump to
* @param asTab bool if true the note will be opened in a new tab (if not already open)
*/
void ScriptingService::setCurrentNote(NoteApi *note, bool asTab = false);
2
3
4
5
6
7
# Voorbeeld
// jump to the note
script.setCurrentNote(note);
// open note in new tab (if not already open)
script.setCurrentNote(note, true);
2
3
4
5
You may want to take a look at the example journal-entry.qml (opens new window).
# Springen naar een submap van een notitie
# Methodeaanroep en parameters
/**
* Springt naar een submap voor notities
*
* @param noteSubFolderPath {QString} pad van de submap, relatief aan de notitiemap
* @param scheidingsteken {QString} scheidingsteken tussen delen van het pad, standaard "/"
* @return waar als de sprong succesvol was
*/
bool ScriptingService :: jumpToNoteSubFolder (const QString & amp; noteSubFolderPath,
QString-scheidingsteken);
2
3
4
5
6
7
8
9
# Voorbeeld
// spring naar de submap notitie "een submap"
script.jumpToNoteSubFolder ("een submap");
// spring naar de submap "sub" van de notitie in "een submap"
script.jumpToNoteSubFolder ("een submap / sub");
2
3
4
5
TIP
U kunt een nieuwe submap voor notities maken in de huidige submap door mainWindow.createNewNoteSubFolder
aan te roepen.
# Een informatieberichtvenster weergeven
# Methodeaanroep en parameters
/**
* Toont een informatieberichtvenster
*
* @param tekst
* @param title (optioneel)
*/
void ScriptingService:: informationMessageBox (QString-tekst, QString-titel);
2
3
4
5
6
7
# Voorbeeld
// toon een informatieberichtvenster
script.informationMessageBox ("De tekst die ik wil laten zien", "Een optionele titel");
2
# Er wordt een berichtvenster met een vraag weergegeven
# Methodeaanroep en parameters
/**
* Toont een vraagberichtvenster
*
* Voor informatie over knoppen zie:
* https://doc.qt.io/qt-5/qmessagebox.html#StandardButton-enum
*
* @param tekst
* @param titel (optioneel)
* @param knoppen knoppen die getoond moeten worden (optioneel)
* @param defaultButton standaardknop die wordt geselecteerd (optioneel)
* @return id van ingedrukte knop
*/
int ScriptingService::questionMessageBox(
QString text, QString title, int buttons, int defaultButton);
2
3
4
5
6
7
8
9
10
11
12
13
14
# Voorbeeld
// toon een vraagberichtvenster met een sollicitatie- en een helpknop
// zie: https://doc.qt.io/qt-5/qmessagebox.html#StandardButton-enum
var resultaat = script.questionMessageBox (
"De tekst die ik wil laten zien", "Een optionele titel", 0x01000000 | 0x02000000, 0x02000000);
script.log (resultaat);
2
3
4
5
Zie voor informatie over knoppen StandardButton (opens new window).
Zie voor informatie over knoppen input-dialogs.qml (opens new window).
# Een open bestandsdialoog weergeven
# Methodeaanroep en parameters
**
* Toont een open bestandsdialoog
*
* @param caption (optioneel)
* @param dir (optioneel)
* @param filter (optioneel)
* @return QString
* /
QString ScriptingService::getOpenFileName(QString caption, QString dir,
QString filter);
2
3
4
5
6
7
8
9
10
# Voorbeeld
// toon een open bestandsdialoog
var fileName = script.getOpenFileName("Please select an image", "/home/user/images", "Images (*.png *.xpm *.jpg)");
2
# Een zekere bestandsdialoog weergeven
# Methodeaanroep en parameters
/**
* Toont een dialoogvenster voor het opslaan van bestanden
*
* @param caption (optional)
* @param dir (optional)
* @param filter (optional)
* @return QString
*/
QString ScriptingService::getSaveFileName(QString caption, QString dir,
QString filter);
2
3
4
5
6
7
8
9
10
# Voorbeeld
// toon een dialoogvenster voor het opslaan van bestanden
var fileName = script.getSaveFileName("Please select HTML file to save", "output.html", "HTML (*.html)");
2
Misschien wilt u het voorbeeld eens bekijken export-notes-as-one-html.qml (opens new window).
# Variabelen voor scriptinstellingen registreren
You need to define your settings variables as properties in your script and register them in a property named settingsVariables
.
De gebruiker kan deze eigenschappen vervolgens instellen in de scriptinstellingen.
# Voorbeeld
// you have to define your registered variables so you can access them later
property string myString;
property string myStringSecret;
property bool myBoolean;
property string myText;
property int myInt;
property string myFile;
property string myDirectory;
property string mySelection;
// register your settings variables so the user can set them in the script settings
//
// unfortunately there is no QVariantHash in Qt, we only can use
// QVariantMap (that has no arbitrary ordering) or QVariantList (which at
// least can be ordered arbitrarily)
property variant settingsVariables: [
{
"identifier": "myString",
"name": "I am a line edit",
"description": "Please enter a valid string:",
"type": "string",
"default": "My default value",
},
{
"identifier": "myStringSecret",
"name": "I am a password field",
"description": "Please enter a valid string:",
"type": "string-secret",
},
{
"identifier": "myBoolean",
"name": "I am a checkbox",
"description": "Some description",
"text": "Check this checkbox",
"type": "boolean",
"default": true,
},
{
"identifier": "myText",
"name": "I am textbox",
"description": "Please enter your text:",
"type": "text",
"default": "This can be a really long text\nwith multiple lines.",
},
{
"identifier": "myInt",
"name": "I am a number selector",
"description": "Please enter a number:",
"type": "integer",
"default": 42,
},
{
"identifier": "myFile",
"name": "I am a file selector",
"description": "Please select the file:",
"type": "file",
"default": "pandoc",
},
{
"identifier": "myDirectory",
"name": "I am a directory selector",
"description": "Please select the directory:",
"type": "directory",
"default": "/home",
},
{
"identifier": "mySelection",
"name": "I am an item selector",
"description": "Please select an item:",
"type": "selection",
"default": "option2",
"items": {"option1": "Text for option 1", "option2": "Text for option 2", "option3": "Text for option 3"},
}
];
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
Daarnaast kunt u de settingsVariables
als volgt overschrijven met een speciale functie registerSettingsVariables()
:
# Voorbeeld
/**
* Registreert de instellingsvariabelen opnieuw
*
* Gebruik deze methode als u code wilt gebruiken om uw variabelen te overschrijven, zoals instelling
* standaardwaarden zijn afhankelijk van het besturingssysteem.
*/
function registerSettingsVariables() {
if (script.platformIsWindows()) {
// override the myFile default value
settingsVariables[3].default = "pandoc.exe"
}
}
2
3
4
5
6
7
8
9
10
11
12
Misschien wilt u het voorbeeld eens bekijken variables.qml (opens new window).
# Persistente variabelen opslaan en laden
# Methodeaanroep en parameters
/**
* Stores a persistent variable
* These variables are accessible globally over all scripts
* Please use a meaningful prefix in your key like "PersistentVariablesTest/myVar"
*
* @param key {QString}
* @param value {QVariant}
*/
void ScriptingService::setPersistentVariable(const QString &key,
const QVariant &value);
/**
* Loads a persistent variable
* These variables are accessible globally over all scripts
*
* @param key {QString}
* @param defaultValue {QVariant} return value if the setting doesn't exist (optional)
* @return
*/
QVariant ScriptingService::getPersistentVariable(const QString &key,
const QVariant &defaultValue);
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
# Voorbeeld
// store persistent variable
script.setPersistentVariable("PersistentVariablesTest/myVar", result);
// load and log persistent variable
script.log(script.getPersistentVariable("PersistentVariablesTest/myVar", "nothing here yet"));
2
3
4
5
Zorg ervoor dat u een betekenisvol voorvoegsel in uw sleutel gebruikt, zoals PersistentVariablesTest/myVar
, omdat de variabelen toegankelijk zijn vanuit alle scripts.
Misschien wilt u het voorbeeld eens bekijken persistent-variables.qml (opens new window).
# Variabelen voor applicatie-instellingen laden
# Methodeaanroep en parameters
/**
* Loads an application settings variable
*
* @param key {QString}
* @param defaultValue {QVariant} return value if the setting doesn't exist (optional)
* @return
*/
QVariant ScriptingService::getApplicationSettingsVariable(const QString &key,
const QVariant &defaultValue);
2
3
4
5
6
7
8
9
# Voorbeeld
// load and log an application settings variable
script.log(script.getApplicationSettingsVariable("gitExecutablePath"));
2
Houd er rekening mee dat instellingen eigenlijk leeg kunnen zijn, daar moet u zelf voor zorgen. defaultValue
wordt alleen gebruikt als de instelling helemaal niet bestaat.
# Een cachemap maken
U kunt bestanden cachen op de standaard cachelocatie van uw systeem.
# Methodeaanroep en parameters
/**
* Retourneert een cachemap voor een script
*
* @param {QString} subDir de submap die moet worden gemaakt en gebruikt
* @return {QString} het pad naar de cache-map
*/
QString ScriptingService::cacheDir(const QString &subDir) const;
2
3
4
5
6
7
# Voorbeeld
// maak de cachemap voor my-script-id
var cacheDirForScript = script.cacheDir("my-script-id");
2
# Een cachemap wissen
U kunt de cachemap van uw script wissen door de naam door te geven aan clearCacheDir().
# Methodeaanroep en parameters
/**
* Wist de cachemap voor een script
*
* @param {QString} subDir de submap die moet worden gewist
* @return {bool} true bij succes
*/
bool ScriptingService::clearCacheDir(const QString &subDir) const;
2
3
4
5
6
7
# Voorbeeld
// clear cache directory of my-script-id
script.clearCacheDir("my-script-id");
2
# Het pad naar de directory van uw script lezen
Als u het pad naar de directory waar uw script is geplaatst, nodig heeft om bijvoorbeeld andere bestanden te laden, moet u een property string scriptDirPath;
registreren. Deze eigenschap wordt ingesteld met het pad naar de directory van het script.
# Voorbeeld
import QtQml 2.0
import QOwnNotesTypes 1.0
Script {
// the path to the script's directory will be set here
property string scriptDirPath;
function init() {
script.log(scriptDirPath);
}
}
2
3
4
5
6
7
8
9
10
11
# Padscheidingstekens omzetten in native
# Methodeaanroep en parameters
/**
* Geeft pad terug met de '/' scheidingstekens geconverteerd naar scheidingstekens die zijn
* geschikt voor het onderliggende besturingssysteem.
*
* On Windows, toNativeDirSeparators("c:/winnt/system32") returns
* "c:\winnt\system32".
*
* @param path
* @return
*/
QString ScriptingService::toNativeDirSeparators(QString path);
2
3
4
5
6
7
8
9
10
11
# Voorbeeld
// retourneert "c:\winnt\system32" op Windows
script.log (script.toNativeDirSeparators ("c:/winnt/system32"));
2
# Padscheidingstekens omzetten van native
# Methodeaanroep en parameters
/**
* Geeft het pad terug met '/' als bestandseparator.
* Op Windows, bijvoorbeeld fromNativeDirSeparators("c:\\winnt\\system32")
* geeft "c: / winnt / system32" terug.
*
* @param path
* @return
*/
QString ScriptingService::fromNativeDirSeparators(QString path);
2
3
4
5
6
7
8
9
# Voorbeeld
// will return "c:/winnt/system32" on Windows
script.log(script.fromNativeDirSeparators("c:\\winnt\\system32"));
2
# Het native directoryscheidingsteken ophalen
# Methodeaanroep en parameters
/**
* Retourneert het native directoryscheidingsteken "/" of "\" in Windows
*
* @return
*/
QString ScriptingService :: dirSeparator ();
2
3
4
5
6
# Voorbeeld
// retourneert "\" in Windows
script.log(script.dirSeparator());
2
# Een lijst ophalen met de paden van alle geselecteerde notities
# Methodeaanroep en parameters
/**
* Retourneert een lijst met de paden van alle geselecteerde notities
*
* @return {QStringList} lijst met geselecteerde notitiepaden
*/
QStringList ScriptingService::selectedNotesPaths();
2
3
4
5
6
# Voorbeeld
// retourneert een lijst met de paden van alle geselecteerde notities
script.log(script.selectedNotesPaths());
2
Misschien wilt u het voorbeeld eens bekijken external-note-diff.qml (opens new window).
# Een lijst ophalen met de id's van alle geselecteerde notities
# Methodeaanroep en parameters
/**
* Retourneert een lijst met de id's van alle geselecteerde notities
*
* @return {QList <int>} lijst met geselecteerde notitie-id's
*/
QList<int> ScriptingService::selectedNotesIds();
2
3
4
5
6
# Voorbeeld
// retourneert een lijst met de paden van alle geselecteerde notities
script.log(script.selectedNotesIds());
2
Misschien wilt u het voorbeeld eens bekijken export-notes-as-one-html.qml (opens new window).
# Een menuactie activeren
# Methodeaanroep en parameters
/**
* Triggers a menu action
*
* @param objectName {QString} object name of the action to trigger
* @param checked {QString} only trigger the action if checked-state is
* different than this parameter (optional, can be 0 or 1)
*/
void ScriptingService::triggerMenuAction(QString objectName, QString checked);
2
3
4
5
6
7
8
# Voorbeeld
// schakel de alleen-lezen modus in
script.triggerMenuAction("actionAllow_note_editing");
// de alleen-lezen modus uitschakelen
script.triggerMenuAction("actionAllow_note_editing", 1);
2
3
4
5
Misschien wilt u het voorbeeld eens bekijken disable-readonly-mode.qml (opens new window).
TIP
U kunt de objectnamen van de menuactie ophalen van hoofdvenster.ui (opens new window). Zoek gewoon naar de Engelse menutitel. Houd er rekening mee dat deze teksten in de loop van de tijd kunnen veranderen.
# Een invoerdialoog openen met een selectievak
# Methodeaanroep en parameters
/**
* Opent een invoerdialoog met een selectievak
*
* @param titel {QString} titel van het dialoogvenster
* @param label {QString} labeltekst van het dialoogvenster
* @param items {QStringList} lijst met te selecteren items
* @param huidige {int} index van het item dat moet worden geselecteerd (standaard: 0)
* @param bewerkbaar {bool} indien true kan de tekst in het dialoogvenster worden bewerkt (standaard: false)
* @return {QString} tekst van het geselecteerde item
*/
QString ScriptingService::inputDialogGetItem(
const QString &title, const QString &label, const QStringList &items,
int current, bool editable);
2
3
4
5
6
7
8
9
10
11
12
13
Er wordt een lege tekenreeks geretourneerd als op Annuleren
is geklikt of Escape
is getoetst.
# Voorbeeld
var result = script.inputDialogGetItem(
"combo box", "Please select an item", ["Item 1", "Item 2", "Item 3"]);
script.log(result);
2
3
Misschien wilt u het voorbeeld eens bekijken input-dialogs.qml (opens new window).
# Een invoerdialoog openen met een regelbewerking
# Methodeaanroep en parameters
/**
* Opent een invoerdialoog met een lijnbewerking
*
* @param titel {QString} titel van het dialoogvenster
* @param label {QString} labeltekst van het dialoogvenster
* @param tekst {QString} tekst in het dialoogvenster (optioneel)
* @return
*/
QString ScriptingService::inputDialogGetText(
const QString &title, const QString &label, const QString &text);
2
3
4
5
6
7
8
9
10
Er wordt een lege tekenreeks geretourneerd als op Annuleren
is geklikt of Escape
is getoetst.
# Voorbeeld
var result = script.inputDialogGetText(
"line edit", "Please enter a name", "current text");
script.log(result);
2
3
# Een invoerdialoog openen met een tekstbewerking met meerdere regels
# Methodeaanroep en parameters
/**
* Opens an input dialog with a multi-line text edit
*
* @param title {QString} title of the dialog
* @param label {QString} label text of the dialog
* @param text {QString} text in the dialog (optional)
* @return
*/
QString ScriptingService::inputDialogGetMultiLineText(
const QString &title, const QString &label, const QString &text);
2
3
4
5
6
7
8
9
10
Er wordt een lege tekenreeks geretourneerd als op Annuleren
is geklikt of Escape
is getoetst.
# Voorbeeld
var result = script.inputDialogGetMultiLineText(
"multi-line edit", "Voer een tekst in", "huidige tekst");
script.log(result);
2
3
# Een dialoogvenster openen om de verschillen tussen twee teksten weer te geven
# Methodeaanroep en parameters
/**
* Opent een dialoogvenster om de verschillen tussen twee teksten weer te geven en laat de gebruiker het resultaat bewerken
*
* @param titel {QString} titel van het dialoogvenster
* @param label {QString} label tekst van het dialoogvenster
* @param tekst1 {QString} eerste tekst
* @param tekst2 {QString} tweede tekst
* @return
*/
QString ScriptingService::textDiffDialog(const QString &title, const QString &label,
const QString & text1, const QString &text2);
2
3
4
5
6
7
8
9
10
11
tekst2
is de tekst die je in het dialoogvenster kunt bewerken. Er wordt een lege tekenreeks geretourneerd als er op Annuleren
is geklikt of op Escape
is gedrukt.
# Voorbeeld
const text = script.noteTextEditSelectedText();
const aiPrompt = "Vertaal de tekst naar het Engels";
const aiResult = script.aiComplete(aiPrompt + ":\n\n" + tekst);
var result = script.textDiffDialog("AI Text Tool", "Resulterende tekst", tekst, aiResult);
script.log(resultaat);
2
3
4
5
6
# Controleren of er een bestand bestaat
# Methodeaanroep en parameters
**
* Controleer of er een bestand bestaat
* @param filePath
* @return
*/
bool ScriptingService::fileExists(QString &filePath);
2
3
4
5
6
# Voorbeeld
var result = script.fileExists(filePath);
script.log(result);
2
# Tekst uit een bestand lezen
# Methodeaanroep en parameters
/**
* Lees tekst uit een bestand
*
* @param filePath {QString} pad van het te laden bestand
* @param codec {QString} bestandscodering (standaard: UTF-8)
* @return de bestandsgegevens of null als het bestand niet bestaat
*/
QString ScriptingService::readFromFile(const QString &filePath, const QString &codec)
2
3
4
5
6
7
8
# Voorbeeld
if(script.fileExists(filePath)){
var data = script.readFromFile(filePath);
script.log(data);
}
2
3
4
# Tekst naar een bestand schrijven
# Methodeaanroep en parameters
/**
* Schrijft een tekst naar een bestand
*
* @param filePath {QString}
* @param data {QString}
* @param createParentDirs {bool} optional (default: false)
* @return
*/
bool ScriptingService::writeToFile(const QString &filePath, const QString &data, bool createParentDirs);
2
3
4
5
6
7
8
9
# Voorbeeld
var result = script.writeToFile(filePath, html);
script.log(result);
2
Misschien wilt u het voorbeeld eens bekijken export-notes-as-one-html.qml (opens new window).
# Werken met websockets
U kunt QOwnNotes op afstand bedienen met WebSocketServer
.
Bekijk het voorbeeld eens websocket-server.qml (opens new window). U kunt de socketserver testen door er verbinding mee te maken op Websocket-test (opens new window).
U kunt ook naar sockets luisteren met WebSocket
. Misschien wilt u dit voorbeeld bekijken websocket-client.qml (opens new window).
Onthoud dat u Qt's QML websocket
-bibliotheek moet hebben geïnstalleerd om dit te gebruiken. U kunt bijvoorbeeld onder Ubuntu Linux installeren qml-module-qtwebsockets
.
# Een markeringsregel toevoegen voor de editor
U kunt markeringsregels rechtstreeks in de editor invoegen door reguliere expressies te definiëren en deze toe te wijzen aan een markeringsstatus.
# Methodeaanroep en parameters
/**
* Voegt een markeringsregel toe aan de syntaxismarkering van de editor
*
* @param patroon {QString} het reguliere expressiepatroon om te markeren
* @param shouldContain {QString} een tekenreeks die in de gemarkeerde tekst moet staan om het patroon te kunnen ontleden
* @param state {int} de staat van de te gebruiken syntax highlighter
* @param captureGroup {int} de capture-groep voor het patroon om te gebruiken voor markering (standaard: 0)
* @param maskedGroup {int} de vastleggroep voor het patroon om te gebruiken voor maskering (standaard: 0)
*/
void ScriptingService::addHighlightingRule(const QString &patroon,
const QString &shouldContain,
int staat,
int captureGroup,
int maskedGroup);
2
3
4
5
6
7
8
9
10
11
12
13
14
# Statussen markeren
Naam | Nr. |
---|---|
NoState | -1 |
Koppeling | 0 |
Afbeelding | 3 |
CodeBlock | 4 |
CodeBlockComment | 5 |
Cursief | 7 |
Vet | 8 |
Lijst | 9 |
Opmerking | 11 |
H1 | 12 |
H2 | 13 |
H3 | 14 |
H4 | 15 |
H5 | 16 |
H6 | 17 |
BlockQuote | 18 |
HorizontalRuler | 21 |
Table | 22 |
InlineCodeBlock | 23 |
MaskedSyntax | 24 |
CurrentLineBackgroundColor | 25 |
BrokenLink | 26 |
FrontmatterBlock | 27 |
TrailingSpace | 28 |
CheckBoxUnChecked | 29 |
CheckBoxChecked | 30 |
StUnderline | 31 |
# Voorbeeld
// Markeer een tekstregel zoals "BLOCK: some text" als blockquote (status 18)
script.addHighlightingRule("^BLOCK: (.+)", "BLOCK:", 18);
// Maskeer alle tekens na 32 tekens op een regel uit (status 24)
// captureGroup 1 betekent dat de uitdrukking van het eerste deel van het patroon tussen haakjes wordt gemarkeerd
// maskedGroup -1 betekent dat er niet gemaskeerd moet worden
script.addHighlightingRule("^.{32}(.+)", "", 24, 1, -1);
2
3
4
5
6
7
U kunt de voorbeelden ook bekijken in highlighting.qml (opens new window).