Beispiele

Custom Drawing Teil 1: Zeichnen im Bild

Zeichnen im Bild

Im folgenden Use Case zeigen wir dir eine Schritt für Schritt Anleitung, wie du ein Tool bauen kannst, in dem du Bilder uploaden, in den Bildern zeichnen und diese wieder abspeichern kannst. Und das alles kinderleicht in Ninox.

Notwendige Widgets

  • Custom Drawing

  • Custom Upload

  • Custom Layout

  • Custom Grid

  • Icon

  • Button

  • Image

Vorgehensweise

Um Custom Drawing für den Use Case "Zeichnen im Bild" bei dir richtig einzusetzen, musst du einige Schritte in deiner Ninox Datenbank beachten.

  1. Anlegen von Tabellen

  • Projekte

    • Mängel (Untertabelle)

      • Dokumente (Untertabelle)

        • Shapes (Untertabelle)

  1. Felder in Shapes anlegen

Folgende Felder solltest du nun in der Untertabelle Shapes anlegen. Beachte bitte, dass die Felder genauso benannt werden, wie hier aufgelistet.

helper_shapeDataValue

  • Bezeichnung: helper_shapeDataValue

  • Typ: Text

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Server

data_shapeData

  • Bezeichnung: data_shapeData

  • Typ: Funktion

  • Code:

let current := this;
let backgroundShape := first(Dokumente.Shapes[helper_shapeDataValue != null and parseJSON(helper_shapeDataValue).type = "image"]);
let parsedShapeData := parseJSON(helper_shapeDataValue);
setItem(parsedShapeData, "x", parsedShapeData.x);
setItem(parsedShapeData, "y", parsedShapeData.y);
parsedShapeData;
let isBackgroundShape := backgroundShape = Nr;
{
	dataTableId: tableId(this),
	dataFieldId: fieldId(this, "shapeDataValue"),
	sortId: if isBackgroundShape then 0 else number(Nr) end,
	isBackgroundShape: isBackgroundShape,
	movable: first(Dokumente.Shapes[helper_shapeDataValue != null and parseJSON(helper_shapeDataValue).type = "image"]) != Nr,
	x: if backgroundShape = true then
		0
	else
		number(parseJSON(helper_shapeDataValue).x)
	end,
	y: if backgroundShape = true then
		0
	else
		number(parseJSON(helper_shapeDataValue).y)
	end,
	width: parsedShapeData.width,
	height: parsedShapeData.height,
	type: parsedShapeData.type
}

helper_disable

  • Bezeichnung: helper_disable

  • Typ: Ja / Nein

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Server

  1. Felder in Dokumente anlegen

Dateiname

  • Bezeichnung: Dateiname

  • Typ: Text

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Server

helper_base64

  • Bezeichnung: helper_base64

  • Typ: Text

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Server

helper_width

  • Bezeichnung: helper_width

  • Typ: Zahl

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Server

helper_height

  • Bezeichnung: helper_height

  • Typ: Zahl

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Server

trigger_addBackgroundImage_browser

  • Bezeichnung: trigger_addBackgroundImage_browser

  • Typ: Text

  • Zuweisung: Pro Datensatz im Server

  • Trigger nach Änderung:

let current := this;
helper_base64 := trigger_addBackgroundImage_browser;
let shapeImage := text({
		type: "image",
		href: trigger_addBackgroundImage_browser,
		width: round(current.helper_width),
		height: round(current.helper_height),
		x: 0,
		y: 0
	});
if cnt(Shapes[data_shapeData.isBackgroundImage]) = 0 then
	let newShape := (create Shapes);
	newShape.(
		Dokumente := current;
		helper_shapeDataValue := shapeImage
	)
end;
'Mängel'.(helper_selectedImageDrawing := number(current.Nr));
trigger_addBackgroundImage_browser := null

trigger_addBackgroundImage_app

  • Bezeichnung: trigger_addBackgroundImage_app

  • Typ: Text

  • Zuweisung: Pro Datensatz im Speicher (Browser)

  • Trigger nach Änderung:

let current := this;
helper_base64 := trigger_addBackgroundImage_app;
let shapeImage := text({
		type: "image",
		href: trigger_addBackgroundImage_app,
		width: round(current.helper_width),
		height: round(current.helper_height),
		x: 0,
		y: 0
	});
