diff --git a/apps/poc-state-plugin/src/app/app.component.ts b/apps/poc-state-plugin/src/app/app.component.ts index d33ef96..ab1cda1 100644 --- a/apps/poc-state-plugin/src/app/app.component.ts +++ b/apps/poc-state-plugin/src/app/app.component.ts @@ -83,6 +83,13 @@ import type { PenpotShape } from '@penpot/plugin-types'; > WORDS STYLES +
@@ -205,6 +212,10 @@ export class AppComponent { this.#sendMessage({ content: 'word-styles' }); } + rotateShapes() { + this.#sendMessage({ content: 'rotate-selection' }); + } + #sendMessage(message: unknown) { parent.postMessage(message, '*'); } diff --git a/apps/poc-state-plugin/src/plugin.ts b/apps/poc-state-plugin/src/plugin.ts index 7d402eb..db99ae1 100644 --- a/apps/poc-state-plugin/src/plugin.ts +++ b/apps/poc-state-plugin/src/plugin.ts @@ -9,300 +9,33 @@ penpot.ui.onMessage<{ content: string; data: unknown }>((message) => { if (message.content === 'close') { penpot.closePlugin(); } else if (message.content === 'ready') { - const page = penpot.getPage(); - const file = penpot.getFile(); - - if (!page || !file) { - return; - } - - const selection = penpot.selection; - const data: string | null = - selection.length === 1 ? selection[0].getPluginData('counter') : null; - const counter = data ? parseInt(data, 10) : 0; - - penpot.ui.sendMessage({ - type: 'init', - content: { - name: page.name, - pageId: page.id, - fileId: file.id, - revn: file.revn, - theme: penpot.getTheme(), - selection, - counter, - }, - }); + init(); } else if (message.content === 'change-name') { - const shape = penpot - .getPage() - ?.getShapeById('' + (message.data as { id: string }).id); - if (shape) { - shape.name = (message.data as { name: string }).name; - } + changeName(message.data as { id: string; name: string }); } else if (message.content === 'create-rect') { - const shape = penpot.createRectangle(); - const center = penpot.viewport.center; - shape.x = center.x; - shape.y = center.y; + createRect(); } else if (message.content === 'move-x') { - const shape = penpot - .getPage() - ?.getShapeById('' + (message.data as { id: string }).id); - if (shape) { - shape.x += 100; - } + moveX(message.data as { id: string }); } else if (message.content === 'move-y') { - const shape = penpot - .getPage() - ?.getShapeById('' + (message.data as { id: string }).id); - if (shape) { - shape.y += 100; - } + moveY(message.data as { id: string }); } else if (message.content === 'resize-w') { - const shape = penpot - .getPage() - ?.getShapeById('' + (message.data as { id: string }).id); - if (shape) { - shape.resize(shape.width * 2, shape.height); - } + resizeW(message.data as { id: string }); } else if (message.content === 'resize-h') { - const shape = penpot - .getPage() - ?.getShapeById('' + (message.data as { id: string }).id); - if (shape) { - shape.resize(shape.width, shape.height * 2); - } + resizeH(message.data as { id: string }); } else if (message.content === 'lorem-ipsum') { - const selection = penpot.selection; - - for (const shape of selection) { - if (penpot.utils.types.isText(shape)) { - shape.characters = `Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam id mauris ut felis finibus congue. Ut odio ipsum, condimentum id tellus sit amet, dapibus sagittis ligula. Pellentesque hendrerit, nulla sit amet aliquet scelerisque, orci nunc commodo tellus, quis hendrerit nisl massa non tellus. - -Phasellus fringilla tortor elit, ac dictum tellus posuere sodales. Ut eget imperdiet ante. Nunc eros magna, tincidunt non finibus in, tempor elementum nunc. Sed commodo magna in arcu aliquam efficitur.`; - } else if (penpot.utils.types.isRectangle(shape)) { - const width = Math.ceil(shape.width); - const height = Math.ceil(shape.height); - penpot - .uploadMediaUrl( - 'placeholder', - `https://picsum.photos/${width}/${height}` - ) - .then((data) => { - shape.fills = [{ fillOpacity: 1, fillImage: data }]; - }) - .catch((err) => console.error(err)); - } - } + loremIpsum(); } else if (message.content === 'add-icon') { - const iconStr = ` - `; - const shape = penpot.createShapeFromSvg(iconStr); - const center = penpot.viewport.center; - shape.x = center.x; - shape.y = center.y; + addIcon(); } else if (message.content === 'create-grid') { - const frame = penpot.createFrame(); - frame.name = 'Frame Grid'; - - const viewport = penpot.viewport; - frame.x = viewport.center.x - 150; - frame.y = viewport.center.y - 200; - frame.resize(300, 400); - - // create grid - const grid = frame.addGridLayout(); - const [numRows, numCols] = GRID; - - for (let i = 0; i < numRows; i++) { - grid.addRow('auto'); - } - - for (let i = 0; i < numCols; i++) { - grid.addColumn('auto'); - } - - grid.alignItems = 'center'; - grid.justifyItems = 'start'; - grid.justifyContent = 'space-between'; - grid.alignContent = 'stretch'; - grid.rowGap = 1; - grid.columnGap = 2; - grid.verticalPadding = 3; - grid.horizontalPadding = 4; - - // create text - for (let row = 0; row < numRows; row++) { - for (let col = 0; col < numCols; col++) { - const text = penpot.createText(`${row + 1} - ${col + 1}`); - text.growType = 'auto-width'; - grid.appendChild(text, row + 1, col + 1); - } - } + createGrid(); } else if (message.content === 'create-colors') { - const frame = penpot.createFrame(); - frame.name = 'Palette'; - - const viewport = penpot.viewport; - frame.x = viewport.center.x - 150; - frame.y = viewport.center.y - 200; - - const colors = penpot.library.local.colors.sort((a, b) => - a.name.toLowerCase() > b.name.toLowerCase() - ? 1 - : a.name.toLowerCase() < b.name.toLowerCase() - ? -1 - : 0 - ); - - if (colors.length === 0) { - // NO colors return - return; - } - - const cols = 3; - const rows = Math.ceil(colors.length / 3); - - const width = cols * 150 + Math.max(0, cols - 1) * 10 + 20; - const height = rows * 100 + Math.max(0, rows - 1) * 10 + 20; - - frame.resize(width, height); - frame.borderRadius = 8; - - // create grid - const grid = frame.addGridLayout(); - - for (let i = 0; i < rows; i++) { - grid.addRow('auto'); - } - - for (let i = 0; i < cols; i++) { - grid.addColumn('auto'); - } - - grid.alignItems = 'center'; - grid.justifyItems = 'start'; - grid.justifyContent = 'stretch'; - grid.alignContent = 'stretch'; - grid.rowGap = 10; - grid.columnGap = 10; - grid.verticalPadding = 10; - grid.horizontalPadding = 10; - - // These properties are not mandatory, if not defined will apply the default values - frame.shadows = [ - { - style: 'drop-shadow', - offsetX: 5, - offsetY: 5, - blur: 4, - spread: 5, - color: { - color: '#000000', - opacity: 0.3, - }, - }, - ]; - - // create text - for (let row = 0; row < rows; row++) { - for (let col = 0; col < cols; col++) { - const i = row * cols + col; - const color = colors[i]; - - if (i >= colors.length) { - return; - } - - const board = penpot.createFrame(); - grid.appendChild(board, row + 1, col + 1); - board.fills = [color.asFill()]; - board.strokes = [ - { strokeColor: '#000000', strokeOpacity: 0.3, strokeStyle: 'solid' }, - ]; - - if (board.layoutChild) { - board.layoutChild.horizontalSizing = 'fill'; - board.layoutChild.verticalSizing = 'fill'; - } - - const flex = board.addFlexLayout(); - flex.alignItems = 'center'; - flex.justifyContent = 'center'; - - const text = penpot.createText(color.name); - text.growType = 'auto-width'; - board.appendChild(text); - } - } + createColors(); } else if (message.content === 'increase-counter') { - const selection = penpot.selection; - const data: string | null = - selection.length === 1 ? selection[0].getPluginData('counter') : null; - let counter = data ? parseInt(data, 10) : 0; - counter++; - - selection[0].setPluginData('counter', '' + counter); - penpot.ui.sendMessage({ type: 'update-counter', content: { counter } }); + increaseCounter(); } else if (message.content === 'word-styles') { - const selection = penpot.selection; - - if (selection.length >= 1 && penpot.utils.types.isText(selection[0])) { - const shape = selection[0]; - const text = shape.characters; - - const isSplit = (c: string) => !!c.match(/\W/); - - if (text.trim() === '') { - return; - } - - let lastWordStart = 0; - let gettingWord = !isSplit(text[0]); - let wordProcessed = 0; - - for (let i = 1; i < text.length; i++) { - if (gettingWord && isSplit(text[i])) { - if (wordProcessed % 2 === 0) { - const range = shape.getRange(lastWordStart, i); - range.fills = [{ fillColor: '#FF0000', fillOpacity: 1 }]; - range.textTransform = 'uppercase'; - range.fontSize = '20'; - } - - wordProcessed++; - gettingWord = false; - } else if (!gettingWord && !isSplit(text[i])) { - lastWordStart = i; - gettingWord = true; - } - } - } + wordStyles(); + } else if (message.content === 'rotate-selection') { + rotateSelection(); } }); @@ -342,3 +75,323 @@ penpot.on('selectionchange', () => { penpot.on('themechange', (theme) => { penpot.ui.sendMessage({ type: 'theme', content: theme }); }); + +function init() { + const page = penpot.getPage(); + const file = penpot.getFile(); + + if (!page || !file) { + return; + } + + const selection = penpot.selection; + const data: string | null = + selection.length === 1 ? selection[0].getPluginData('counter') : null; + const counter = data ? parseInt(data, 10) : 0; + + penpot.ui.sendMessage({ + type: 'init', + content: { + name: page.name, + pageId: page.id, + fileId: file.id, + revn: file.revn, + theme: penpot.getTheme(), + selection, + counter, + }, + }); +} + +function changeName(data: { id: string; name: string }) { + const shape = penpot.getPage()?.getShapeById('' + data.id); + if (shape) { + shape.name = data.name; + } +} + +function createRect() { + const shape = penpot.createRectangle(); + const center = penpot.viewport.center; + shape.x = center.x; + shape.y = center.y; +} + +function moveX(data: { id: string }) { + const shape = penpot.getPage()?.getShapeById('' + data.id); + if (shape) { + shape.x += 100; + } +} + +function moveY(data: { id: string }) { + const shape = penpot.getPage()?.getShapeById('' + data.id); + if (shape) { + shape.y += 100; + } +} + +function resizeW(data: { id: string }) { + const shape = penpot.getPage()?.getShapeById('' + data.id); + if (shape) { + shape.resize(shape.width * 2, shape.height); + } +} + +function resizeH(data: { id: string }) { + const shape = penpot.getPage()?.getShapeById('' + data.id); + if (shape) { + shape.resize(shape.width, shape.height * 2); + } +} + +function loremIpsum() { + const selection = penpot.selection; + + for (const shape of selection) { + if (penpot.utils.types.isText(shape)) { + shape.characters = `Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam id mauris ut felis finibus congue. Ut odio ipsum, condimentum id tellus sit amet, dapibus sagittis ligula. Pellentesque hendrerit, nulla sit amet aliquet scelerisque, orci nunc commodo tellus, quis hendrerit nisl massa non tellus. + +Phasellus fringilla tortor elit, ac dictum tellus posuere sodales. Ut eget imperdiet ante. Nunc eros magna, tincidunt non finibus in, tempor elementum nunc. Sed commodo magna in arcu aliquam efficitur.`; + } else if (penpot.utils.types.isRectangle(shape)) { + const width = Math.ceil(shape.width); + const height = Math.ceil(shape.height); + penpot + .uploadMediaUrl( + 'placeholder', + `https://picsum.photos/${width}/${height}` + ) + .then((data) => { + shape.fills = [{ fillOpacity: 1, fillImage: data }]; + }) + .catch((err) => console.error(err)); + } + } +} + +function addIcon() { + const iconStr = ` + `; + const shape = penpot.createShapeFromSvg(iconStr); + const center = penpot.viewport.center; + shape.x = center.x; + shape.y = center.y; +} + +function createGrid() { + const frame = penpot.createFrame(); + frame.name = 'Frame Grid'; + + const viewport = penpot.viewport; + frame.x = viewport.center.x - 150; + frame.y = viewport.center.y - 200; + frame.resize(300, 400); + + // create grid + const grid = frame.addGridLayout(); + const [numRows, numCols] = GRID; + + for (let i = 0; i < numRows; i++) { + grid.addRow('auto'); + } + + for (let i = 0; i < numCols; i++) { + grid.addColumn('auto'); + } + + grid.alignItems = 'center'; + grid.justifyItems = 'start'; + grid.justifyContent = 'space-between'; + grid.alignContent = 'stretch'; + grid.rowGap = 1; + grid.columnGap = 2; + grid.verticalPadding = 3; + grid.horizontalPadding = 4; + + // create text + for (let row = 0; row < numRows; row++) { + for (let col = 0; col < numCols; col++) { + const text = penpot.createText(`${row + 1} - ${col + 1}`); + text.growType = 'auto-width'; + grid.appendChild(text, row + 1, col + 1); + } + } +} + +function createColors() { + const frame = penpot.createFrame(); + frame.name = 'Palette'; + + const viewport = penpot.viewport; + frame.x = viewport.center.x - 150; + frame.y = viewport.center.y - 200; + + const colors = penpot.library.local.colors.sort((a, b) => + a.name.toLowerCase() > b.name.toLowerCase() + ? 1 + : a.name.toLowerCase() < b.name.toLowerCase() + ? -1 + : 0 + ); + + if (colors.length === 0) { + // NO colors return + return; + } + + const cols = 3; + const rows = Math.ceil(colors.length / 3); + + const width = cols * 150 + Math.max(0, cols - 1) * 10 + 20; + const height = rows * 100 + Math.max(0, rows - 1) * 10 + 20; + + frame.resize(width, height); + frame.borderRadius = 8; + + // create grid + const grid = frame.addGridLayout(); + + for (let i = 0; i < rows; i++) { + grid.addRow('auto'); + } + + for (let i = 0; i < cols; i++) { + grid.addColumn('auto'); + } + + grid.alignItems = 'center'; + grid.justifyItems = 'start'; + grid.justifyContent = 'stretch'; + grid.alignContent = 'stretch'; + grid.rowGap = 10; + grid.columnGap = 10; + grid.verticalPadding = 10; + grid.horizontalPadding = 10; + + // These properties are not mandatory, if not defined will apply the default values + frame.shadows = [ + { + style: 'drop-shadow', + offsetX: 5, + offsetY: 5, + blur: 4, + spread: 5, + color: { + color: '#000000', + opacity: 0.3, + }, + }, + ]; + + // create text + for (let row = 0; row < rows; row++) { + for (let col = 0; col < cols; col++) { + const i = row * cols + col; + const color = colors[i]; + + if (i >= colors.length) { + return; + } + + const board = penpot.createFrame(); + grid.appendChild(board, row + 1, col + 1); + board.fills = [color.asFill()]; + board.strokes = [ + { strokeColor: '#000000', strokeOpacity: 0.3, strokeStyle: 'solid' }, + ]; + + if (board.layoutChild) { + board.layoutChild.horizontalSizing = 'fill'; + board.layoutChild.verticalSizing = 'fill'; + } + + const flex = board.addFlexLayout(); + flex.alignItems = 'center'; + flex.justifyContent = 'center'; + + const text = penpot.createText(color.name); + text.growType = 'auto-width'; + board.appendChild(text); + } + } +} + +function increaseCounter() { + const selection = penpot.selection; + const data: string | null = + selection.length === 1 ? selection[0].getPluginData('counter') : null; + let counter = data ? parseInt(data, 10) : 0; + counter++; + + selection[0].setPluginData('counter', '' + counter); + penpot.ui.sendMessage({ type: 'update-counter', content: { counter } }); +} + +function wordStyles() { + const selection = penpot.selection; + + if (selection.length >= 1 && penpot.utils.types.isText(selection[0])) { + const shape = selection[0]; + const text = shape.characters; + + const isSplit = (c: string) => !!c.match(/\W/); + + if (text.trim() === '') { + return; + } + + let lastWordStart = 0; + let gettingWord = !isSplit(text[0]); + let wordProcessed = 0; + + for (let i = 1; i < text.length; i++) { + if (gettingWord && isSplit(text[i])) { + if (wordProcessed % 2 === 0) { + const range = shape.getRange(lastWordStart, i); + range.fills = [{ fillColor: '#FF0000', fillOpacity: 1 }]; + range.textTransform = 'uppercase'; + range.fontSize = '20'; + } + + wordProcessed++; + gettingWord = false; + } else if (!gettingWord && !isSplit(text[i])) { + lastWordStart = i; + gettingWord = true; + } + } + } +} + +function rotateSelection() { + const selection = penpot.selection; + const center = penpot.utils.geometry.center(selection); + + selection.forEach((shape) => { + shape.rotate(10, center); + }); +}