Hello dear ProseMirror community,
I’m trying to create a PM-based PoC for our text report generation engine, which is already in production for several years. I need some help / inspiration in modelling it.
The Engine
The text generation engine is template-based and has simple “rewriting” rules (a.k.a. placeholders), which imply a finite context-free grammar. For one specific placeholder, which rule to apply is directly or indirectly, but only dependent on the external data source (e.g. user input).
Here take food as an analogy:
Data source
Vegetarian: Undefined | No | Vegetarian | Vegan
(If !Vegetarian=Yes) Meat: Beef? Pork? Chicken? Fish? Other?
(If Meat.Other=Selected) Specify other meat: <Text Input>
Vegetables: Broccoli? Cabbage? Cucumber? …
Fruits: Apple? Banana? Kiwi? …
Note that “Meat”, “Vegetables” and “Fruits” are multiple selects.
Engine Output
–Profile–
I don’t eat meat. | I like eating meat. | I like eating beef. | I like eating beef and pork. | I like eating beef, pork and snail.
…
–Recipes–
-Beef dishes- (If Meat.Beef=Selected)
Roast beef [Image Roast beef]
Ginger beef [Image Ginger beef]
…
Under the hood it’s a tree of rewrites:
[h1]–Profile–[/h1]
{MEAT}{VEGETABLES}{FRUITS}
[h1]–Recipes–[/h1]
{MEAT_RECIPE}{VEGETABLE_RECIPE}{FRUIT_RECIPE}
With:
{MEAT} = I like eating
Meat.Beef → beef
Meat.Duck → duck
…
{MEAT_RECIPE} = {BEEF_RECIPE}{DUCK_RECIPE}…
{BEEF_RECIPE} = [h2]-Beef dishes-[/h2][br]Roast beef[image] …
…
That’s a simple example. Our {PLACEHOLDER} may contain inline or block content and supports a very simple markup syntax. The inline content (e.g. a word in a sentence) may cross the boundary between 2 placeholders like {"a"}{"pple"}
.
Feature 1: Switchable View Modes
Currently I’m planning for an editor PoC for our template editors and translators. I’m thinking about having 2 view modes of a placeholder:
Placeholder (Atomic): I am {VEGETARIAN}.
The placeholder interacts like a #hashtag or @mention.
In-place edit (Inline): The text of a placeholder will be interposed in place. The user may switch between one of the following variants and can directly edit the rich-text in the placeholder.
I am {“vegetarian and I love vegetables”}. (Placeholder output for Vegetarian=vegetarian)
I am {“vegan and I love vegetables”}. (Vegetarian=vegan)
I am {“a meat-eater”}. (Vegetarian=undefined or no)
So the user can switch the view to {PLACEHOLDER} or different {“in-place”} variants while the rest of the editor stay the same. The switch operation alone should not create an undo history (from the user perspective) because the underlying data is not changed.
Feature 2: In-place Edit and Change Propagation
In the in-place edit view if there’s a text edit on the selected variant, the changes could be optionally propagated to other selection variants, when part of the changed text also exists in that variant.
Here an en->de translation example, the user edits on the in-place variant "Vegetarian=vegan"
I am {“vegan and I love vegetables”}. → Ich bin {“Veganer und ich liebe Gemüse”}.
The change will be propagated to "Vegetarian=vegetarian"
{“vegetarian and I love vegetables”} → {“?vegetarian? und ich liebe Gemüse”}
But not to "Vegetarian=undefined or no"
{“a meat-eater”} → N/A
Here "a meat-eater"
is not changed since it doesn’t match the sentence being edited. In the background a simple natural language comparison algorithm determines the rules for the propagation. The UI can list all variants of this placeholder and allows the user to revert the propagation (“Don’t change other variants”).
Feature 3: Mostly natural language related
There are also processes which are not aligned to the boundary of this placeholder model (similar to the PM class Slice vs. the Node/Mark model). For example we do named-entity (also called term in the translation industry) recognition and tagging, which operates on the word level, but one word may actually cross placeholders as I mentioned before, e.g. {"A"}{"pple"}
. Linting on sentences is another important use case.
Your opinion?
I see many similarities in the data model of PM as it also operates on a tree (the DOM). Our placeholder in the in-place view for example is very much a PM Mark. I’m considering first giving it a try here. For the “switchable view modes” of placeholders and the “edit propagation” I probably need to shadow their PM nodes?
Since I’m quite new to PM, I’d like to ask your opinions on how to better map our models to PM and how to manage history for the “switchable view modes” feature. What level of customization do you think would be a better approach?
Thank you!