if cnt(Shapes[data_shapeData.isBackgroundImage]) = 0 then
	let newShape := (create Shapes);
	newShape.(
		Dokumente := current;
		helper_shapeDataValue := shapeImage
	)
end;
trigger_addBackgroundImage_app := null;
'Mängel'.(helper_selectedImageDrawing := number(current.Nr))

trigger_deleteImage

  • Bezeichnung: trigger_deleteImage

  • Typ: Ja / Nein

  • Zuweisung: Pro Datensatz im Speicher (Browser)

  • Trigger nach Änderung:

let current := this;
'Mängel'.(helper_selectedImageDrawing := first(current.'Mängel'.Dokumente[Nr != current.Nr]));
delete this
  1. Felder in Mängel anlegen

Bitte erstelle dir hier einen extra Karteireiter "Helper". Das ist hilfreich, da auf dem ersten Karteireiter dein Widget Custom Drawing angezeigt wird und keine unnötigen Ninox-Felder zu sehen sein sollen. In dem Karteireiter "Helper" legst du nun folgende Felder an:

helper_selectedImageDrawing

  • Bezeichnung: helper_selectedImageDrawing

  • Typ: Zahl

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Speicher (Browser)

helper_showLayer

  • Bezeichnung: helper_showLayer

  • Typ: Ja / Nein

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Speicher (Browser)

helper_currentColor

  • Bezeichnung: helper_currentColor

  • Typ: Text

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Speicher (Browser)

helper_strokeWidth

  • Bezeichnung: helper_strokeWidth

  • Typ: Zahl

  • Trigger nach Änderung: keine

  • Zuweisung: Pro Datensatz im Speicher (Browser)

trigger_undoLastShape

  • Bezeichnung: trigger_undoLastShape

  • Typ: Text

  • Zuweisung: Pro Datensatz im Speicher (Browser)

  • Trigger nach Änderung:

let recordSelectedImageDrawing := record(Dokumente,number(helper_selectedImageDrawing));
if trigger_undoLastShape != null then
	let lastEdited := last(recordSelectedImageDrawing.Shapes order by 'Erstellt am');
	if trigger_undoLastShape = "undo" then
		last(recordSelectedImageDrawing.Shapes[helper_disable != true and data_shapeData.isBackgroundShape != true] order by 'Erstellt am').(helper_disable := true)
	else
		if trigger_undoLastShape = "redo" then
			lastEdited.(helper_disable := if lastEdited.helper_disable = true then
					null
				else
					true
				end)
		end
	end;
	trigger_undoLastShape := null
end
  1. Widgets in Mängel einfügen

In der Tabelle Mängel fügst du nun den folgenden Anwendungscode ein. Dieser umfasst die oben beschriebenen Widgets.

let current := this;
let fontColor := "#7a83a2";
let recordSelectedImageDrawing := record(Dokumente,helper_selectedImageDrawing);
let colors := ["#FF5733", "#33FF57", "#3357FF", "#FF33A6", "#FFC300", "#DAF7A6", "#900C3F", "#581845", "#C70039", "#FFC0CB"];
let colorSet := arcCustomGrid({
			uniqueId: "colorGrid",
			width: "100%",
			height: "auto",
			columns: "auto-fill",
			columnWidth: "38px",
			gap: "5px",
			paddingY: "20px",
			paddingX: "0",
			blocks: for item in colors do
				{
					width: "38px",
					height: "38px",
					alignX: "center",
					alignY: "center",
					backgroundColor: item,
					styles: html(---
border-radius: 50%; border: 4px solid { if helper_currentColor = item then }#575b79{  else }#272A40{  end };
					---),
					value: "",
					clickAction: {
						type: "update",
						recordId: current.Nr,
						field: fieldId(current.Nr, "helper_currentColor"),
						value: item
					}
				}
			end
		});
