git-hash-art 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.release-it.json +17 -0
- package/README.md +138 -0
- package/bin/generateExamples.js +59 -0
- package/dist/main.js +974 -0
- package/dist/main.js.map +1 -0
- package/eslint.config.mjs +29 -0
- package/examples/angular-1024x1024-f31a6c3e.png +0 -0
- package/examples/banner-1920x480-d847ffd4.png +0 -0
- package/examples/complex-2048x2048-deadbeef.png +0 -0
- package/examples/instagram-square-1080x1080-ff00ff00.png +0 -0
- package/examples/instagram-story-1080x1920-abc123de.png +0 -0
- package/examples/linkedin-banner-1584x396-bbbbbbbb.png +0 -0
- package/examples/minimal-1024x1024-00000000.png +0 -0
- package/examples/phone-wallpaper-1170x2532-ffffffff.png +0 -0
- package/examples/react-1024x1024-46192e59.png +0 -0
- package/examples/tablet-wallpaper-2048x2732-12345678.png +0 -0
- package/examples/twitter-header-1500x500-77777777.png +0 -0
- package/examples/ultrawide-3440x1440-a3e126e5.png +0 -0
- package/package.json +32 -0
- package/src/index.js +226 -0
- package/src/lib/canvas/colors.js +102 -0
- package/src/lib/canvas/draw.js +73 -0
- package/src/lib/canvas/shapes/basic.js +93 -0
- package/src/lib/canvas/shapes/complex.js +205 -0
- package/src/lib/canvas/shapes/index.js +9 -0
- package/src/lib/canvas/shapes/sacred.js +173 -0
- package/src/lib/canvas/shapes/utils.js +37 -0
- package/src/lib/constants.js +138 -0
- package/src/lib/utils.js +59 -0
package/dist/main.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"mappings":"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","sources":["src/index.js","src/lib/canvas/colors.js","src/lib/utils.js","src/lib/canvas/shapes/index.js","src/lib/canvas/shapes/basic.js","src/lib/canvas/shapes/complex.js","src/lib/constants.js","src/lib/canvas/shapes/utils.js","src/lib/canvas/shapes/sacred.js","src/lib/canvas/draw.js"],"sourcesContent":["import { createCanvas } from \"canvas\";\nimport fs from \"fs\";\nimport path from \"path\";\nimport { SacredColorScheme } from \"./lib/canvas/colors\";\nimport { enhanceShapeGeneration } from \"./lib/canvas/draw\";\nimport { shapes } from \"./lib/canvas/shapes\";\nimport { getRandomFromHash } from \"./lib/utils\";\n\n/**\n * @typedef {Object} ArtConfig\n * @property {number} width - Canvas width in pixels\n * @property {number} height - Canvas height in pixels\n * @property {number} [gridSize=4] - Number of grid cells (gridSize x gridSize)\n * @property {number} [layers=5] - Number of layers to generate\n * @property {number} [shapesPerLayer] - Base number of shapes per layer (defaults to grid cells * 1.5)\n * @property {number} [minShapeSize=20] - Minimum shape size\n * @property {number} [maxShapeSize=180] - Maximum shape size\n * @property {number} [baseOpacity=0.6] - Starting opacity for first layer\n * @property {number} [opacityReduction=0.1] - How much to reduce opacity per layer\n */\n\n/**\n * Generate an abstract art image from a git hash with custom configuration\n * @param {string} gitHash - The git hash to use as a seed\n * @param {ArtConfig} [config={}] - Configuration options\n * @returns {Buffer} PNG buffer of the generated image\n */\nfunction generateImageFromHash(gitHash, config = {}) {\n // Default configuration\n const defaultConfig = {\n width: 2048,\n height: 2048,\n gridSize: 12,\n layers: 2,\n minShapeSize: 20,\n maxShapeSize: 600,\n baseOpacity: 0.8,\n opacityReduction: 0.4,\n };\n\n // Merge provided config with defaults\n const finalConfig = { ...defaultConfig, ...config };\n const {\n width,\n height,\n gridSize,\n layers,\n minShapeSize,\n maxShapeSize,\n baseOpacity,\n opacityReduction,\n } = finalConfig;\n\n // Calculate shapes per layer based on grid size if not provided\n finalConfig.shapesPerLayer =\n finalConfig.shapesPerLayer || Math.floor(gridSize * gridSize * 1.5);\n\n const canvas = createCanvas(width, height);\n const ctx = canvas.getContext(\"2d\");\n \n const colorScheme = new SacredColorScheme(gitHash);\n const colors = colorScheme.getColorPalette(\"chakra\");\n\n // Create a gradient background\n const gradient = ctx.createLinearGradient(0, 0, width, height);\n gradient.addColorStop(0, colorScheme.baseScheme[0]);\n gradient.addColorStop(1, colorScheme.baseScheme[1]);\n ctx.fillStyle = gradient;\n ctx.fillRect(0, 0, width, height);\n\n const shapeNames = Object.keys(shapes);\n\n const cellWidth = width / gridSize;\n const cellHeight = height / gridSize;\n\n // Scale shape sizes based on canvas dimensions\n const scaleFactor = Math.min(width, height) / 1024;\n const adjustedMinSize = minShapeSize * scaleFactor;\n const adjustedMaxSize = maxShapeSize * scaleFactor;\n\n for (let layer = 0; layer < layers; layer++) {\n const numShapes =\n finalConfig.shapesPerLayer +\n Math.floor(\n getRandomFromHash(gitHash, layer, 0, finalConfig.shapesPerLayer / 2)\n );\n const layerOpacity = baseOpacity - layer * opacityReduction;\n\n for (let i = 0; i < numShapes; i++) {\n const gridX = Math.floor(i / gridSize);\n const gridY = i % gridSize;\n\n const cellOffsetX = getRandomFromHash(\n gitHash,\n layer * numShapes + i * 2,\n 0,\n cellWidth\n );\n const cellOffsetY = getRandomFromHash(\n gitHash,\n layer * numShapes + i * 2 + 1,\n 0,\n cellHeight\n );\n\n const x = gridX * cellWidth + cellOffsetX;\n const y = gridY * cellHeight + cellOffsetY;\n\n const shape =\n shapeNames[\n Math.floor(\n getRandomFromHash(\n gitHash,\n layer * numShapes + i * 3,\n 0,\n shapeNames.length\n )\n )\n ];\n const size =\n adjustedMinSize +\n getRandomFromHash(\n gitHash,\n layer * numShapes + i * 4,\n 0,\n adjustedMaxSize - adjustedMinSize\n );\n const rotation = getRandomFromHash(\n gitHash,\n layer * numShapes + i * 5,\n 0,\n 360\n );\n\n const fillColorIndex = Math.floor(\n getRandomFromHash(gitHash, layer * numShapes + i * 6, 0, colors.length)\n );\n const strokeColorIndex = Math.floor(\n getRandomFromHash(gitHash, layer * numShapes + i * 7, 0, colors.length)\n );\n\n ctx.globalAlpha = layerOpacity;\n // drawShape(\n // ctx,\n // shape,\n // x,\n // y,\n // colors[fillColorIndex],\n // colors[strokeColorIndex],\n // 2 * scaleFactor,\n // size,\n // rotation\n // );\n enhanceShapeGeneration(ctx, shape, x, y, {\n fillColor: colors[fillColorIndex],\n strokeColor: colors[strokeColorIndex],\n strokeWidth: 1.5 * scaleFactor,\n size,\n rotation,\n // Optionally add pattern combinations\n // patterns:\n // Math.random() > 0.7 ? PatternPresets.cosmicTree(size) : [],\n proportionType: \"GOLDEN_RATIO\",\n });\n }\n\n // Add connecting lines scaled to canvas size\n ctx.globalAlpha = 0.2;\n ctx.strokeStyle = colors[colors.length - 1];\n ctx.lineWidth = 1 * scaleFactor;\n\n const numLines = Math.floor((15 * (width * height)) / (1024 * 1024));\n for (let i = 0; i < numLines; i++) {\n const x1 = getRandomFromHash(gitHash, i * 4, 0, width);\n const y1 = getRandomFromHash(gitHash, i * 4 + 1, 0, height);\n const x2 = getRandomFromHash(gitHash, i * 4 + 2, 0, width);\n const y2 = getRandomFromHash(gitHash, i * 4 + 3, 0, height);\n\n ctx.beginPath();\n ctx.moveTo(x1, y1);\n ctx.lineTo(x2, y2);\n ctx.stroke();\n }\n }\n\n return canvas.toBuffer(\"image/png\");\n}\n\n/**\n * Save the generated image to a file\n * @param {Buffer} imageBuffer - The PNG buffer of the generated image\n * @param {string} outputDir - The directory to save the image\n * @param {string} gitHash - The git hash used to generate the image\n * @param {string} [label=''] - Label for the output file\n * @param {number} width - The width of the generated image\n * @param {number} height - The height of the generated image\n * @returns {string} Path to the saved image\n */\nfunction saveImageToFile(imageBuffer, outputDir, gitHash, label = \"\", width, height) {\n if (!fs.existsSync(outputDir)) {\n fs.mkdirSync(outputDir, { recursive: true });\n }\n\n const filename = label\n ? `${label}-${width}x${height}-${gitHash.slice(0, 8)}.png`\n : `${gitHash.slice(0, 8)}-${width}x${height}.png`;\n\n const outputPath = path.join(outputDir, filename);\n fs.writeFileSync(outputPath, imageBuffer);\n console.log(`Generated: ${outputPath}`);\n\n return outputPath;\n}\n\nexport { generateImageFromHash, saveImageToFile };\n\n// Usage example:\n/*\nimport { generateImageFromHash, saveImageToFile } from 'git-hash-art';\n\nconst gitHash = '1234567890abcdef1234567890abcdef12345678';\nconst imageBuffer = generateImageFromHash(gitHash, { width: 1024, height: 1024 });\nconst savedImagePath = saveImageToFile(imageBuffer, './output', gitHash, 'example', 1024, 1024);\nconsole.log(`Image saved to: ${savedImagePath}`);\n*/\n\n","import ColorScheme from \"color-scheme\";\nimport { gitHashToSeed } from \"../utils\";\n\n\n/**\n * Generates a color scheme based on a given Git hash.\n *\n * @param {string} gitHash - The Git hash used to generate the color scheme.\n * @returns {string[]} An array of hex color codes representing the generated color scheme.\n */\nexport function generateColorScheme(gitHash) {\n const seed = gitHashToSeed(gitHash);\n const scheme = new ColorScheme();\n scheme\n .from_hue(seed % 360)\n .scheme(\"analogic\")\n .variation(\"soft\");\n\n let colors = scheme.colors().map((hex) => `#${hex}`);\n\n const contrastingHue = (seed + 180) % 360;\n const contrastingScheme = new ColorScheme();\n contrastingScheme.from_hue(contrastingHue).scheme(\"mono\").variation(\"soft\");\n colors.push(`#${contrastingScheme.colors()[0]}`);\n\n return colors;\n}\n\n// Enhanced color scheme generation for sacred geometry\nexport class SacredColorScheme {\n constructor(gitHash) {\n this.seed = this.gitHashToSeed(gitHash);\n this.baseScheme = this.generateBaseScheme();\n this.complementaryScheme = this.generateComplementaryScheme();\n this.metallic = this.generateMetallicColors();\n }\n\n gitHashToSeed(hash) {\n return parseInt(hash.slice(0, 8), 16);\n }\n\n generateBaseScheme() {\n const scheme = new ColorScheme();\n return scheme\n .from_hue(this.seed % 360)\n .scheme(\"analogic\")\n .variation(\"soft\")\n .colors()\n .map((hex) => `#${hex}`);\n }\n\n generateComplementaryScheme() {\n const complementaryHue = (this.seed + 180) % 360;\n const scheme = new ColorScheme();\n return scheme\n .from_hue(complementaryHue)\n .scheme(\"mono\")\n .variation(\"soft\")\n .colors()\n .map((hex) => `#${hex}`);\n }\n\n generateMetallicColors() {\n return {\n gold: \"#FFD700\",\n silver: \"#C0C0C0\",\n copper: \"#B87333\",\n bronze: \"#CD7F32\",\n };\n }\n\n getColorPalette(type = \"sacred\") {\n switch (type) {\n case \"sacred\":\n return {\n primary: this.baseScheme[0],\n secondary: this.baseScheme[1],\n accent: this.complementaryScheme[0],\n metallic: this.metallic.gold,\n };\n case \"elemental\":\n return {\n earth: this.baseScheme[0],\n water: this.baseScheme[1],\n air: this.baseScheme[2],\n fire: this.complementaryScheme[0],\n };\n case \"chakra\":\n return {\n root: \"#FF0000\",\n sacral: \"#FF7F00\",\n solar: \"#FFFF00\",\n heart: \"#00FF00\",\n throat: \"#0000FF\",\n third_eye: \"#4B0082\",\n crown: \"#8F00FF\",\n };\n default:\n return this.baseScheme;\n }\n }\n}\n","import { shapes } from \"./canvas/shapes\";\n\nexport function gitHashToSeed(gitHash) {\n return parseInt(gitHash.slice(0, 8), 16);\n}\n\nexport function getRandomFromHash(hash, index, min, max) {\n const hexPair = hash.substr((index * 2) % hash.length, 2);\n const decimal = parseInt(hexPair, 16);\n return min + (decimal / 255) * (max - min);\n}\n\n// Golden ratio and other important proportions\nexport const Proportions = {\n GOLDEN_RATIO: 1.618034,\n SQUARE_ROOT_2: Math.sqrt(2),\n SQUARE_ROOT_3: Math.sqrt(3),\n SQUARE_ROOT_5: Math.sqrt(5),\n PI: Math.PI,\n PHI: (1 + Math.sqrt(5)) / 2,\n};\n\n// Pattern combination utilities\nexport class PatternCombiner {\n static getProportionalSize(baseSize, proportion) {\n return baseSize * proportion;\n }\n\n static centerPattern(ctx, width, height) {\n ctx.translate(width / 2, height / 2);\n }\n\n // Combines sacred geometry patterns with proper proportions\n static layerPatterns(ctx, patterns, config) {\n const {\n baseSize,\n baseOpacity = 0.6,\n opacityReduction = 0.1,\n rotationOffset = 0,\n proportionType = \"GOLDEN_RATIO\",\n } = config;\n\n patterns.forEach((pattern, index) => {\n const size = this.getProportionalSize(\n baseSize,\n Math.pow(Proportions[proportionType], index)\n );\n const opacity = Math.max(0.1, baseOpacity - index * opacityReduction);\n const rotation = rotationOffset * index;\n\n ctx.save();\n ctx.globalAlpha = opacity;\n ctx.rotate((rotation * Math.PI) / 180);\n\n shapes[pattern.type](ctx, size, pattern.config);\n ctx.restore();\n });\n }\n}\n","import { basicShapes } from \"./basic\";\nimport { complexShapes } from \"./complex\";\nimport { sacredShapes } from \"./sacred\";\n\nexport const shapes = {\n ...basicShapes,\n ...complexShapes,\n ...sacredShapes,\n};\n","export const drawCircle = (ctx, size) => {\n ctx.beginPath();\n ctx.arc(0, 0, size / 2, 0, Math.PI * 2);\n};\n\nexport const drawSquare = (ctx, size) => {\n ctx.beginPath();\n ctx.rect(-size / 2, -size / 2, size, size);\n};\n\nexport const drawTriangle = (ctx, size) => {\n ctx.beginPath();\n ctx.moveTo(0, -size / 2);\n ctx.lineTo(-size / 2, size / 2);\n ctx.lineTo(size / 2, size / 2);\n ctx.closePath();\n};\n\nexport const drawHexagon = (ctx, size) => {\n ctx.beginPath();\n for (let i = 0; i < 6; i++) {\n const angle = (Math.PI / 8) * i;\n const x = (size / 2) * Math.cos(angle);\n const y = (size / 2) * Math.sin(angle);\n if (i === 0) ctx.moveTo(x, y);\n else ctx.lineTo(x, y);\n }\n ctx.closePath();\n};\n\nexport const drawStar = (ctx, size) => {\n ctx.beginPath();\n for (let i = 0; i < 10; i++) {\n const angle = Math.PI / 5 + (Math.PI / 5) * i * 3;\n const radius = i % 2 === 0 ? size / 2 : size / 4;\n const x = radius * Math.cos(angle);\n const y = radius * Math.sin(angle);\n if (i === 0) ctx.moveTo(x, y);\n else ctx.lineTo(x, y);\n }\n ctx.closePath();\n};\n\nexport const drawJackedStar = (ctx, size) => {\n ctx.beginPath();\n for (let i = 0; i < 10; i++) {\n const angle = Math.PI / 30 + (Math.PI / 30) * i * 8;\n const radius = i % 2 === 0 ? size / 2 : size / 8;\n const x = radius * Math.cos(angle);\n const y = radius * Math.sin(angle);\n if (i === 0) ctx.moveTo(x, y);\n else ctx.lineTo(x, y);\n }\n ctx.closePath();\n};\n\nexport const drawHeart = (ctx, size) => {\n ctx.beginPath();\n ctx.moveTo(0, size / 4);\n ctx.quadraticCurveTo(size / 2, size / 4, 0, -size / 4);\n ctx.quadraticCurveTo(-size / 2, size / 4, 0, size / 4);\n};\n\nexport const drawDiamond = (ctx, size) => {\n ctx.beginPath();\n ctx.moveTo(0, -size / 2);\n ctx.lineTo(size / 2, 0);\n ctx.lineTo(0, size / 2);\n ctx.lineTo(-size / 2, 0);\n ctx.closePath();\n};\n\nexport const drawCube = (ctx, size) => {\n ctx.beginPath();\n ctx.moveTo(-size / 2, -size / 2);\n ctx.lineTo(size / 2, -size / 2);\n ctx.lineTo(size / 2, size / 2);\n ctx.lineTo(-size / 2, size / 2);\n ctx.closePath();\n};\n\n// Optional: Create a shape map for easier lookup\nexport const basicShapes = {\n circle: drawCircle,\n square: drawSquare,\n triangle: drawTriangle,\n hexagon: drawHexagon,\n star: drawStar,\n \"jacked-star\": drawJackedStar,\n heart: drawHeart,\n diamond: drawDiamond,\n cube: drawCube\n};","import { defaultShapeConfig } from \"../../constants\";\nimport { applyTransforms, createCirclePoints, restoreContext } from \"./utils\";\n\nexport const ShapeConfigs = {\n platonic: {\n tetrahedron: { vertices: 4, faces: 4 },\n cube: { vertices: 8, faces: 6 },\n octahedron: { vertices: 6, faces: 8 },\n dodecahedron: { vertices: 20, faces: 12 },\n icosahedron: { vertices: 12, faces: 20 },\n },\n fibonacci: {\n iterations: 13,\n growthFactor: 1.618034, // Golden ratio\n },\n goldenRatio: {\n iterations: 8,\n ratio: 1.618034,\n },\n};\n\nexport const drawPlatonicSolid = (ctx, size, type, config = {}) => {\n const finalConfig = { ...defaultShapeConfig, ...config };\n applyTransforms(ctx, size, finalConfig);\n\n const { \n vertices, \n // faces \n } = ShapeConfigs.platonic[type];\n const radius = size / 2;\n\n // Calculate vertices based on platonic solid type\n const points = createCirclePoints(0, 0, radius, vertices);\n\n ctx.beginPath();\n // Draw edges between vertices\n points.forEach((p1, i) => {\n points.slice(i + 1).forEach((p2) => {\n ctx.moveTo(p1.x, p1.y);\n ctx.lineTo(p2.x, p2.y);\n });\n });\n\n if (finalConfig.fillStyle !== \"transparent\") {\n ctx.fill();\n }\n ctx.stroke();\n restoreContext(ctx);\n};\n\nexport const drawFibonacciSpiral = (ctx, size, config = {}) => {\n const finalConfig = {\n ...defaultShapeConfig,\n ...ShapeConfigs.fibonacci,\n ...config,\n };\n applyTransforms(ctx, size, finalConfig);\n\n let current = 1;\n let previous = 1;\n let scale = size / Math.pow(finalConfig.growthFactor, finalConfig.iterations);\n\n ctx.beginPath();\n for (let i = 0; i < finalConfig.iterations; i++) {\n const radius = scale * current;\n const centerX = radius / 2;\n const centerY = radius / 2;\n\n ctx.arc(centerX, centerY, radius, Math.PI, Math.PI * 1.5);\n\n // Calculate next Fibonacci number\n const next = current + previous;\n previous = current;\n current = next;\n\n // Transform for next iteration\n ctx.translate(radius, 0);\n ctx.rotate(Math.PI / 2);\n }\n\n ctx.stroke();\n restoreContext(ctx);\n};\n\nexport const drawIslamicPattern = (ctx, size, config = {}) => {\n const finalConfig = { ...defaultShapeConfig, ...config };\n applyTransforms(ctx, size, finalConfig);\n\n const gridSize = 8;\n const unit = size / gridSize;\n\n ctx.beginPath();\n // Create base grid\n for (let i = 0; i <= gridSize; i++) {\n for (let j = 0; j <= gridSize; j++) {\n const x = (i - gridSize / 2) * unit;\n const y = (j - gridSize / 2) * unit;\n\n // Draw star pattern at each intersection\n const radius = unit / 2;\n for (let k = 0; k < 8; k++) {\n const angle = (k / 8) * Math.PI * 2;\n const x1 = x + Math.cos(angle) * radius;\n const y1 = y + Math.sin(angle) * radius;\n if (k === 0) {\n ctx.moveTo(x1, y1);\n } else {\n ctx.lineTo(x1, y1);\n }\n }\n ctx.closePath();\n }\n }\n\n if (finalConfig.fillStyle !== \"transparent\") {\n ctx.fill();\n }\n ctx.stroke();\n restoreContext(ctx);\n};\n\nexport const drawCelticKnot = (ctx, size, config = {}) => {\n const finalConfig = { ...defaultShapeConfig, ...config };\n applyTransforms(ctx, size, finalConfig);\n\n const gridSize = 4;\n const unit = size / gridSize;\n\n const drawKnotSegment = (x, y, type) => {\n ctx.beginPath();\n switch (type) {\n case \"over\":\n ctx.moveTo(x, y);\n ctx.bezierCurveTo(\n x + unit / 2,\n y,\n x + unit / 2,\n y + unit,\n x + unit,\n y + unit\n );\n break;\n case \"under\":\n ctx.moveTo(x, y + unit);\n ctx.bezierCurveTo(x + unit / 2, y + unit, x + unit / 2, y, x + unit, y);\n break;\n }\n ctx.stroke();\n };\n\n // Create knot pattern\n for (let i = 0; i < gridSize; i++) {\n for (let j = 0; j < gridSize; j++) {\n const x = (i - gridSize / 2) * unit;\n const y = (j - gridSize / 2) * unit;\n drawKnotSegment(x, y, (i + j) % 2 === 0 ? \"over\" : \"under\");\n }\n }\n\n restoreContext(ctx);\n};\n\nexport const drawMerkaba = (ctx, size, config = {}) => {\n const finalConfig = { ...defaultShapeConfig, ...config };\n applyTransforms(ctx, size, finalConfig);\n\n const radius = size / 2;\n\n // Draw two intersecting tetrahedra\n ctx.beginPath();\n // First tetrahedron\n const points1 = createCirclePoints(0, 0, radius, 3);\n points1.forEach((p1, i) => {\n points1.slice(i + 1).forEach((p2) => {\n ctx.moveTo(p1.x, p1.y);\n ctx.lineTo(p2.x, p2.y);\n });\n });\n\n // Second tetrahedron (rotated)\n ctx.rotate(Math.PI / 6);\n const points2 = createCirclePoints(0, 0, radius, 3);\n points2.forEach((p1, i) => {\n points2.slice(i + 1).forEach((p2) => {\n ctx.moveTo(p1.x, p1.y);\n ctx.lineTo(p2.x, p2.y);\n });\n });\n\n if (finalConfig.fillStyle !== \"transparent\") {\n ctx.fill();\n }\n ctx.stroke();\n restoreContext(ctx);\n};\n\nexport const complexShapes = {\n platonicSolid: (ctx, size, type = \"tetrahedron\", config) =>\n drawPlatonicSolid(ctx, size, type, config),\n fibonacciSpiral: (ctx, size, config) =>\n drawFibonacciSpiral(ctx, size, config),\n islamicPattern: (ctx, size, config) => drawIslamicPattern(ctx, size, config),\n celticKnot: (ctx, size, config) => drawCelticKnot(ctx, size, config),\n merkaba: (ctx, size, config) => drawMerkaba(ctx, size, config),\n};\n","export const PRESETS = {\n // Standard sizes with different hashes\n react: {\n hash: \"46192e59d42f741c761cbea79462a8b3815dd905\",\n width: 1024,\n height: 1024,\n },\n angular: {\n hash: \"f31a6c3e94420f43c0cd323a5a6a99376ee59ff8\",\n width: 1024,\n height: 1024,\n gridSize: 6, // Higher density grid\n },\n // Wide format variations\n banner: {\n hash: \"d847ffd4269b22c54d6e85ad3c1892a298e961fb\",\n width: 1920,\n height: 480,\n gridSize: 8, // More horizontal cells for wide format\n shapesPerLayer: 40,\n },\n ultrawide: {\n hash: \"a3e126e537ed2cd11ddf3a96c37066e97c7afee6\",\n width: 3440,\n height: 1440,\n gridSize: 12, // Extra wide needs more cells\n shapesPerLayer: 60,\n },\n // Social media sizes\n \"instagram-square\": {\n hash: \"ff00ff00ff00ff00ff00ff00ff00ff00ff00ff0\",\n width: 1080,\n height: 1080,\n },\n \"instagram-story\": {\n hash: \"abc123def456abc123def456abc123def456abc1\",\n width: 1080,\n height: 1920,\n gridSize: 6,\n layers: 6,\n },\n \"twitter-header\": {\n hash: \"7777777777777777777777777777777777777777\",\n width: 1500,\n height: 500,\n gridSize: 8,\n shapesPerLayer: 35,\n },\n \"linkedin-banner\": {\n hash: \"bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb\",\n width: 1584,\n height: 396,\n gridSize: 8,\n shapesPerLayer: 35,\n },\n // Mobile sizes\n \"phone-wallpaper\": {\n hash: \"ffffffffffffffffffffffffffffffffaaaaaaaa\",\n width: 1170,\n height: 2532, // iPhone 13 Pro size\n gridSize: 5,\n layers: 6,\n },\n \"tablet-wallpaper\": {\n hash: \"123456789abcdef0123456789abcdef012345678\",\n width: 2048,\n height: 2732, // iPad Pro size\n gridSize: 7,\n layers: 6,\n },\n // Special configurations\n minimal: {\n hash: \"000000000000000000000000000000000fffffff\",\n width: 1024,\n height: 1024,\n layers: 3,\n baseOpacity: 0.8,\n shapesPerLayer: 15,\n },\n complex: {\n hash: \"deadbeefdeadbeefdeadbeefdeadbeefdeadbeef\",\n width: 2048,\n height: 2048,\n gridSize: 8,\n layers: 7,\n shapesPerLayer: 50,\n minShapeSize: 30,\n maxShapeSize: 250,\n },\n};\n\n// Basic configuration that applies to all shapes\nexport const defaultShapeConfig = {\n strokeStyle: \"#000000\",\n fillStyle: \"transparent\",\n lineWidth: 1,\n rotation: 0,\n iterations: 1,\n animate: false, // For future use\n};\n\n// Base configuration types that can be extended per shape\nexport const ShapeConfigTypes = {\n BASIC: \"basic\",\n DETAILED: \"detailed\",\n ANIMATED: \"animated\",\n};\n\n// Golden ratio and other important proportions\nexport const Proportions = {\n GOLDEN_RATIO: 1.618034,\n SQUARE_ROOT_2: Math.sqrt(2),\n SQUARE_ROOT_3: Math.sqrt(3),\n SQUARE_ROOT_5: Math.sqrt(5),\n PI: Math.PI,\n PHI: (1 + Math.sqrt(5)) / 2,\n};\n\n// Helper for creating common sacred geometry combinations\nexport const PatternPresets = {\n flowerOfLifeMandala: (size) => [\n // { type: \"flowerOfLife\", config: { size } },\n { type: \"merkaba\", config: { size: size * 0.8 } },\n { type: \"sriYantra\", config: { size: size * 0.5 } },\n ],\n\n platonicProgression: (size) => [\n { type: \"platonicSolid\", config: { size, type: \"tetrahedron\" } },\n { type: \"platonicSolid\", config: { size: size * 0.8, type: \"cube\" } },\n { type: \"platonicSolid\", config: { size: size * 0.6, type: \"octahedron\" } },\n ],\n\n cosmicTree: (size) => [\n // { type: \"treeOfLife\", config: { size } },\n { type: \"fibonacciSpiral\", config: { size: size * 0.9 } },\n { type: \"metatronsCube\", config: { size: size * 0.7 } },\n ],\n};\n","export const degToRad = (degrees) => (degrees * Math.PI) / 180;\n\nexport const applyTransforms = (ctx, size, config) => {\n ctx.save();\n ctx.translate(0, 0);\n if (config.rotation) {\n ctx.rotate(degToRad(config.rotation));\n }\n ctx.lineWidth = config.lineWidth;\n ctx.strokeStyle = config.strokeStyle;\n ctx.fillStyle = config.fillStyle;\n};\n\nexport const restoreContext = (ctx) => {\n ctx.restore();\n};\n\n// Animation configuration stub for future use\nexport const createAnimationConfig = (type) => ({\n enabled: false,\n duration: 1000,\n easing: \"linear\",\n type,\n // Add more animation-specific properties as needed\n});\n\nexport const createCirclePoints = (cx, cy, radius, segments) => {\n const points = [];\n for (let i = 0; i < segments; i++) {\n const angle = (i / segments) * Math.PI * 2;\n points.push({\n x: cx + Math.cos(angle) * radius,\n y: cy + Math.sin(angle) * radius,\n });\n }\n return points;\n};\n","import { createCirclePoints } from \"./utils\";\n\nexport const drawFlowerOfLife = (ctx, size) => {\n const radius = size / 6;\n const centers = [\n { x: 0, y: 0 },\n { x: radius * Math.sqrt(3), y: 0 },\n { x: radius * Math.sqrt(3) / 2, y: 1.5 * radius },\n { x: -radius * Math.sqrt(3) / 2, y: 1.5 * radius },\n { x: -radius * Math.sqrt(3), y: 0 },\n { x: -radius * Math.sqrt(3) / 2, y: -1.5 * radius },\n { x: radius * Math.sqrt(3) / 2, y: -1.5 * radius }\n ];\n\n ctx.beginPath();\n centers.forEach(center => {\n ctx.moveTo(center.x + radius, center.y);\n ctx.arc(center.x, center.y, radius, 0, Math.PI * 2);\n });\n};\n\nexport const drawTreeOfLife = (ctx, size) => {\n const radius = size / 12;\n const spacing = radius * 2.5;\n \n // Sephirot positions (traditional layout)\n const positions = [\n { x: 0, y: -spacing * 2 }, // Kether\n { x: -spacing, y: -spacing }, // Chokmah\n { x: spacing, y: -spacing }, // Binah\n { x: -spacing, y: 0 }, // Chesed\n { x: spacing, y: 0 }, // Geburah\n { x: 0, y: 0 }, // Tiphereth\n { x: -spacing, y: spacing }, // Netzach\n { x: spacing, y: spacing }, // Hod\n { x: 0, y: spacing * 2 }, // Yesod\n { x: 0, y: spacing * 3 } // Malkuth\n ];\n\n // Draw circles\n ctx.beginPath();\n positions.forEach(pos => {\n ctx.moveTo(pos.x + radius, pos.y);\n ctx.arc(pos.x, pos.y, radius, 0, Math.PI * 2);\n });\n\n // Draw connecting lines\n ctx.moveTo(positions[0].x, positions[0].y);\n positions.forEach((pos, i) => {\n if (i > 0) {\n positions.slice(i + 1).forEach(nextPos => {\n ctx.moveTo(pos.x, pos.y);\n ctx.lineTo(nextPos.x, nextPos.y);\n });\n }\n });\n};\n\nexport const drawMetatronsCube = (ctx, size) => {\n const radius = size / 3;\n \n // Create 13 points - one center and 12 vertices of an icosahedron\n \n // const phi = (1 + Math.sqrt(5)) / 2; // Golden ratio\n \n const vertices = [\n { x: 0, y: 0 }, // Center point\n ...createCirclePoints(0, 0, radius, 6), // Inner hexagon\n ...createCirclePoints(0, 0, radius * 1.5, 6) // Outer hexagon\n ];\n\n ctx.beginPath();\n // Draw all connecting lines\n vertices.forEach((v1, i) => {\n vertices.slice(i + 1).forEach(v2 => {\n ctx.moveTo(v1.x, v1.y);\n ctx.lineTo(v2.x, v2.y);\n });\n });\n};\n\nexport const drawSriYantra = (ctx, size) => {\n const radius = size / 2;\n ctx.beginPath();\n\n // Draw outer triangles\n for (let i = 0; i < 9; i++) {\n const angle = (i / 9) * Math.PI * 2;\n const x1 = Math.cos(angle) * radius;\n const y1 = Math.sin(angle) * radius;\n const x2 = Math.cos(angle + Math.PI / 9) * radius;\n const y2 = Math.sin(angle + Math.PI / 9) * radius;\n \n ctx.moveTo(0, 0);\n ctx.lineTo(x1, y1);\n ctx.lineTo(x2, y2);\n ctx.lineTo(0, 0);\n }\n\n // Draw inner triangles\n const innerRadius = radius * 0.6;\n for (let i = 0; i < 9; i++) {\n const angle = (i / 9) * Math.PI * 2 + Math.PI / 18;\n const x1 = Math.cos(angle) * innerRadius;\n const y1 = Math.sin(angle) * innerRadius;\n const x2 = Math.cos(angle + Math.PI / 9) * innerRadius;\n const y2 = Math.sin(angle + Math.PI / 9) * innerRadius;\n \n ctx.moveTo(0, 0);\n ctx.lineTo(x1, y1);\n ctx.lineTo(x2, y2);\n ctx.lineTo(0, 0);\n }\n};\n\nexport const drawSeedOfLife = (ctx, size) => {\n const radius = size / 4;\n const centers = [\n { x: 0, y: 0 },\n ...createCirclePoints(0, 0, radius * 2, 6)\n ];\n\n ctx.beginPath();\n centers.forEach(center => {\n ctx.moveTo(center.x + radius, center.y);\n ctx.arc(center.x, center.y, radius, 0, Math.PI * 2);\n });\n};\n\nexport const drawVesicaPiscis = (ctx, size) => {\n const radius = size / 3;\n ctx.beginPath();\n \n // Draw two overlapping circles\n ctx.arc(-radius/2, 0, radius, 0, Math.PI * 2);\n ctx.arc(radius/2, 0, radius, 0, Math.PI * 2);\n};\n\nexport const drawTorus = (ctx, size) => {\n const outerRadius = size / 2;\n const innerRadius = size / 4;\n const steps = 36;\n \n ctx.beginPath();\n for (let i = 0; i < steps; i++) {\n const angle1 = (i / steps) * Math.PI * 2;\n // const angle2 = ((i + 1) / steps) * Math.PI * 2;\n \n for (let j = 0; j < steps; j++) {\n const phi1 = (j / steps) * Math.PI * 2;\n const phi2 = ((j + 1) / steps) * Math.PI * 2;\n \n const x1 = (outerRadius + innerRadius * Math.cos(phi1)) * Math.cos(angle1);\n const y1 = (outerRadius + innerRadius * Math.cos(phi1)) * Math.sin(angle1);\n const x2 = (outerRadius + innerRadius * Math.cos(phi2)) * Math.cos(angle1);\n const y2 = (outerRadius + innerRadius * Math.cos(phi2)) * Math.sin(angle1);\n \n ctx.moveTo(x1, y1);\n ctx.lineTo(x2, y2);\n }\n }\n};\n\n// Add to the existing shapes map\nexport const sacredShapes = {\n // flowerOfLife: drawFlowerOfLife,\n // treeOfLife: drawTreeOfLife,\n metatronsCube: drawMetatronsCube,\n sriYantra: drawSriYantra,\n seedOfLife: drawSeedOfLife,\n vesicaPiscis: drawVesicaPiscis,\n torus: drawTorus\n};","import { PatternCombiner } from \"../utils\";\nimport { shapes } from \"./shapes\";\n\nexport function drawShape(\n ctx,\n shape,\n x,\n y,\n fillColor,\n strokeColor,\n strokeWidth,\n size,\n rotation\n) {\n ctx.save();\n ctx.translate(x, y);\n ctx.rotate((rotation * Math.PI) / 180);\n ctx.fillStyle = fillColor;\n ctx.strokeStyle = strokeColor;\n ctx.lineWidth = strokeWidth;\n\n const drawFunction = shapes[shape];\n if (drawFunction) {\n drawFunction(ctx, size);\n }\n\n ctx.fill();\n ctx.stroke();\n ctx.restore();\n}\n\n// Integration with existing generation logic\nexport function enhanceShapeGeneration(ctx, shape, x, y, config) {\n const {\n fillColor,\n strokeColor,\n strokeWidth,\n size,\n rotation,\n patterns = [],\n proportionType = \"GOLDEN_RATIO\",\n baseOpacity = 0.6,\n opacityReduction = 0.1,\n } = config;\n\n ctx.save();\n ctx.translate(x, y);\n ctx.rotate((rotation * Math.PI) / 180);\n\n // Draw base shape\n ctx.fillStyle = fillColor;\n ctx.strokeStyle = strokeColor;\n ctx.lineWidth = strokeWidth;\n\n const drawFunction = shapes[shape];\n if (drawFunction) {\n drawFunction(ctx, size);\n }\n\n // Layer additional patterns if specified\n if (patterns.length > 0) {\n PatternCombiner.layerPatterns(ctx, patterns, {\n baseSize: size,\n baseOpacity,\n opacityReduction,\n proportionType,\n });\n }\n\n ctx.fill();\n ctx.stroke();\n ctx.restore();\n}\n"],"names":[],"version":3,"file":"main.js.map"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { FlatCompat } from "@eslint/eslintrc";
|
|
2
|
+
import js from "@eslint/js";
|
|
3
|
+
import globals from "globals";
|
|
4
|
+
import path from "node:path";
|
|
5
|
+
import { fileURLToPath } from "node:url";
|
|
6
|
+
|
|
7
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
8
|
+
const __dirname = path.dirname(__filename);
|
|
9
|
+
const compat = new FlatCompat({
|
|
10
|
+
baseDirectory: __dirname,
|
|
11
|
+
recommendedConfig: js.configs.recommended,
|
|
12
|
+
allConfig: js.configs.all,
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
export default [
|
|
16
|
+
...compat.extends("eslint:recommended"),
|
|
17
|
+
{
|
|
18
|
+
languageOptions: {
|
|
19
|
+
globals: {
|
|
20
|
+
...globals.browser,
|
|
21
|
+
...globals.node,
|
|
22
|
+
},
|
|
23
|
+
|
|
24
|
+
ecmaVersion: 12,
|
|
25
|
+
sourceType: "module",
|
|
26
|
+
},
|
|
27
|
+
rules: {},
|
|
28
|
+
},
|
|
29
|
+
];
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
package/package.json
ADDED
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "git-hash-art",
|
|
3
|
+
"version": "0.0.1",
|
|
4
|
+
"author": "gfargo <ghfargo@gmail.com>",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"scripts": {
|
|
7
|
+
"watch": "parcel watch",
|
|
8
|
+
"build": "parcel build",
|
|
9
|
+
"build:examples": "node bin/generateExamples.js",
|
|
10
|
+
"lint": "eslint src",
|
|
11
|
+
"release": "release-it",
|
|
12
|
+
"test:publish": "yarn lint && npm publish --dry-run",
|
|
13
|
+
"prepublishOnly": "yarn build"
|
|
14
|
+
},
|
|
15
|
+
"source": "src/index.js",
|
|
16
|
+
"main": "dist/main.js",
|
|
17
|
+
"module": "dist/module.js",
|
|
18
|
+
"license": "MIT",
|
|
19
|
+
"dependencies": {
|
|
20
|
+
"@eslint/eslintrc": "^3.1.0",
|
|
21
|
+
"@eslint/js": "^9.13.0",
|
|
22
|
+
"canvas": "^2.11.2",
|
|
23
|
+
"color-scheme": "^1.0.1",
|
|
24
|
+
"release-it": "^17.10.0"
|
|
25
|
+
},
|
|
26
|
+
"devDependencies": {
|
|
27
|
+
"eslint": "^9.13.0",
|
|
28
|
+
"git-coco": "^0.14.1",
|
|
29
|
+
"parcel": "^2.12.0"
|
|
30
|
+
},
|
|
31
|
+
"packageManager": "yarn@1.22.22+sha512.a6b2f7906b721bba3d67d4aff083df04dad64c399707841b7acf00f6b133b7ac24255f2652fa22ae3534329dc6180534e98d17432037ff6fd140556e2bb3137e"
|
|
32
|
+
}
|
package/src/index.js
ADDED
|
@@ -0,0 +1,226 @@
|
|
|
1
|
+
import { createCanvas } from "canvas";
|
|
2
|
+
import fs from "fs";
|
|
3
|
+
import path from "path";
|
|
4
|
+
import { SacredColorScheme } from "./lib/canvas/colors";
|
|
5
|
+
import { enhanceShapeGeneration } from "./lib/canvas/draw";
|
|
6
|
+
import { shapes } from "./lib/canvas/shapes";
|
|
7
|
+
import { getRandomFromHash } from "./lib/utils";
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* @typedef {Object} ArtConfig
|
|
11
|
+
* @property {number} width - Canvas width in pixels
|
|
12
|
+
* @property {number} height - Canvas height in pixels
|
|
13
|
+
* @property {number} [gridSize=4] - Number of grid cells (gridSize x gridSize)
|
|
14
|
+
* @property {number} [layers=5] - Number of layers to generate
|
|
15
|
+
* @property {number} [shapesPerLayer] - Base number of shapes per layer (defaults to grid cells * 1.5)
|
|
16
|
+
* @property {number} [minShapeSize=20] - Minimum shape size
|
|
17
|
+
* @property {number} [maxShapeSize=180] - Maximum shape size
|
|
18
|
+
* @property {number} [baseOpacity=0.6] - Starting opacity for first layer
|
|
19
|
+
* @property {number} [opacityReduction=0.1] - How much to reduce opacity per layer
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Generate an abstract art image from a git hash with custom configuration
|
|
24
|
+
* @param {string} gitHash - The git hash to use as a seed
|
|
25
|
+
* @param {ArtConfig} [config={}] - Configuration options
|
|
26
|
+
* @returns {Buffer} PNG buffer of the generated image
|
|
27
|
+
*/
|
|
28
|
+
function generateImageFromHash(gitHash, config = {}) {
|
|
29
|
+
// Default configuration
|
|
30
|
+
const defaultConfig = {
|
|
31
|
+
width: 2048,
|
|
32
|
+
height: 2048,
|
|
33
|
+
gridSize: 12,
|
|
34
|
+
layers: 2,
|
|
35
|
+
minShapeSize: 20,
|
|
36
|
+
maxShapeSize: 600,
|
|
37
|
+
baseOpacity: 0.8,
|
|
38
|
+
opacityReduction: 0.4,
|
|
39
|
+
};
|
|
40
|
+
|
|
41
|
+
// Merge provided config with defaults
|
|
42
|
+
const finalConfig = { ...defaultConfig, ...config };
|
|
43
|
+
const {
|
|
44
|
+
width,
|
|
45
|
+
height,
|
|
46
|
+
gridSize,
|
|
47
|
+
layers,
|
|
48
|
+
minShapeSize,
|
|
49
|
+
maxShapeSize,
|
|
50
|
+
baseOpacity,
|
|
51
|
+
opacityReduction,
|
|
52
|
+
} = finalConfig;
|
|
53
|
+
|
|
54
|
+
// Calculate shapes per layer based on grid size if not provided
|
|
55
|
+
finalConfig.shapesPerLayer =
|
|
56
|
+
finalConfig.shapesPerLayer || Math.floor(gridSize * gridSize * 1.5);
|
|
57
|
+
|
|
58
|
+
const canvas = createCanvas(width, height);
|
|
59
|
+
const ctx = canvas.getContext("2d");
|
|
60
|
+
|
|
61
|
+
const colorScheme = new SacredColorScheme(gitHash);
|
|
62
|
+
const colors = colorScheme.getColorPalette("chakra");
|
|
63
|
+
|
|
64
|
+
// Create a gradient background
|
|
65
|
+
const gradient = ctx.createLinearGradient(0, 0, width, height);
|
|
66
|
+
gradient.addColorStop(0, colorScheme.baseScheme[0]);
|
|
67
|
+
gradient.addColorStop(1, colorScheme.baseScheme[1]);
|
|
68
|
+
ctx.fillStyle = gradient;
|
|
69
|
+
ctx.fillRect(0, 0, width, height);
|
|
70
|
+
|
|
71
|
+
const shapeNames = Object.keys(shapes);
|
|
72
|
+
|
|
73
|
+
const cellWidth = width / gridSize;
|
|
74
|
+
const cellHeight = height / gridSize;
|
|
75
|
+
|
|
76
|
+
// Scale shape sizes based on canvas dimensions
|
|
77
|
+
const scaleFactor = Math.min(width, height) / 1024;
|
|
78
|
+
const adjustedMinSize = minShapeSize * scaleFactor;
|
|
79
|
+
const adjustedMaxSize = maxShapeSize * scaleFactor;
|
|
80
|
+
|
|
81
|
+
for (let layer = 0; layer < layers; layer++) {
|
|
82
|
+
const numShapes =
|
|
83
|
+
finalConfig.shapesPerLayer +
|
|
84
|
+
Math.floor(
|
|
85
|
+
getRandomFromHash(gitHash, layer, 0, finalConfig.shapesPerLayer / 2)
|
|
86
|
+
);
|
|
87
|
+
const layerOpacity = baseOpacity - layer * opacityReduction;
|
|
88
|
+
|
|
89
|
+
for (let i = 0; i < numShapes; i++) {
|
|
90
|
+
const gridX = Math.floor(i / gridSize);
|
|
91
|
+
const gridY = i % gridSize;
|
|
92
|
+
|
|
93
|
+
const cellOffsetX = getRandomFromHash(
|
|
94
|
+
gitHash,
|
|
95
|
+
layer * numShapes + i * 2,
|
|
96
|
+
0,
|
|
97
|
+
cellWidth
|
|
98
|
+
);
|
|
99
|
+
const cellOffsetY = getRandomFromHash(
|
|
100
|
+
gitHash,
|
|
101
|
+
layer * numShapes + i * 2 + 1,
|
|
102
|
+
0,
|
|
103
|
+
cellHeight
|
|
104
|
+
);
|
|
105
|
+
|
|
106
|
+
const x = gridX * cellWidth + cellOffsetX;
|
|
107
|
+
const y = gridY * cellHeight + cellOffsetY;
|
|
108
|
+
|
|
109
|
+
const shape =
|
|
110
|
+
shapeNames[
|
|
111
|
+
Math.floor(
|
|
112
|
+
getRandomFromHash(
|
|
113
|
+
gitHash,
|
|
114
|
+
layer * numShapes + i * 3,
|
|
115
|
+
0,
|
|
116
|
+
shapeNames.length
|
|
117
|
+
)
|
|
118
|
+
)
|
|
119
|
+
];
|
|
120
|
+
const size =
|
|
121
|
+
adjustedMinSize +
|
|
122
|
+
getRandomFromHash(
|
|
123
|
+
gitHash,
|
|
124
|
+
layer * numShapes + i * 4,
|
|
125
|
+
0,
|
|
126
|
+
adjustedMaxSize - adjustedMinSize
|
|
127
|
+
);
|
|
128
|
+
const rotation = getRandomFromHash(
|
|
129
|
+
gitHash,
|
|
130
|
+
layer * numShapes + i * 5,
|
|
131
|
+
0,
|
|
132
|
+
360
|
|
133
|
+
);
|
|
134
|
+
|
|
135
|
+
const fillColorIndex = Math.floor(
|
|
136
|
+
getRandomFromHash(gitHash, layer * numShapes + i * 6, 0, colors.length)
|
|
137
|
+
);
|
|
138
|
+
const strokeColorIndex = Math.floor(
|
|
139
|
+
getRandomFromHash(gitHash, layer * numShapes + i * 7, 0, colors.length)
|
|
140
|
+
);
|
|
141
|
+
|
|
142
|
+
ctx.globalAlpha = layerOpacity;
|
|
143
|
+
// drawShape(
|
|
144
|
+
// ctx,
|
|
145
|
+
// shape,
|
|
146
|
+
// x,
|
|
147
|
+
// y,
|
|
148
|
+
// colors[fillColorIndex],
|
|
149
|
+
// colors[strokeColorIndex],
|
|
150
|
+
// 2 * scaleFactor,
|
|
151
|
+
// size,
|
|
152
|
+
// rotation
|
|
153
|
+
// );
|
|
154
|
+
enhanceShapeGeneration(ctx, shape, x, y, {
|
|
155
|
+
fillColor: colors[fillColorIndex],
|
|
156
|
+
strokeColor: colors[strokeColorIndex],
|
|
157
|
+
strokeWidth: 1.5 * scaleFactor,
|
|
158
|
+
size,
|
|
159
|
+
rotation,
|
|
160
|
+
// Optionally add pattern combinations
|
|
161
|
+
// patterns:
|
|
162
|
+
// Math.random() > 0.7 ? PatternPresets.cosmicTree(size) : [],
|
|
163
|
+
proportionType: "GOLDEN_RATIO",
|
|
164
|
+
});
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
// Add connecting lines scaled to canvas size
|
|
168
|
+
ctx.globalAlpha = 0.2;
|
|
169
|
+
ctx.strokeStyle = colors[colors.length - 1];
|
|
170
|
+
ctx.lineWidth = 1 * scaleFactor;
|
|
171
|
+
|
|
172
|
+
const numLines = Math.floor((15 * (width * height)) / (1024 * 1024));
|
|
173
|
+
for (let i = 0; i < numLines; i++) {
|
|
174
|
+
const x1 = getRandomFromHash(gitHash, i * 4, 0, width);
|
|
175
|
+
const y1 = getRandomFromHash(gitHash, i * 4 + 1, 0, height);
|
|
176
|
+
const x2 = getRandomFromHash(gitHash, i * 4 + 2, 0, width);
|
|
177
|
+
const y2 = getRandomFromHash(gitHash, i * 4 + 3, 0, height);
|
|
178
|
+
|
|
179
|
+
ctx.beginPath();
|
|
180
|
+
ctx.moveTo(x1, y1);
|
|
181
|
+
ctx.lineTo(x2, y2);
|
|
182
|
+
ctx.stroke();
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
return canvas.toBuffer("image/png");
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
/**
|
|
190
|
+
* Save the generated image to a file
|
|
191
|
+
* @param {Buffer} imageBuffer - The PNG buffer of the generated image
|
|
192
|
+
* @param {string} outputDir - The directory to save the image
|
|
193
|
+
* @param {string} gitHash - The git hash used to generate the image
|
|
194
|
+
* @param {string} [label=''] - Label for the output file
|
|
195
|
+
* @param {number} width - The width of the generated image
|
|
196
|
+
* @param {number} height - The height of the generated image
|
|
197
|
+
* @returns {string} Path to the saved image
|
|
198
|
+
*/
|
|
199
|
+
function saveImageToFile(imageBuffer, outputDir, gitHash, label = "", width, height) {
|
|
200
|
+
if (!fs.existsSync(outputDir)) {
|
|
201
|
+
fs.mkdirSync(outputDir, { recursive: true });
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
const filename = label
|
|
205
|
+
? `${label}-${width}x${height}-${gitHash.slice(0, 8)}.png`
|
|
206
|
+
: `${gitHash.slice(0, 8)}-${width}x${height}.png`;
|
|
207
|
+
|
|
208
|
+
const outputPath = path.join(outputDir, filename);
|
|
209
|
+
fs.writeFileSync(outputPath, imageBuffer);
|
|
210
|
+
console.log(`Generated: ${outputPath}`);
|
|
211
|
+
|
|
212
|
+
return outputPath;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
export { generateImageFromHash, saveImageToFile };
|
|
216
|
+
|
|
217
|
+
// Usage example:
|
|
218
|
+
/*
|
|
219
|
+
import { generateImageFromHash, saveImageToFile } from 'git-hash-art';
|
|
220
|
+
|
|
221
|
+
const gitHash = '1234567890abcdef1234567890abcdef12345678';
|
|
222
|
+
const imageBuffer = generateImageFromHash(gitHash, { width: 1024, height: 1024 });
|
|
223
|
+
const savedImagePath = saveImageToFile(imageBuffer, './output', gitHash, 'example', 1024, 1024);
|
|
224
|
+
console.log(`Image saved to: ${savedImagePath}`);
|
|
225
|
+
*/
|
|
226
|
+
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import ColorScheme from "color-scheme";
|
|
2
|
+
import { gitHashToSeed } from "../utils";
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Generates a color scheme based on a given Git hash.
|
|
7
|
+
*
|
|
8
|
+
* @param {string} gitHash - The Git hash used to generate the color scheme.
|
|
9
|
+
* @returns {string[]} An array of hex color codes representing the generated color scheme.
|
|
10
|
+
*/
|
|
11
|
+
export function generateColorScheme(gitHash) {
|
|
12
|
+
const seed = gitHashToSeed(gitHash);
|
|
13
|
+
const scheme = new ColorScheme();
|
|
14
|
+
scheme
|
|
15
|
+
.from_hue(seed % 360)
|
|
16
|
+
.scheme("analogic")
|
|
17
|
+
.variation("soft");
|
|
18
|
+
|
|
19
|
+
let colors = scheme.colors().map((hex) => `#${hex}`);
|
|
20
|
+
|
|
21
|
+
const contrastingHue = (seed + 180) % 360;
|
|
22
|
+
const contrastingScheme = new ColorScheme();
|
|
23
|
+
contrastingScheme.from_hue(contrastingHue).scheme("mono").variation("soft");
|
|
24
|
+
colors.push(`#${contrastingScheme.colors()[0]}`);
|
|
25
|
+
|
|
26
|
+
return colors;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
// Enhanced color scheme generation for sacred geometry
|
|
30
|
+
export class SacredColorScheme {
|
|
31
|
+
constructor(gitHash) {
|
|
32
|
+
this.seed = this.gitHashToSeed(gitHash);
|
|
33
|
+
this.baseScheme = this.generateBaseScheme();
|
|
34
|
+
this.complementaryScheme = this.generateComplementaryScheme();
|
|
35
|
+
this.metallic = this.generateMetallicColors();
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
gitHashToSeed(hash) {
|
|
39
|
+
return parseInt(hash.slice(0, 8), 16);
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
generateBaseScheme() {
|
|
43
|
+
const scheme = new ColorScheme();
|
|
44
|
+
return scheme
|
|
45
|
+
.from_hue(this.seed % 360)
|
|
46
|
+
.scheme("analogic")
|
|
47
|
+
.variation("soft")
|
|
48
|
+
.colors()
|
|
49
|
+
.map((hex) => `#${hex}`);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
generateComplementaryScheme() {
|
|
53
|
+
const complementaryHue = (this.seed + 180) % 360;
|
|
54
|
+
const scheme = new ColorScheme();
|
|
55
|
+
return scheme
|
|
56
|
+
.from_hue(complementaryHue)
|
|
57
|
+
.scheme("mono")
|
|
58
|
+
.variation("soft")
|
|
59
|
+
.colors()
|
|
60
|
+
.map((hex) => `#${hex}`);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
generateMetallicColors() {
|
|
64
|
+
return {
|
|
65
|
+
gold: "#FFD700",
|
|
66
|
+
silver: "#C0C0C0",
|
|
67
|
+
copper: "#B87333",
|
|
68
|
+
bronze: "#CD7F32",
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
getColorPalette(type = "sacred") {
|
|
73
|
+
switch (type) {
|
|
74
|
+
case "sacred":
|
|
75
|
+
return {
|
|
76
|
+
primary: this.baseScheme[0],
|
|
77
|
+
secondary: this.baseScheme[1],
|
|
78
|
+
accent: this.complementaryScheme[0],
|
|
79
|
+
metallic: this.metallic.gold,
|
|
80
|
+
};
|
|
81
|
+
case "elemental":
|
|
82
|
+
return {
|
|
83
|
+
earth: this.baseScheme[0],
|
|
84
|
+
water: this.baseScheme[1],
|
|
85
|
+
air: this.baseScheme[2],
|
|
86
|
+
fire: this.complementaryScheme[0],
|
|
87
|
+
};
|
|
88
|
+
case "chakra":
|
|
89
|
+
return {
|
|
90
|
+
root: "#FF0000",
|
|
91
|
+
sacral: "#FF7F00",
|
|
92
|
+
solar: "#FFFF00",
|
|
93
|
+
heart: "#00FF00",
|
|
94
|
+
throat: "#0000FF",
|
|
95
|
+
third_eye: "#4B0082",
|
|
96
|
+
crown: "#8F00FF",
|
|
97
|
+
};
|
|
98
|
+
default:
|
|
99
|
+
return this.baseScheme;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import { PatternCombiner } from "../utils";
|
|
2
|
+
import { shapes } from "./shapes";
|
|
3
|
+
|
|
4
|
+
export function drawShape(
|
|
5
|
+
ctx,
|
|
6
|
+
shape,
|
|
7
|
+
x,
|
|
8
|
+
y,
|
|
9
|
+
fillColor,
|
|
10
|
+
strokeColor,
|
|
11
|
+
strokeWidth,
|
|
12
|
+
size,
|
|
13
|
+
rotation
|
|
14
|
+
) {
|
|
15
|
+
ctx.save();
|
|
16
|
+
ctx.translate(x, y);
|
|
17
|
+
ctx.rotate((rotation * Math.PI) / 180);
|
|
18
|
+
ctx.fillStyle = fillColor;
|
|
19
|
+
ctx.strokeStyle = strokeColor;
|
|
20
|
+
ctx.lineWidth = strokeWidth;
|
|
21
|
+
|
|
22
|
+
const drawFunction = shapes[shape];
|
|
23
|
+
if (drawFunction) {
|
|
24
|
+
drawFunction(ctx, size);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
ctx.fill();
|
|
28
|
+
ctx.stroke();
|
|
29
|
+
ctx.restore();
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// Integration with existing generation logic
|
|
33
|
+
export function enhanceShapeGeneration(ctx, shape, x, y, config) {
|
|
34
|
+
const {
|
|
35
|
+
fillColor,
|
|
36
|
+
strokeColor,
|
|
37
|
+
strokeWidth,
|
|
38
|
+
size,
|
|
39
|
+
rotation,
|
|
40
|
+
patterns = [],
|
|
41
|
+
proportionType = "GOLDEN_RATIO",
|
|
42
|
+
baseOpacity = 0.6,
|
|
43
|
+
opacityReduction = 0.1,
|
|
44
|
+
} = config;
|
|
45
|
+
|
|
46
|
+
ctx.save();
|
|
47
|
+
ctx.translate(x, y);
|
|
48
|
+
ctx.rotate((rotation * Math.PI) / 180);
|
|
49
|
+
|
|
50
|
+
// Draw base shape
|
|
51
|
+
ctx.fillStyle = fillColor;
|
|
52
|
+
ctx.strokeStyle = strokeColor;
|
|
53
|
+
ctx.lineWidth = strokeWidth;
|
|
54
|
+
|
|
55
|
+
const drawFunction = shapes[shape];
|
|
56
|
+
if (drawFunction) {
|
|
57
|
+
drawFunction(ctx, size);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Layer additional patterns if specified
|
|
61
|
+
if (patterns.length > 0) {
|
|
62
|
+
PatternCombiner.layerPatterns(ctx, patterns, {
|
|
63
|
+
baseSize: size,
|
|
64
|
+
baseOpacity,
|
|
65
|
+
opacityReduction,
|
|
66
|
+
proportionType,
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
ctx.fill();
|
|
71
|
+
ctx.stroke();
|
|
72
|
+
ctx.restore();
|
|
73
|
+
}
|