1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-14 18:27:44 +02:00

Plugins: Adding support for command boolean expressions

This commit is contained in:
Laurent Cozic 2020-10-17 16:50:07 +01:00
parent f529adac99
commit f8f46db910
7 changed files with 92 additions and 13 deletions

View File

@ -275,4 +275,59 @@ describe('services_CommandService', function() {
expect(propValue).toBe('hello');
}));
it('should allow isEnabled expressions', asyncTest(async () => {
const service = newService();
registerCommand(service, createCommand('test1', {
isEnabled: 'selectedNoteCount == 1 && isMarkdownEditor',
execute: () => {},
}));
expect(service.isEnabled('test1', null, {
selectedNoteCount: 2,
isMarkdownEditor: true,
})).toBe(false);
expect(service.isEnabled('test1', null, {
selectedNoteCount: 1,
isMarkdownEditor: true,
})).toBe(true);
expect(service.isEnabled('test1', null, {
selectedNoteCount: 1,
isMarkdownEditor: false,
})).toBe(false);
}));
it('should enable and disable toolbar buttons depending on boolean expression', asyncTest(async () => {
const service = newService();
const toolbarButtonUtils = new ToolbarButtonUtils(service);
const state:any = {
selectedNoteIds: ['note1', 'note2'],
};
registerCommand(service, createCommand('test1', {
execute: () => {},
isEnabled: 'selectedNoteCount == 1 && selectedNoteId == note2',
}));
{
const toolbarInfos = toolbarButtonUtils.commandsToToolbarButtons(state, ['test1']);
expect(toolbarInfos[0].enabled).toBe(false);
}
{
state.selectedNoteIds = ['note1'];
const toolbarInfos = toolbarButtonUtils.commandsToToolbarButtons(state, ['test1']);
expect(toolbarInfos[0].enabled).toBe(false);
}
{
state.selectedNoteIds = ['note2'];
const toolbarInfos = toolbarButtonUtils.commandsToToolbarButtons(state, ['test1']);
expect(toolbarInfos[0].enabled).toBe(true);
}
}));
});

View File

