├── .gitignore ├── .prettierrc ├── masks ├── squircle.svg ├── squircle@2x.svg └── squircle@3x.svg ├── .editorconfig ├── .eslintrc ├── .github └── workflows │ └── test.yml ├── test.js ├── LICENSE ├── README.md ├── package.json ├── index.ts ├── CHANGELOG.md └── index.js /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | npm-debug.log 4 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "singleQuote": true, 3 | "tabWidth": 2, 4 | "printWidth": 80, 5 | "arrowParens": "avoid" 6 | } 7 | -------------------------------------------------------------------------------- /masks/squircle.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /masks/squircle@2x.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /masks/squircle@3x.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [] 4 | indent_style = tab 5 | indent_size = 2 6 | end_of_line = lf 7 | charset = utf-8 8 | trim_trailing_whitespace = true 9 | insert_final_newline = false 10 | -------------------------------------------------------------------------------- /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "jest": true 4 | }, 5 | "extends": ["airbnb-base", "prettier"], 6 | "plugins": ["prettier"], 7 | "rules": { 8 | "import/no-extraneous-dependencies": "off", 9 | "prettier/prettier": ["error"], 10 | "func-names": "off" 11 | } 12 | } 13 | -------------------------------------------------------------------------------- /.github/workflows/test.yml: -------------------------------------------------------------------------------- 1 | # This workflow will do a clean install of node dependencies, build the source code and run tests across different versions of node 2 | # For more information see: https://help.github.com/actions/language-and-framework-guides/using-nodejs-with-github-actions 3 | 4 | name: Tests 5 | 6 | on: 7 | push: 8 | branches: [main] 9 | pull_request: 10 | branches: [main] 11 | 12 | jobs: 13 | build: 14 | runs-on: ubuntu-latest 15 | 16 | strategy: 17 | matrix: 18 | node-version: [20.x] 19 | 20 | steps: 21 | - uses: actions/checkout@v4 22 | - name: Use Node.js ${{ matrix.node-version }} 23 | uses: actions/setup-node@v4 24 | with: 25 | node-version: ${{ matrix.node-version }} 26 | - run: npm ci 27 | - run: npm test 28 | -------------------------------------------------------------------------------- /test.js: -------------------------------------------------------------------------------- 1 | const plugin = require('./index.ts'); 2 | 3 | const svgMask1x = `data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='150' height='150' viewBox='0 0 150 150'%3E%3Crect width='150' height='150' rx='50' fill='black'/%3E%3C/svg%3E`; 4 | 5 | describe('Smooth Corners Plugin', () => { 6 | it('should generate basic utilities with default sizes', () => { 7 | const utilities = {}; 8 | const addUtilities = jest.fn(utils => { 9 | Object.assign(utilities, utils); 10 | }); 11 | 12 | plugin.handler({ addUtilities, theme: () => ({}) }); 13 | 14 | // Test presence of utility classes 15 | expect(Object.keys(utilities)).toContain('.smooth-corners-sm'); 16 | expect(Object.keys(utilities)).toContain('.smooth-corners-md'); 17 | expect(Object.keys(utilities)).toContain('.smooth-corners-lg'); 18 | 19 | // Test mask-border properties 20 | expect( 21 | utilities['.smooth-corners-sm']['@supports (mask-border-width: 60px)'][ 22 | 'mask-border' 23 | ] 24 | ).toBe(`url("${svgMask1x}") 49% fill / 25px`); 25 | }); 26 | }); 27 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2021 Gary Tokman 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 |
2 | 3 | npm downloads 4 | 5 | 8 | discord users online 9 | 10 |
11 | 12 | # Corner Smoothing 13 | 14 | Frame 1 15 | 16 | > Adjusts a rounded corner to create a continuous curve. Inspired by [Rob](https://github.com/robb/Continuous-Corners-CSS). 17 | 18 | ## Installation 19 | 20 | Install the plugin from npm: 21 | 22 | ```sh 23 | npm install tailwind-corner-smoothing --save-dev 24 | ``` 25 | 26 | # Usage 27 | v4 28 | 29 | ```js 30 | // Global.css / App.css 31 | 32 | @plugin "tailwind-corner-smoothing"; 33 | 34 | ``` 35 | 36 | 37 | v3 38 | ```js 39 | // tailwind.config.js 40 | module.exports = { 41 | plugins: [require('tailwind-corner-smoothing')], 42 | }; 43 | ``` 44 | 45 | # Example 46 | 47 | ```html 48 |
Smooth corners!
49 | ``` 50 | 51 | # License 52 | 53 | MIT 54 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "tailwind-corner-smoothing", 3 | "version": "1.2.0", 4 | "description": "Adjusts a rounded corner to create a continuous curve.", 5 | "main": "index.js", 6 | "files": [ 7 | "index.js" 8 | ], 9 | "scripts": { 10 | "lint": "eslint .", 11 | "lint:fix": "eslint . --fix", 12 | "format": "prettier --write --loglevel=error . !CHANGELOG.md", 13 | "test": "jest", 14 | "test:watch": "jest --watch", 15 | "release": "npx tsc index.ts && standard-version", 16 | "postrelease": "git push --follow-tags origin main && npm publish" 17 | }, 18 | "repository": { 19 | "type": "git", 20 | "url": "git+https://github.com/gtokman/corner-smoothing.git" 21 | }, 22 | "bugs": { 23 | "url": "https://github.com/gtokman/corner-smoothing/issues" 24 | }, 25 | "homepage": "https://github.com/gtokman/corner-smoothing#readme", 26 | "keywords": [ 27 | "create-tailwind-plugin", 28 | "tailwind-css-plugin", 29 | "tailwindcss", 30 | "plugin" 31 | ], 32 | "author": "Gary Tokman", 33 | "license": "MIT", 34 | "devDependencies": { 35 | "eslint": "^7.2.0", 36 | "eslint-config-airbnb-base": "^14.2.1", 37 | "eslint-config-prettier": "^6.15.0", 38 | "eslint-plugin-import": "^2.22.1", 39 | "eslint-plugin-jest": "^24.1.3", 40 | "eslint-plugin-prettier": "^3.1.4", 41 | "husky": "^4.3.0", 42 | "jest": "^26.6.3", 43 | "jest-matcher-css": "^1.1.0", 44 | "lint-staged": "^10.5.2", 45 | "lodash": "^4.17.20", 46 | "postcss": "^8.1.9", 47 | "prettier": "^2.2.0", 48 | "standard-version": "^9.5.0", 49 | "tailwindcss": "latest", 50 | "fs": "^0.0.1-security" 51 | } 52 | } 53 | -------------------------------------------------------------------------------- /index.ts: -------------------------------------------------------------------------------- 1 | const plugin = require('tailwindcss/plugin'); 2 | 3 | // Inline SVGs as Base64 encoded strings 4 | const svgMask1x = `data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='150' height='150' viewBox='0 0 150 150'%3E%3Crect width='150' height='150' rx='50' fill='black'/%3E%3C/svg%3E`; 5 | const svgMask2x = `data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='300' height='300' viewBox='0 0 150 150'%3E%3Crect width='150' height='150' rx='50' fill='black'/%3E%3C/svg%3E`; 6 | const svgMask3x = `data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='450' height='450' viewBox='0 0 150 150'%3E%3Crect width='150' height='150' rx='50' fill='black'/%3E%3C/svg%3E`; 7 | 8 | module.exports = plugin( 9 | function ({ addUtilities, theme }) { 10 | const options = theme('cornerSmoothing') || {}; 11 | const sizes = options.sizes || { 12 | sm: '25px', 13 | md: '60px', 14 | lg: '75px', 15 | }; 16 | 17 | const utilities = {}; 18 | 19 | Object.entries(sizes).forEach(([key, size]) => { 20 | utilities[`.smooth-corners-${key}`] = { 21 | '@supports (mask-border-width: 60px)': { 22 | 'mask-border': `url("${ 23 | options.maskPath ?? svgMask1x 24 | }") 49% fill / ${size}`, 25 | '@media (min-resolution: 2x)': { 26 | 'mask-border': `url("${ 27 | options.maskPath ?? svgMask2x 28 | }") 49% fill / ${size}`, 29 | }, 30 | '@media (min-resolution: 3x)': { 31 | 'mask-border': `url("${ 32 | options.maskPath ?? svgMask3x 33 | }") 49% fill / ${size}`, 34 | }, 35 | }, 36 | }; 37 | }); 38 | 39 | addUtilities(utilities); 40 | }, 41 | { 42 | theme: { 43 | cornerSmoothing: { 44 | maskPath: svgMask1x, 45 | sizes: { 46 | sm: '25px', 47 | md: '60px', 48 | lg: '75px', 49 | }, 50 | }, 51 | }, 52 | } 53 | ); 54 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Changelog 2 | 3 | All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines. 4 | 5 | ## [1.2.0](https://github.com/gtokman/corner-smoothing/compare/v1.1.1...v1.2.0) (2024-11-05) 6 | 7 | 8 | ### Features 9 | 10 | * convert to use base64 ([a8b214c](https://github.com/gtokman/corner-smoothing/commit/a8b214c58ca016e8ca5a996efbe993cfa3c41712)) 11 | 12 | 13 | ### Bug Fixes 14 | 15 | * update ([e5c2588](https://github.com/gtokman/corner-smoothing/commit/e5c2588db59bc8f33b9adeeb69269739a1006371)) 16 | * updates ([a2fde35](https://github.com/gtokman/corner-smoothing/commit/a2fde352a188ae9705ed65ce60228afe2db3facf)) 17 | * updates to where the assets live ([7575fe1](https://github.com/gtokman/corner-smoothing/commit/7575fe19842f6f0d2ed4e38dab55327df70c765a)) 18 | 19 | ### [1.1.1](https://github.com/gtokman/corner-smoothing/compare/v1.1.0...v1.1.1) (2024-11-05) 20 | 21 | 22 | ### Bug Fixes 23 | 24 | * update the package name ([8d80c6d](https://github.com/gtokman/corner-smoothing/commit/8d80c6d0918bd02e8887ade8f2357427866b14ea)) 25 | * updates ([c33ad22](https://github.com/gtokman/corner-smoothing/commit/c33ad22b62db02b59998c8620c7e0bb29f5885f3)) 26 | 27 | ## 1.1.0 (2024-11-05) 28 | 29 | 30 | ### Features 31 | 32 | * create extension ([a16e914](https://github.com/gtokman/corner-smoothing/commit/a16e914f79b486e8e2e37656e239292e8fb13ee5)) 33 | * create extension ([e8866a4](https://github.com/gtokman/corner-smoothing/commit/e8866a4a8be263dc4b07f4ef1dbe123374bd1806)) 34 | 35 | 36 | ### Bug Fixes 37 | 38 | * update ([f9fb921](https://github.com/gtokman/corner-smoothing/commit/f9fb921612ea6bf98b917c953088b3aaf99055d8)) 39 | * update readme ([48eff93](https://github.com/gtokman/corner-smoothing/commit/48eff935c0c24a2fe7146958dee1a2b257d177ac)) 40 | * update to include files ([725d0fb](https://github.com/gtokman/corner-smoothing/commit/725d0fbc20e89acd8c9a3c6fcbc85f9961fb1523)) 41 | * updates ([fb23b2a](https://github.com/gtokman/corner-smoothing/commit/fb23b2ac6b891be6becafabc8d33007229bf5f89)) 42 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | var plugin = require('tailwindcss/plugin'); 2 | var base64Mask1x = ''; 3 | var base64Mask2x = ''; 4 | var base64Mask3x = ''; 5 | module.exports = plugin(function (_a) { 6 | var addUtilities = _a.addUtilities, theme = _a.theme; 7 | var options = theme('cornerSmoothing') || {}; 8 | var sizes = options.sizes || { 9 | sm: '25px', 10 | md: '60px', 11 | lg: '75px', 12 | }; 13 | var utilities = {}; 14 | Object.entries(sizes).forEach(function (_a) { 15 | var key = _a[0], size = _a[1]; 16 | utilities[".smooth-corners-".concat(key)] = { 17 | '@supports (mask-border-width: 60px)': { 18 | 'mask-border': "url(\"".concat(options.maskPath || base64Mask1x, "\") 49% fill / ").concat(size), 19 | '@media (min-resolution: 2x)': { 20 | 'mask-border': "url(\"".concat(options.maskPath || base64Mask2x, "\") 49% fill / ").concat(size), 21 | }, 22 | '@media (min-resolution: 3x)': { 23 | 'mask-border': "url(\"".concat(options.maskPath || base64Mask3x, "\") 49% fill / ").concat(size), 24 | }, 25 | }, 26 | }; 27 | }); 28 | addUtilities(utilities); 29 | }, { 30 | theme: { 31 | cornerSmoothing: { 32 | maskPath: "".concat(base64Mask1x), 33 | maskPath2x: "".concat(base64Mask2x), 34 | maskPath3x: "".concat(base64Mask3x), 35 | sizes: { 36 | sm: '25px', 37 | md: '60px', 38 | lg: '75px', 39 | }, 40 | }, 41 | }, 42 | }); 43 | --------------------------------------------------------------------------------