let strokeSizes := [5, 10, 15, 20, 25, 30];
let strokeSet := arcCustomLayout({
			uniqueId: "colorGrid",
			width: "100%",
			height: "45px",
			direction: "horizontal",
			gap: "5px",
			alignX: "between",
			alignY: "center",
			paddingY: "0",
			paddingX: "0",
			blocks: for item in strokeSizes do
				{
					width: "fraction",
					height: "100%",
					alignX: "center",
					alignY: "center",
					backgroundColor: "",
					styles: html(---
align-items:center; justify-content:center;
 border-bottom:3px solid { if item = helper_strokeWidth then }{ if current.helper_currentColor then }{ current.helper_currentColor }{  else }#575b79{  end }{  else }#181C2B{  end }; { if item != 30 then }border-right:1px solid #262b40;{  else }{  end }
					---),
					value: html(---
<div style="width:{ item }px; height:{ item }px;background-color:{ if current.helper_currentColor then }{ current.helper_currentColor }{  else }#fff{  end };border-radius:50%;"></div>
					---),
					clickAction: {
						type: "update",
						recordId: current.Nr,
						field: fieldId(current.Nr, "helper_strokeWidth"),
						value: item
					}
				}
			end
		});
let imageGrid := arcCustomGrid({
			uniqueId: "imageGrid " + Nr,
			embedded: true,
			width: "100%",
			height: "100%",
			columns: "auto-fill",
			columnWidth: "50px",
			gap: "10px",
			paddingY: "0",
			paddingX: "0",
			blocks: let images := Dokumente.[{
						blockId: "",
						width: "100%",
						height: "100px",
						alignX: "center",
						alignY: "center",
						paddingY: "",
						paddingX: "",
						backgroundColor: "",
						lineHeight: "",
						styles: "",
						value: arcCustomImage({
								uniqueId: "image mangel " + Nr,
								title: Dateiname,
								width: "100%",
								height: "100%",
								backgroundSize: "cover",
								backgroundColor: "",
								style: "",
								radius: "5px",
								borderColor: "",
								link: helper_base64
							}),
						clickAction: {
							type: "update",
							recordId: current.Nr,
							field: fieldId(current.Nr, "helper_selectedImageDrawing"),
							value: number(Nr)
						}
					}];
			let new := [{
						blockId: "",
						width: "100%",
						height: "100px",
						alignX: "center",
						alignY: "center",
						paddingY: "",
						paddingX: "",
						backgroundColor: "#272a40",
						lineHeight: "",
						styles: "",
						value: arcCustomUpload({
								uniqueId: "multi image upload in drawing " + current.Nr,
								capture: true,
								multiupload: true,
								embedded: true,
								container: {
									icon: "",
									label: "Bilder Upload",
									height: "100%",
									width: "100%",
									value: arcCustomIcon({
											name: "plus",
											color: "#fff"
										})
								},
								filename: {
									tableId: tableId("Dokumente"),
									fieldId: fieldId("Dokumente", "Dateiname")
								},
								image: {
									tableId: tableId("Dokumente"),
									fieldId: if ninoxApp() = "web" then
										fieldId("Dokumente", "trigger_addBackgroundImage_browser")
									else
										fieldId("Dokumente", "trigger_addBackgroundImage_app")
									end,
									width: 2000,
									height: 1000,
									widthFieldId: fieldId("Dokumente", "helper_width"),
									heightFieldId: fieldId("Dokumente", "helper_height")
								},
								changeFieldValues: [{
										fieldId: fieldId("Dokumente", "Mängel"),
										value: number(current.Nr)
									}]
							})
					}];
			array(images, new)
		});
let drawingSticker := [{
			uid: "red_sticker",
			image: "",
			icon: "",
			label: "Red Sticker",
			width: 100,
			height: 100,
			originX: 0.5,
			originY: 0.89,
			default: true
		}, {
			uid: "green_sticker",
			image: "",
			icon: "",
			label: "Green Sticker",
			width: 100,
			height: 100,
			originX: 0.5,
			originY: 0.89
		}, {
			uid: "red_arrow_up_right",
			image: "",
			icon: "",
			label: "Pfeil 1",
			width: 100,
			height: 100,
			originX: 0.5,
			originY: 0.5,
			default: true
		}, {
			uid: "red_arrow_down_left",
			image: "",
			icon: "",
			label: "Pfeil 2",
			width: 100,
			height: 100,
			originX: 0,
			originY: -0.0001
		}];