@ -13,7 +13,7 @@ export const runtime = ():CommandRuntime => {
return CommandService.instance().execute('newNote', { template: template, isTodo: true });
},
isEnabled: () => {
return CommandService.instance().isEnabled('newNote', {});
return CommandService.instance().isEnabled('newNote', {}, null);
},
title: () => {
return _('New to-do');

View File

@ -788,7 +788,7 @@ function useMenu(props:Props) {
useEffect(() => {
for (const commandName in props.menuItemProps) {
if (!props.menuItemProps[commandName]) continue;
menuItemSetEnabled(commandName, CommandService.instance().isEnabled(commandName, props.menuItemProps[commandName]));
menuItemSetEnabled(commandName, CommandService.instance().isEnabled(commandName, props.menuItemProps[commandName], null));
}
const layoutButtonSequenceOptions = Setting.enumOptions('layoutButtonSequence');

View File

@ -2,12 +2,15 @@ import eventManager from 'lib/eventManager';
import markdownUtils, { MarkdownTableHeader, MarkdownTableRow } from 'lib/markdownUtils';
import BaseService from 'lib/services/BaseService';
import shim from 'lib/shim';
import BooleanExpression from './BooleanExpression';
type LabelFunction = () => string;
type IsEnabledFunction = (props:any) => boolean;
type IsEnabledExpression = string;
export interface CommandRuntime {
execute(props:any):Promise<any>
isEnabled?(props:any):boolean
isEnabled?: IsEnabledFunction | IsEnabledExpression;
// "state" type is "AppState" but in order not to introduce a
// dependency to the desktop app (so that the service can
@ -197,11 +200,23 @@ export default class CommandService extends BaseService {
}, 10);
}
isEnabled(commandName:string, props:any):boolean {
public booleanExpressionContextFromState(state:any) {
return {
selectedNoteCount: state.selectedNoteIds.length,
selectedNoteId: state.selectedNoteIds.length === 1 ? state.selectedNoteIds[0] : null,
};
}
isEnabled(commandName:string, props:any, booleanExpressionContext:any):boolean {
const command = this.commandByName(commandName);
if (!command || !command.runtime) return false;
// if (!command.runtime.props) return false;
return command.runtime.isEnabled(props);
if (typeof command.runtime.isEnabled === 'function') {
return command.runtime.isEnabled(props);
} else {
const exp = new BooleanExpression(command.runtime.isEnabled);
return exp.evaluate(booleanExpressionContext);
}
}
commandMapStateToProps(commandName:string, state:any):any {

View File

@ -35,8 +35,13 @@ export default class ToolbarButtonUtils {
return this.service_;
}
private commandToToolbarButton(commandName:string, props:any):ToolbarButtonInfo {
if (this.toolbarButtonCache_[commandName] && !propsHaveChanged(this.toolbarButtonCache_[commandName].props, props)) {
private commandToToolbarButton(commandName:string, props:any, booleanExpressionContext:any):ToolbarButtonInfo {
const newEnabled = this.service.isEnabled(commandName, props, booleanExpressionContext);
if (
this.toolbarButtonCache_[commandName] &&
!propsHaveChanged(this.toolbarButtonCache_[commandName].props, props) &&
this.toolbarButtonCache_[commandName].info.enabled === newEnabled) {
return this.toolbarButtonCache_[commandName].info;
}
@ -46,7 +51,7 @@ export default class ToolbarButtonUtils {
name: commandName,
tooltip: this.service.label(commandName),
iconName: command.declaration.iconName,
enabled: this.service.isEnabled(commandName, props),
enabled: newEnabled,
onClick: async () => {
this.service.execute(commandName, props);
},
@ -67,6 +72,8 @@ export default class ToolbarButtonUtils {
public commandsToToolbarButtons(state:any, commandNames:string[]):ToolbarButtonInfo[] {
const output:ToolbarButtonInfo[] = [];
const booleanExpressionContext = this.service.booleanExpressionContextFromState(state);
for (const commandName of commandNames) {
if (commandName === '-') {
output.push(separatorItem as any);
@ -74,7 +81,7 @@ export default class ToolbarButtonUtils {
}
const props = this.service.commandMapStateToProps(commandName, state);
output.push(this.commandToToolbarButton(commandName, props));
output.push(this.commandToToolbarButton(commandName, props, booleanExpressionContext));
}
return stateUtils.selectArrayShallow({ array: output }, commandNames.join('_'));

View File

@ -87,7 +87,7 @@ function filterLogs(logs, platform) {
if (platform === 'android' && prefix.indexOf('android') >= 0) addIt = true;
if (platform === 'ios' && prefix.indexOf('ios') >= 0) addIt = true;
if (platform === 'desktop' && prefix.indexOf('desktop') >= 0) addIt = true;
if (platform === 'desktop' && (prefix.indexOf('desktop') >= 0 || prefix.indexOf('api') >= 0)) addIt = true;
if (platform === 'desktop' && (prefix.indexOf('desktop') >= 0 || prefix.indexOf('api') >= 0 || prefix.indexOf('plugins') >= 0)) addIt = true;
if (platform === 'cli' && prefix.indexOf('cli') >= 0) addIt = true;
if (platform === 'clipper' && prefix.indexOf('clipper') >= 0) addIt = true;
@ -121,7 +121,7 @@ function formatCommitMessage(msg, author, options) {
const isPlatformPrefix = prefix => {
prefix = prefix.split(',').map(p => p.trim().toLowerCase());
for (const p of prefix) {
if (['android', 'mobile', 'ios', 'desktop', 'cli', 'clipper', 'all', 'api'].indexOf(p) >= 0) return true;
if (['android', 'mobile', 'ios', 'desktop', 'cli', 'clipper', 'all', 'api', 'plugins'].indexOf(p) >= 0) return true;
}
return false;
};
@ -129,6 +129,7 @@ function formatCommitMessage(msg, author, options) {
if (splitted.length) {
const platform = splitted[0].trim().toLowerCase();
if (platform === 'api') subModule = 'api';
if (platform === 'plugins') subModule = 'plugins';
if (isPlatformPrefix(platform)) {
splitted.splice(0, 1);
}

View File

@ -360,7 +360,8 @@
"CliClient/tests/support/plugins/settings/**/node_modules/": true,
"CliClient/tests/support/plugins/selected_text/dist/*": true,
"CliClient/tests/support/plugins/selected_text/**/node_modules/": true,
"**/CliClient/build/": true
"**/CliClient/build/": true,
"**/*.js": {"when": "$(basename).ts"},
},
"spellright.language": [
"en"