├── .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 |
4 |
--------------------------------------------------------------------------------
/masks/squircle@2x.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/masks/squircle@3x.svg:
--------------------------------------------------------------------------------
1 |
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 |
11 |
12 | # Corner Smoothing
13 |
14 |
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 |
--------------------------------------------------------------------------------