let drawingArea := arcCustomDrawing({
			uniqueId: "issue board" + Nr,
			embedded: true,
			height: "100%",
			tableId: tableId("Shapes"),
			fieldId: fieldId("Shapes", "helper_shapeDataValue"),
			changeFieldValues: [{
					fieldId: fieldId("Shapes", "Dokumente"),
					value: helper_selectedImageDrawing
				}],
			exportSettings: {
				allowedTypes: ["jpg"],
				target: "field",
				recordId: recordSelectedImageDrawing.Nr,
				fieldId: fieldId(recordSelectedImageDrawing.Nr, "helper_base64")
			},
			canvas: {
				width: recordSelectedImageDrawing.helper_width,
				height: recordSelectedImageDrawing.helper_height,
				offsetX: 0,
				offsetY: 0
			},
			zooming: {
				step: 0.25,
				min: 0.3,
				max: 4
			},
			drawingSettings: {
				strokeWidth: if helper_strokeWidth then helper_strokeWidth else 5 end,
				strokeColor: if helper_currentColor then helper_currentColor else "#e9595c" end
			},
			artboard: {
				width: recordSelectedImageDrawing.helper_width,
				height: recordSelectedImageDrawing.helper_height
			},
			stickerTypes: drawingSticker,
			shapes: (recordSelectedImageDrawing.Shapes[helper_disable != true] order by data_shapeData.sortId).[{
					shapeId: Nr,
					movable: data_shapeData.movable,
					shapeDataValue: helper_shapeDataValue,
					sidebarItem: {
						header: "Object ID:" + Nr,
						content: arcCustomButton({
								title: "Öffnen",
								actions: [{
										type: "popup",
										recordId: Nr
									}]
							})
					}
				}],
			rightSideContent: {
				show: true,
				header: arcCustomLayout({
						uniqueId: "drawing right side header " + Nr,
						embedded: true,
						direction: "vertical",
						alignX: "left",
						alignY: "top",
						width: "100%",
						height: "100%",
						gap: "10px",
						backgroundColor: "",
						paddingY: "20px",
						paddingX: "20px",
						styles: "",
						scrollSettings: {
							scrollY: false,
							scrollX: false
						},
						blocks: [{
								width: "",
								height: "auto",
								lineHeight: "",
								alignX: "left",
								color: "",
								value: imageGrid
							}]
					}),
				content: {
					showShapeLayers: helper_showLayer,
					value: ""
				},
				footer: arcCustomLayout({
						uniqueId: "drawing right side footer " + Nr,
						embedded: true,
						direction: "vertical",
						alignX: "left",
						alignY: "center",
						width: "100%",
						height: "",
						gap: "0",
						backgroundColor: "",
						paddingY: "0",
						paddingX: "0",
						styles: "",
						scrollSettings: {
							scrollY: false,
							scrollX: false
						},
						blocks: [{
								width: "100%",
								height: "fraction",
								lineHeight: "",
								alignX: "left",
								color: "",
								styles: "border-bottom: 1px solid #262b40;",
								value: ""
							}, {
								width: "100%",
								height: "auto",
								lineHeight: "",
								alignX: "left",
								color: "",
								styles: "border-bottom: 1px solid #262b40;",
								value: strokeSet
							}, {
								width: "100%",
								height: "auto",
								lineHeight: "",
								alignX: "left",
								color: "",
								styles: "border-bottom: 1px solid #262b40;padding:8px;",
								value: colorSet
							}, {
								width: "100%",
								height: "auto",
								lineHeight: "",
								alignX: "left",
								color: "",
								styles: "padding:8px",
								value: arcCustomButton({
										uniqueId: "undo action" + Nr,
										title: "Aktion rückgängig",
										width: "100%",
										height: "",
										alignY: "",
										alignX: "30px",
										gap: "5px",
										fontSize: "16px",
										fontColor: "#fff",
										backgroundColor: "#272A40",
										borderColor: "transparent",
										borderRadius: "5px",
										showBadge: false,
										badgeTitle: "",
										badgeColor: "",
										badgeBackground: "",
										badgeBorderColor: "",
										badgePosition: "",
										actions: let id := this;
										[{
												type: "update",
												recordId: current.Nr,
												field: fieldId(current.Nr, "trigger_undoLastShape"),
												value: "undo"
											}]
									})
							}, {
								width: "100%",
								height: "auto",
								lineHeight: "",
								alignX: "left",
								color: "",
								styles: "padding:8px",
								value: arcCustomButton({
										uniqueId: "undo action" + Nr,
										title: "Löschen",
										width: "100%",
										height: "",
										alignY: "",
										alignX: "30px",
										gap: "5px",
										fontSize: "16px",
										fontColor: "#fff",
										backgroundColor: "#e9595c",
										borderColor: "transparent",
										borderRadius: "5px",
										showBadge: false,
										badgeTitle: "",
										badgeColor: "",
										badgeBackground: "",
										badgeBorderColor: "",
										badgePosition: "",
										actions: let id := this;
										[{
												type: "update",
												recordId: recordSelectedImageDrawing.Nr,
												field: fieldId(recordSelectedImageDrawing.Nr, "trigger_deleteImage"),
												value: true
											}]
									})
							}]
					})
			}
		});
