gdpr audit implemented, email log, vollmachten, pdf delete cancel data privacy and vollmachten, removed message no id card in engergy car, and other contracts that are not telecom contracts, added insert counter for engery

This commit is contained in:
2026-03-21 11:59:53 +01:00
parent 89cf92eaf5
commit f2876f877e
1491 changed files with 265550 additions and 1292 deletions
+23
View File
@@ -0,0 +1,23 @@
This module defines a plugin for attaching _input rules_ to an editor,
which can react to or transform text typed by the user. It also comes
with a bunch of default rules that can be enabled in this plugin.
@InputRule
@inputRules
@undoInputRule
The module comes with a number of predefined rules:
@emDash
@ellipsis
@openDoubleQuote
@closeDoubleQuote
@openSingleQuote
@closeSingleQuote
@smartQuotes
These utility functions take schema-specific parameters and create
input rules specific to that schema.
@wrappingInputRule
@textblockTypeInputRule
+4
View File
@@ -0,0 +1,4 @@
export {InputRule, inputRules, undoInputRule} from "./inputrules"
export {emDash, ellipsis, openDoubleQuote, closeDoubleQuote, openSingleQuote, closeSingleQuote,
smartQuotes} from "./rules"
export {wrappingInputRule, textblockTypeInputRule} from "./rulebuilders"
+167
View File
@@ -0,0 +1,167 @@
import {Plugin, Transaction, EditorState, TextSelection, Command} from "prosemirror-state"
import {EditorView} from "prosemirror-view"
/// Input rules are regular expressions describing a piece of text
/// that, when typed, causes something to happen. This might be
/// changing two dashes into an emdash, wrapping a paragraph starting
/// with `"> "` into a blockquote, or something entirely different.
export class InputRule {
/// @internal
handler: (state: EditorState, match: RegExpMatchArray, start: number, end: number) => Transaction | null
/// @internal
undoable: boolean
inCode: boolean | "only"
inCodeMark: boolean | "only"
/// Create an input rule. The rule applies when the user typed
/// something and the text directly in front of the cursor matches
/// `match`, which should end with `$`.
///
/// The `handler` can be a string, in which case the matched text, or
/// the first matched group in the regexp, is replaced by that
/// string.
///
/// Or a it can be a function, which will be called with the match
/// array produced by
/// [`RegExp.exec`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/RegExp/exec),
/// as well as the start and end of the matched range, and which can
/// return a [transaction](#state.Transaction) that describes the
/// rule's effect, or null to indicate the input was not handled.
constructor(
/// @internal
readonly match: RegExp,
handler: string | ((state: EditorState, match: RegExpMatchArray, start: number, end: number) => Transaction | null),
options: {
/// When set to false,
/// [`undoInputRule`](#inputrules.undoInputRule) doesn't work on
/// this rule.
undoable?: boolean,
/// By default, input rules will not apply inside nodes marked
/// as [code](#model.NodeSpec.code). Set this to true to change
/// that, or to `"only"` to _only_ match in such nodes.
inCode?: boolean | "only"
/// When set to `false`, this rule will not fire inside marks
/// marked as [code](#model.MarkSpec.code). The default is
/// `true`.
inCodeMark?: boolean
} = {}
) {
this.match = match
this.handler = typeof handler == "string" ? stringHandler(handler) : handler
this.undoable = options.undoable !== false
this.inCode = options.inCode || false
this.inCodeMark = options.inCodeMark !== false
}
}
function stringHandler(string: string) {
return function(state: EditorState, match: RegExpMatchArray, start: number, end: number) {
let insert = string
if (match[1]) {
let offset = match[0].lastIndexOf(match[1])
insert += match[0].slice(offset + match[1].length)
start += offset
let cutOff = start - end
if (cutOff > 0) {
insert = match[0].slice(offset - cutOff, offset) + insert
start = end
}
}
return state.tr.insertText(insert, start, end)
}
}
const MAX_MATCH = 500
type PluginState = {transform: Transaction, from: number, to: number, text: string} | null
/// Create an input rules plugin. When enabled, it will cause text
/// input that matches any of the given rules to trigger the rule's
/// action.
export function inputRules({rules}: {rules: readonly InputRule[]}) {
let plugin: Plugin<PluginState> = new Plugin<PluginState>({
state: {
init() { return null },
apply(this: typeof plugin, tr, prev) {
let stored = tr.getMeta(this)
if (stored) return stored
return tr.selectionSet || tr.docChanged ? null : prev
}
},
props: {
handleTextInput(view, from, to, text) {
return run(view, from, to, text, rules, plugin)
},
handleDOMEvents: {
compositionend: (view) => {
setTimeout(() => {
let {$cursor} = view.state.selection as TextSelection
if ($cursor) run(view, $cursor.pos, $cursor.pos, "", rules, plugin)
})
}
}
},
isInputRules: true
})
return plugin
}
function run(view: EditorView, from: number, to: number, text: string, rules: readonly InputRule[], plugin: Plugin) {
if (view.composing) return false
let state = view.state, $from = state.doc.resolve(from)
let textBefore = $from.parent.textBetween(Math.max(0, $from.parentOffset - MAX_MATCH), $from.parentOffset,
null, "\ufffc") + text
for (let i = 0; i < rules.length; i++) {
let rule = rules[i];
if (!rule.inCodeMark && $from.marks().some(m => m.type.spec.code)) continue
if ($from.parent.type.spec.code) {
if (!rule.inCode) continue
} else if (rule.inCode === "only") {
continue
}
let match = rule.match.exec(textBefore)
if (!match || match[0].length < text.length) continue
let startPos = from - (match[0].length - text.length)
if (!rule.inCodeMark) {
let hasMark = false
state.doc.nodesBetween(startPos, $from.pos, node => {
if (node.isInline && node.marks.some(m => m.type.spec.code)) hasMark = true
})
if (hasMark) continue
}
let tr = rule.handler(state, match, startPos, to)
if (!tr) continue
if (rule.undoable) tr.setMeta(plugin, {transform: tr, from, to, text})
view.dispatch(tr)
return true
}
return false
}
/// This is a command that will undo an input rule, if applying such a
/// rule was the last thing that the user did.
export const undoInputRule: Command = (state, dispatch) => {
let plugins = state.plugins
for (let i = 0; i < plugins.length; i++) {
let plugin = plugins[i], undoable
if ((plugin.spec as any).isInputRules && (undoable = plugin.getState(state))) {
if (dispatch) {
let tr = state.tr, toUndo = undoable.transform
for (let j = toUndo.steps.length - 1; j >= 0; j--)
tr.step(toUndo.steps[j].invert(toUndo.docs[j]))
if (undoable.text) {
let marks = tr.doc.resolve(undoable.from).marks()
tr.replaceWith(undoable.from, undoable.to, state.schema.text(undoable.text, marks))
} else {
tr.delete(undoable.from, undoable.to)
}
dispatch(tr)
}
return true
}
}
return false
}
+59
View File
@@ -0,0 +1,59 @@
import {InputRule} from "./inputrules"
import {findWrapping, canJoin} from "prosemirror-transform"
import {NodeType, Node, Attrs} from "prosemirror-model"
/// Build an input rule for automatically wrapping a textblock when a
/// given string is typed. The `regexp` argument is
/// directly passed through to the `InputRule` constructor. You'll
/// probably want the regexp to start with `^`, so that the pattern can
/// only occur at the start of a textblock.
///
/// `nodeType` is the type of node to wrap in. If it needs attributes,
/// you can either pass them directly, or pass a function that will
/// compute them from the regular expression match.
///
/// By default, if there's a node with the same type above the newly
/// wrapped node, the rule will try to [join](#transform.Transform.join) those
/// two nodes. You can pass a join predicate, which takes a regular
/// expression match and the node before the wrapped node, and can
/// return a boolean to indicate whether a join should happen.
export function wrappingInputRule(
regexp: RegExp,
nodeType: NodeType,
getAttrs: Attrs | null | ((matches: RegExpMatchArray) => Attrs | null) = null,
joinPredicate?: (match: RegExpMatchArray, node: Node) => boolean
) {
return new InputRule(regexp, (state, match, start, end) => {
let attrs = getAttrs instanceof Function ? getAttrs(match) : getAttrs
let tr = state.tr.delete(start, end)
let $start = tr.doc.resolve(start), range = $start.blockRange(), wrapping = range && findWrapping(range, nodeType, attrs)
if (!wrapping) return null
tr.wrap(range!, wrapping)
let before = tr.doc.resolve(start - 1).nodeBefore
if (before && before.type == nodeType && canJoin(tr.doc, start - 1) &&
(!joinPredicate || joinPredicate(match, before)))
tr.join(start - 1)
return tr
})
}
/// Build an input rule that changes the type of a textblock when the
/// matched text is typed into it. You'll usually want to start your
/// regexp with `^` to that it is only matched at the start of a
/// textblock. The optional `getAttrs` parameter can be used to compute
/// the new node's attributes, and works the same as in the
/// `wrappingInputRule` function.
export function textblockTypeInputRule(
regexp: RegExp,
nodeType: NodeType,
getAttrs: Attrs | null | ((match: RegExpMatchArray) => Attrs | null) = null
) {
return new InputRule(regexp, (state, match, start, end) => {
let $start = state.doc.resolve(start)
let attrs = getAttrs instanceof Function ? getAttrs(match) : getAttrs
if (!$start.node(-1).canReplaceWith($start.index(-1), $start.indexAfter(-1), nodeType)) return null
return state.tr
.delete(start, end)
.setBlockType(start, start, nodeType, attrs)
})
}
+17
View File
@@ -0,0 +1,17 @@
import {InputRule} from "./inputrules"
/// Converts double dashes to an emdash.
export const emDash = new InputRule(/--$/, "—", {inCodeMark: false})
/// Converts three dots to an ellipsis character.
export const ellipsis = new InputRule(/\.\.\.$/, "…", {inCodeMark: false})
/// “Smart” opening double quotes.
export const openDoubleQuote = new InputRule(/(?:^|[\s\{\[\(\<'"\u2018\u201C])(")$/, "“", {inCodeMark: false})
/// “Smart” closing double quotes.
export const closeDoubleQuote = new InputRule(/"$/, "”", {inCodeMark: false})
/// “Smart” opening single quotes.
export const openSingleQuote = new InputRule(/(?:^|[\s\{\[\(\<'"\u2018\u201C])(')$/, "", {inCodeMark: false})
/// “Smart” closing single quotes.
export const closeSingleQuote = new InputRule(/'$/, "", {inCodeMark: false})
/// Smart-quote related input rules.
export const smartQuotes: readonly InputRule[] = [openDoubleQuote, closeDoubleQuote, openSingleQuote, closeSingleQuote]