arcCustomLayout({
		uniqueId: "drawing " + Nr,
		embedded: false,
		fullscreen: true,
		ninoxVersion: "3.13",
		page: false,
		fullscreenMode: if isAdminMode() then "" else "full" end,
		showAdminTools: false,
		hideHeaderIcons: false,
		direction: "vertical",
		alignX: "left",
		alignY: "center",
		width: "100%",
		height: "100%",
		gap: "0",
		backgroundColor: "#181d2c",
		paddingY: "",
		paddingX: "",
		styles: "",
		scrollSettings: {
			scrollY: true
		},
		blocks: [if false then
				{
					width: "",
					height: "100px",
					lineHeight: "",
					alignX: "left",
					paddingX: "",
					styles: "",
					value: arcCustomLayout({
							uniqueId: "layout header " + Nr,
							embedded: true,
							direction: "horizontal",
							alignX: "left",
							alignY: "center",
							width: "100%",
							height: "100%",
							gap: "10px",
							backgroundColor: "",
							paddingY: "",
							paddingX: "",
							styles: "border-bottom: 1px solid #262b40;",
							scrollSettings: {
								scrollY: true
							},
							blocks: [{
									width: "fraction",
									height: "auto",
									lineHeight: "",
									alignX: "left",
									color: "",
									styles: "font-size:24px;color:#7a83a2;text-wrap:nowrap;padding:0 40px;",
									value: html(---
 Title
									---)
								}, {
									width: "auto",
									height: "auto",
									lineHeight: "",
									alignX: "right",
									color: "",
									value: arcCustomButton({
											uniqueId: "Button schließen" + Nr,
											title: "Schließen",
											width: "",
											height: "",
											alignY: "",
											alignX: "",
											paddingX: "",
											paddingY: "",
											gap: "5px",
											icon: arcCustomIcon({
													name: "x",
													color: fontColor
												}),
											iconPosition: "right",
											fontSize: "",
											fontColor: fontColor,
											backgroundColor: "transparent",
											borderColor: "transparent",
											borderRadius: "5px",
											actions: [{
													type: "openRecord",
													tab: "Allgemeines",
													recordId: current.Nr
												}]
										})
								}]
						})
				}
			end, if false then
				{
					width: "100%",
					height: "50px",
					lineHeight: "",
					alignX: "left",
					color: "",
					backgroundColor: "#141925",
					styles: "border-bottom: 1px solid #262b40;border-top: 1px solid #262b40;",
					value: arcCustomLayout({
							uniqueId: "layout action bar " + Nr,
							embedded: true,
							fullscreen: false,
							ninoxVersion: "",
							page: true,
							fullscreenMode: "",
							showAdminTools: true,
							hideHeaderIcons: true,
							direction: "horizontal",
							alignX: "left",
							alignY: "center",
							width: "",
							height: "",
							gap: "10px",
							backgroundColor: "",
							paddingY: "",
							paddingX: "",
							styles: "",
							scrollSettings: {
								scrollY: false,
								scrollX: false
							},
							blocks: [{
									width: "200px",
									height: "100%",
									lineHeight: "",
									alignX: "left",
									color: "",
									styles: "",
									value: "block 1"
								}, {
									width: "",
									height: "auto",
									lineHeight: "",
									alignX: "left",
									color: "",
									value: "block 2"
								}]
						})
				}
			end, {
				width: "100%",
				height: "100%",
				lineHeight: "",
				alignX: "left",
				color: "",
				backgroundColor: "#181d2c",
				value: drawingArea
			}]
	})

Tadaaaaa 🧙🏼‍♀️ Das Widget Zeichnen im Bild sollte nun funktionsfähig bei dir laufen.

Arc Rider Ventures GmbH

© 2025

Arc Rider Ventures GmbH

© 2025