Compare commits
62 Commits
0.555.0
...
angular-pa
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4257c22f4e | ||
|
|
4ca96b7b9c | ||
|
|
e2c1696c46 | ||
|
|
e9bd52b3a4 | ||
|
|
19dd078a94 | ||
|
|
18b1b037aa | ||
|
|
d6149a2e1c | ||
|
|
dbf5328ee8 | ||
|
|
f6c141221b | ||
|
|
a784e9922c | ||
|
|
1a1843cb2f | ||
|
|
7ced22b514 | ||
|
|
1b72561da4 | ||
|
|
69bf052ee5 | ||
|
|
6b4075b89b | ||
|
|
7a68e10b12 | ||
|
|
a4531a9985 | ||
|
|
3edcd9e0c3 | ||
|
|
1075461aab | ||
|
|
0b8f99326c | ||
|
|
7abb61630e | ||
|
|
3b0d158ea1 | ||
|
|
124572c83b | ||
|
|
4fcfb6a4d1 | ||
|
|
0f732b411d | ||
|
|
ce09c31f08 | ||
|
|
c2b059fb60 | ||
|
|
b3c80d027a | ||
|
|
20f30bb5ea | ||
|
|
c47ae67a3b | ||
|
|
7866a5a5c6 | ||
|
|
92bc88b001 | ||
|
|
e75fbcdec4 | ||
|
|
4cef8283a7 | ||
|
|
330f4b37db | ||
|
|
fd31cb44a8 | ||
|
|
790d30dbfa | ||
|
|
e7c075785f | ||
|
|
6d4c91707d | ||
|
|
c0ea92ebe7 | ||
|
|
42dc5508dd | ||
|
|
4dda432471 | ||
|
|
0775d8647e | ||
|
|
83ef8fc98d | ||
|
|
5b56ef705d | ||
|
|
dafe529892 | ||
|
|
151c5b145c | ||
|
|
d6f9043096 | ||
|
|
b4405f05ab | ||
|
|
9076da5f1b | ||
|
|
2e7d806282 | ||
|
|
c4e5730bc4 | ||
|
|
02b35e2518 | ||
|
|
f183c3ba20 | ||
|
|
67e9efb801 | ||
|
|
2e4c9a65be | ||
|
|
de4e8d0acd | ||
|
|
1f113d4274 | ||
|
|
dffffc7aff | ||
|
|
f78061b488 | ||
|
|
076e0bbcd9 | ||
|
|
33bb95edcd |
9
.github/ISSUE_TEMPLATE/02_bug_report.yml
vendored
@@ -13,16 +13,19 @@ body:
|
|||||||
description: Which Lucide packages are affected? You may select more than one.
|
description: Which Lucide packages are affected? You may select more than one.
|
||||||
options:
|
options:
|
||||||
- label: lucide
|
- label: lucide
|
||||||
- label: lucide-angular
|
- label: lucide-angular (old version)
|
||||||
|
- label: '@lucide/angular (new version)'
|
||||||
|
- label: '@lucide/astro'
|
||||||
- label: lucide-flutter
|
- label: lucide-flutter
|
||||||
- label: lucide-preact
|
- label: lucide-preact
|
||||||
- label: lucide-react
|
- label: lucide-react
|
||||||
- label: lucide-react-native
|
- label: lucide-react-native
|
||||||
- label: lucide-solid
|
- label: lucide-solid
|
||||||
- label: lucide-svelte
|
- label: lucide-static
|
||||||
|
- label: lucide-svelte (old version)
|
||||||
|
- label: '@lucide/svelte (new version)'
|
||||||
- label: lucide-vue
|
- label: lucide-vue
|
||||||
- label: lucide-vue-next
|
- label: lucide-vue-next
|
||||||
- label: lucide-astro
|
|
||||||
- label: Figma plugin
|
- label: Figma plugin
|
||||||
- label: source/main
|
- label: source/main
|
||||||
- label: other/not relevant
|
- label: other/not relevant
|
||||||
|
|||||||
10
.github/ISSUE_TEMPLATE/04_feature_request.yml
vendored
@@ -13,19 +13,23 @@ body:
|
|||||||
description: Which Lucide project do you wish this feature were added to? You may select more than one.
|
description: Which Lucide project do you wish this feature were added to? You may select more than one.
|
||||||
options:
|
options:
|
||||||
- label: lucide
|
- label: lucide
|
||||||
- label: lucide-angular
|
- label: lucide-angular (old version)
|
||||||
|
- label: '@lucide/angular (new version)'
|
||||||
|
- label: '@lucide/astro'
|
||||||
- label: lucide-flutter
|
- label: lucide-flutter
|
||||||
- label: lucide-preact
|
- label: lucide-preact
|
||||||
- label: lucide-react
|
- label: lucide-react
|
||||||
- label: lucide-react-native
|
- label: lucide-react-native
|
||||||
- label: lucide-solid
|
- label: lucide-solid
|
||||||
- label: lucide-svelte
|
- label: lucide-static
|
||||||
|
- label: lucide-svelte (old version)
|
||||||
|
- label: '@lucide/svelte (new version)'
|
||||||
- label: lucide-vue
|
- label: lucide-vue
|
||||||
- label: lucide-vue-next
|
- label: lucide-vue-next
|
||||||
- label: lucide-astro
|
|
||||||
- label: Figma plugin
|
- label: Figma plugin
|
||||||
- label: all JS packages
|
- label: all JS packages
|
||||||
- label: site
|
- label: site
|
||||||
|
- label: other/not relevant
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: textarea
|
- type: textarea
|
||||||
|
|||||||
1
.github/labeler.yml
vendored
@@ -59,6 +59,7 @@
|
|||||||
🅰️ angular package:
|
🅰️ angular package:
|
||||||
- changed-files:
|
- changed-files:
|
||||||
- any-glob-to-any-file:
|
- any-glob-to-any-file:
|
||||||
|
- 'packages/angular/*'
|
||||||
- 'packages/lucide-angular/*'
|
- 'packages/lucide-angular/*'
|
||||||
|
|
||||||
# For changes in the lucide preact package
|
# For changes in the lucide preact package
|
||||||
|
|||||||
41
.github/workflows/angular.yml
vendored
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
name: Lucide Angular checks
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
paths:
|
||||||
|
- packages/angular/**
|
||||||
|
- tools/build-icons/**
|
||||||
|
- pnpm-lock.yaml
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v6
|
||||||
|
- uses: pnpm/action-setup@v4
|
||||||
|
- uses: actions/setup-node@v6
|
||||||
|
with:
|
||||||
|
cache: 'pnpm'
|
||||||
|
node-version-file: 'package.json'
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: pnpm install --frozen-lockfile
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
run: pnpm --filter @lucide/angular build
|
||||||
|
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v6
|
||||||
|
- uses: pnpm/action-setup@v4
|
||||||
|
- uses: actions/setup-node@v6
|
||||||
|
with:
|
||||||
|
cache: 'pnpm'
|
||||||
|
node-version-file: 'package.json'
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: pnpm install --frozen-lockfile
|
||||||
|
|
||||||
|
- name: Test
|
||||||
|
run: pnpm --filter @lucide/angular test
|
||||||
62
.github/workflows/ci.yml
vendored
@@ -7,6 +7,10 @@ on:
|
|||||||
paths:
|
paths:
|
||||||
- icons/**/*.svg
|
- icons/**/*.svg
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
id-token: write # Required for OIDC
|
||||||
|
contents: write
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
create-release:
|
create-release:
|
||||||
if: github.repository == 'lucide-icons/lucide' && startsWith(github.event.head_commit.message, 'feat(icons)')
|
if: github.repository == 'lucide-icons/lucide' && startsWith(github.event.head_commit.message, 'feat(icons)')
|
||||||
@@ -15,9 +19,9 @@ jobs:
|
|||||||
VERSION: ${{ steps.new-version.outputs.NEW_VERSION }}
|
VERSION: ${{ steps.new-version.outputs.NEW_VERSION }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -32,25 +36,19 @@ jobs:
|
|||||||
id: latest-tag
|
id: latest-tag
|
||||||
run: echo "LATEST_TAG=$(git describe --tags `git rev-list --tags --max-count=1`)" >> $GITHUB_OUTPUT
|
run: echo "LATEST_TAG=$(git describe --tags `git rev-list --tags --max-count=1`)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Log latest tag
|
||||||
|
run: echo '${{ steps.latest-tag.outputs.LATEST_TAG }}'
|
||||||
|
|
||||||
- name: Check if we can patch
|
- name: Check if we can patch
|
||||||
run: .github/workflows/version-up.sh --minor
|
run: pnpm semver $LATEST_TAG -i minor
|
||||||
|
env:
|
||||||
|
LATEST_TAG: ${{ steps.latest-tag.outputs.LATEST_TAG }}
|
||||||
|
|
||||||
- name: Create new version
|
- name: Create new version
|
||||||
id: new-version
|
id: new-version
|
||||||
run: echo "NEW_VERSION=$(.github/workflows/version-up.sh --minor)" >> $GITHUB_OUTPUT
|
run: echo "NEW_VERSION=$(pnpm semver $LATEST_TAG -i minor)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Create change log
|
|
||||||
id: change-log
|
|
||||||
run: |
|
|
||||||
CHANGE_LOG=$(pnpm run generate:changelog --old-tag=${{ steps.latest-tag.outputs.LATEST_TAG }})
|
|
||||||
CHANGE_LOG=$(tail -n +5 <<< $CHANGE_LOG)
|
|
||||||
echo $CHANGE_LOG
|
|
||||||
EOF=$(dd if=/dev/urandom bs=15 count=1 status=none | base64)
|
|
||||||
echo "CHANGE_LOG<<$EOF" >> $GITHUB_OUTPUT
|
|
||||||
echo "$CHANGE_LOG" >> $GITHUB_OUTPUT
|
|
||||||
echo "$EOF" >> $GITHUB_OUTPUT
|
|
||||||
env:
|
env:
|
||||||
GITHUB_API_KEY: ${{ secrets.GITHUB_TOKEN }}
|
LATEST_TAG: ${{ steps.latest-tag.outputs.LATEST_TAG }}
|
||||||
|
|
||||||
- name: Check output
|
- name: Check output
|
||||||
run: |
|
run: |
|
||||||
@@ -64,38 +62,6 @@ jobs:
|
|||||||
name: Version ${{ steps.new-version.outputs.NEW_VERSION }}
|
name: Version ${{ steps.new-version.outputs.NEW_VERSION }}
|
||||||
generate_release_notes: true
|
generate_release_notes: true
|
||||||
|
|
||||||
test-semantic-release:
|
|
||||||
if: github.repository == 'lucide-icons/lucide'
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Semantic Release
|
|
||||||
id: semantic
|
|
||||||
uses: cycjimmy/semantic-release-action@v4
|
|
||||||
with:
|
|
||||||
tag_format: ${version}
|
|
||||||
branches: |
|
|
||||||
['new-release-workflow']
|
|
||||||
extends: |
|
|
||||||
semantic-release-monorepo
|
|
||||||
extra_plugins: |
|
|
||||||
@semantic-release/github
|
|
||||||
@semantic-release/git
|
|
||||||
@semantic-release/release-notes-generator
|
|
||||||
conventional-changelog-conventionalcommits
|
|
||||||
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
|
|
||||||
- name: Log output
|
|
||||||
if: steps.semantic.outputs.new_release_published == 'true'
|
|
||||||
run: |
|
|
||||||
echo ${{ steps.semantic.outputs.new_release_version }}
|
|
||||||
echo ${{ steps.semantic.outputs.new_release_major_version }}
|
|
||||||
echo ${{ steps.semantic.outputs.new_release_minor_version }}
|
|
||||||
echo ${{ steps.semantic.outputs.new_release_patch_version }}
|
|
||||||
|
|
||||||
start-release:
|
start-release:
|
||||||
if: github.repository == 'lucide-icons/lucide'
|
if: github.repository == 'lucide-icons/lucide'
|
||||||
needs: create-release
|
needs: create-release
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
name: Close Issue with Banned Phrases
|
name: Close Icon Requests with Brand Terms
|
||||||
|
|
||||||
on:
|
on:
|
||||||
issues:
|
issues:
|
||||||
@@ -11,25 +11,38 @@ jobs:
|
|||||||
issues: write
|
issues: write
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
- name: Check for blocked phrases in issue title
|
- name: Load stopwords from JSON & check issue title & body
|
||||||
|
if: contains(github.event.issue.labels.*.name, '🙌 icon request')
|
||||||
run: |
|
run: |
|
||||||
ISSUE_TITLE=$(jq -r '.issue.title' "$GITHUB_EVENT_PATH")
|
ISSUE_TITLE=$(jq -r '.issue.title' "$GITHUB_EVENT_PATH")
|
||||||
BLOCKED_PHRASES=("twitter" "whatsapp" "logo" "google" "tiktok" "facebook" "slack" "discord" "bluesky" "spotify" "behance" "pix" "x.com" "telegram")
|
ISSUE_BODY=$(jq -r '.issue.body // ""' "$GITHUB_EVENT_PATH")
|
||||||
|
ICON_NAME_SECTION=$(printf '%s\n' "$ISSUE_BODY" | awk '/### Icon name/{flag=1; next} /^### /{flag=0} flag')
|
||||||
|
|
||||||
# Check title and body for blocked phrases
|
jq -r 'to_entries[] | "\(.key) \(.value)"' brand-stopwords.json | while read -r KEY VALUE; do
|
||||||
for PHRASE in "${BLOCKED_PHRASES[@]}"
|
SAFE_KEY=$(printf '%s\n' "$KEY" | sed 's/[][\.^$*]/\\&/g')
|
||||||
do
|
SAFE_VALUE=$(printf '%s\n' "$VALUE" | sed 's/[][\.^$*]/\\&/g')
|
||||||
if echo "$ISSUE_TITLE" | grep -i "$PHRASE"; then
|
|
||||||
gh issue close ${{ github.event.issue.number }} --reason "not planned" --comment "This looks like a duplicate, use the [search](https://github.com/lucide-icons/lucide/issues?q=is%3Aissue+$PHRASE) to find similar issues.
|
|
||||||
|
|
||||||
Read [our official statement about brand logos in Lucide](https://github.com/lucide-icons/lucide/blob/main/BRAND_LOGOS_STATEMENT.md).
|
if echo "$ISSUE_TITLE" | grep -iqE "$SAFE_KEY|$SAFE_VALUE" || \
|
||||||
|
{ [ -n "$ICON_NAME_SECTION" ] && echo "$ICON_NAME_SECTION" | grep -iqE "$SAFE_KEY|$SAFE_VALUE"; }; then
|
||||||
|
|
||||||
|
gh issue close ${{ github.event.issue.number }} \
|
||||||
|
--reason "not_planned" \
|
||||||
|
--comment "It looks like this request is about **${VALUE}**, which is a brand logo.
|
||||||
|
|
||||||
|
Lucide **does not accept** brand logos, and we do not plan to add them in the future. This is due to a combination of **legal restrictions**, **design consistency concerns**, and **practical maintenance reasons**.
|
||||||
|
|
||||||
|
[Click here to read our official statement about brand logos in Lucide.](./BRAND_LOGOS_STATEMENT.md)
|
||||||
|
|
||||||
|
You can [search for similar issues.](https://github.com/lucide-icons/lucide/issues?q=is%3Aissue+${VALUE})
|
||||||
|
|
||||||
|
We’re always happy to help on [Discord](https://discord.gg/EH6nSts)."
|
||||||
|
|
||||||
Always happy to help on [Discord](https://discord.gg/EH6nSts)."
|
|
||||||
gh issue lock ${{ github.event.issue.number }} --reason spam
|
gh issue lock ${{ github.event.issue.number }} --reason spam
|
||||||
exit 1
|
exit 0
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
env:
|
|
||||||
GH_TOKEN: ${{ github.token }}
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
|||||||
2
.github/workflows/labeler.yml
vendored
@@ -9,5 +9,5 @@ jobs:
|
|||||||
pull-requests: write
|
pull-requests: write
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: actions/labeler@v5
|
- uses: actions/labeler@v5
|
||||||
|
|||||||
4
.github/workflows/lint-code.yml
vendored
@@ -9,9 +9,9 @@ jobs:
|
|||||||
lint-code:
|
lint-code:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
6
.github/workflows/linting-icons.yml
vendored
@@ -13,8 +13,8 @@ jobs:
|
|||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
- name: Get changed files
|
- name: Get changed files
|
||||||
@@ -34,6 +34,6 @@ jobs:
|
|||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- name: Check Uniqueness of Aliases
|
- name: Check Uniqueness of Aliases
|
||||||
run: "! cat <(printf \"%s\\n\" icons/*.json | while read -r name; do basename \"$name\" .json; done) <(jq -cr 'select(.aliases) | .aliases[] | if type==\"string\" then . else .name end' icons/*.json) | sort | uniq -c | grep -ve '^\\s*1 '"
|
run: "! cat <(printf \"%s\\n\" icons/*.json | while read -r name; do basename \"$name\" .json; done) <(jq -cr 'select(.aliases) | .aliases[] | if type==\"string\" then . else .name end' icons/*.json) | sort | uniq -c | grep -ve '^\\s*1 '"
|
||||||
|
|||||||
8
.github/workflows/lucide-angular.yml
vendored
@@ -11,9 +11,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -27,9 +27,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
4
.github/workflows/lucide-astro.yml
vendored
@@ -12,9 +12,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v2
|
- uses: pnpm/action-setup@v2
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
|
|||||||
4
.github/workflows/lucide-font.yml
vendored
@@ -10,9 +10,9 @@ jobs:
|
|||||||
lucide-font:
|
lucide-font:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
4
.github/workflows/lucide-preact.yml
vendored
@@ -13,9 +13,9 @@ jobs:
|
|||||||
lucide-preact:
|
lucide-preact:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
8
.github/workflows/lucide-react-native.yml
vendored
@@ -13,9 +13,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -29,9 +29,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
8
.github/workflows/lucide-react.yml
vendored
@@ -14,9 +14,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -30,9 +30,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
4
.github/workflows/lucide-shared.yml
vendored
@@ -10,9 +10,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
8
.github/workflows/lucide-solid.yml
vendored
@@ -13,9 +13,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -29,9 +29,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
4
.github/workflows/lucide-static.yml
vendored
@@ -11,9 +11,9 @@ jobs:
|
|||||||
lucide-static:
|
lucide-static:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
8
.github/workflows/lucide-svelte-5.yml
vendored
@@ -13,9 +13,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -29,9 +29,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
8
.github/workflows/lucide-svelte.yml
vendored
@@ -13,9 +13,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -29,9 +29,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
8
.github/workflows/lucide-vue-next.yml
vendored
@@ -13,9 +13,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -29,9 +29,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
8
.github/workflows/lucide.yml
vendored
@@ -12,9 +12,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -28,9 +28,9 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
@@ -14,8 +14,8 @@ jobs:
|
|||||||
pull-requests: write
|
pull-requests: write
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|
||||||
@@ -25,7 +25,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
files: icons/*.svg
|
files: icons/*.svg
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
- name: Install svgson for code preview (safer and faster than installing all deps)
|
- name: Install svgson for code preview (safer and faster than installing all deps)
|
||||||
run: npm install svgson@5.3.1 --force
|
run: npm install svgson@5.3.1 --force
|
||||||
|
|
||||||
|
|||||||
@@ -16,11 +16,11 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
# We checkout the main branch of main repository for security reasons.
|
# We checkout the main branch of main repository for security reasons.
|
||||||
# This is to prevent the workflow from running on a forked repository.
|
# This is to prevent the workflow from running on a forked repository.
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
with:
|
with:
|
||||||
repository: lucide-icons/lucide
|
repository: lucide-icons/lucide
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
|
|||||||
38
.github/workflows/release.yml
vendored
@@ -18,9 +18,13 @@ on:
|
|||||||
description: Version
|
description: Version
|
||||||
required: true
|
required: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
id-token: write # Required for OIDC
|
||||||
|
contents: write
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
pre-release:
|
pre-release:
|
||||||
if: github.repository == 'lucide-icons/lucide' && contains('["ericfennis", "karsa-mistmere"]', github.actor)
|
if: github.repository == 'lucide-icons/lucide' && contains('["ericfennis", "karsa-mistmere", "jguddas"]', github.actor)
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
outputs:
|
outputs:
|
||||||
VERSION: ${{ steps.get_version.outputs.VERSION }}
|
VERSION: ${{ steps.get_version.outputs.VERSION }}
|
||||||
@@ -39,7 +43,8 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: pre-release
|
needs: pre-release
|
||||||
permissions:
|
permissions:
|
||||||
id-token: write
|
id-token: write # Required for OIDC
|
||||||
|
contents: read
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
@@ -53,13 +58,14 @@ jobs:
|
|||||||
'lucide-preact',
|
'lucide-preact',
|
||||||
'lucide-solid',
|
'lucide-solid',
|
||||||
'lucide-svelte',
|
'lucide-svelte',
|
||||||
|
'@lucide/angular',
|
||||||
'@lucide/astro',
|
'@lucide/astro',
|
||||||
'@lucide/svelte',
|
'@lucide/svelte',
|
||||||
]
|
]
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -67,9 +73,6 @@ jobs:
|
|||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: pnpm install --frozen-lockfile
|
run: pnpm install --frozen-lockfile
|
||||||
|
|
||||||
- name: Set Auth Token
|
|
||||||
run: npm config set //registry.npmjs.org/:_authToken ${{ inputs.NPM_TOKEN || secrets.NPM_TOKEN }}
|
|
||||||
|
|
||||||
- name: Set new version
|
- name: Set new version
|
||||||
run: pnpm --filter ${{ matrix.package }} version --new-version ${{ needs.pre-release.outputs.VERSION }} --no-git-tag-version
|
run: pnpm --filter ${{ matrix.package }} version --new-version ${{ needs.pre-release.outputs.VERSION }} --no-git-tag-version
|
||||||
|
|
||||||
@@ -89,12 +92,14 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: [pre-release, lucide-font]
|
needs: [pre-release, lucide-font]
|
||||||
permissions:
|
permissions:
|
||||||
id-token: write
|
id-token: write # Required for OIDC
|
||||||
|
contents: read
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: actions/download-artifact@v4
|
- uses: actions/download-artifact@v4
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -102,9 +107,6 @@ jobs:
|
|||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: pnpm install --frozen-lockfile
|
run: pnpm install --frozen-lockfile
|
||||||
|
|
||||||
- name: Set Auth Token
|
|
||||||
run: npm config set //registry.npmjs.org/:_authToken ${{ secrets.NPM_TOKEN }}
|
|
||||||
|
|
||||||
- name: Set new version
|
- name: Set new version
|
||||||
run: pnpm --filter lucide-static version --new-version ${{ needs.pre-release.outputs.VERSION }} --no-git-tag-version
|
run: pnpm --filter lucide-static version --new-version ${{ needs.pre-release.outputs.VERSION }} --no-git-tag-version
|
||||||
|
|
||||||
@@ -124,9 +126,9 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: pre-release
|
needs: pre-release
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: pnpm/action-setup@v4
|
- uses: pnpm/action-setup@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
cache: 'pnpm'
|
cache: 'pnpm'
|
||||||
node-version-file: 'package.json'
|
node-version-file: 'package.json'
|
||||||
@@ -150,9 +152,11 @@ jobs:
|
|||||||
if: github.repository == 'lucide-icons/lucide'
|
if: github.repository == 'lucide-icons/lucide'
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: [pre-release, lucide-font]
|
needs: [pre-release, lucide-font]
|
||||||
|
permissions:
|
||||||
|
id-token: write # Required for OIDC
|
||||||
|
contents: write
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
- uses: actions/download-artifact@v4
|
- uses: actions/download-artifact@v4
|
||||||
- name: Zip font and icons
|
- name: Zip font and icons
|
||||||
run: |
|
run: |
|
||||||
|
|||||||
5
.github/workflows/request-review.yml
vendored
@@ -2,7 +2,8 @@ name: 'Request Review'
|
|||||||
on:
|
on:
|
||||||
pull_request_target:
|
pull_request_target:
|
||||||
types: [opened]
|
types: [opened]
|
||||||
paths: icons/*.svg
|
paths:
|
||||||
|
- icons/*.svg
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
request-review:
|
request-review:
|
||||||
@@ -12,7 +13,7 @@ jobs:
|
|||||||
contents: read
|
contents: read
|
||||||
pull-requests: write
|
pull-requests: write
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v6
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
ref: refs/pull/${{ github.event.pull_request.number }}/merge
|
ref: refs/pull/${{ github.event.pull_request.number }}/merge
|
||||||
|
|||||||
284
.github/workflows/version-up.sh
vendored
@@ -1,284 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
## Copyright (C) 2017, Oleksandr Kucherenko
|
|
||||||
## Last revisit: 2017-09-29
|
|
||||||
|
|
||||||
## get highest version tag for all branches
|
|
||||||
function highest_tag(){
|
|
||||||
local TAG=$(git describe --tags `git rev-list --tags --max-count=1`)
|
|
||||||
echo "$TAG"
|
|
||||||
}
|
|
||||||
|
|
||||||
## extract current branch name
|
|
||||||
function current_branch(){
|
|
||||||
## expected: heads/{branch_name}
|
|
||||||
## expected: {branch_name}
|
|
||||||
local BRANCH=$(git rev-parse --abbrev-ref HEAD | cut -d"/" -f2)
|
|
||||||
echo "$BRANCH"
|
|
||||||
}
|
|
||||||
|
|
||||||
## get latest/head commit hash number
|
|
||||||
function head_hash(){
|
|
||||||
local COMMIT_HASH=$(git rev-parse --verify HEAD)
|
|
||||||
echo "$COMMIT_HASH"
|
|
||||||
}
|
|
||||||
|
|
||||||
## extract tag commit hash code, tag name provided by argument
|
|
||||||
function tag_hash(){
|
|
||||||
local TAG_HASH=$(git log -1 --format=format:"%H" $1 2>/dev/null | tail -n1)
|
|
||||||
echo "$TAG_HASH"
|
|
||||||
}
|
|
||||||
|
|
||||||
## get latest revision number
|
|
||||||
function latest_revision(){
|
|
||||||
local REV=$(git rev-list --count HEAD 2>/dev/null)
|
|
||||||
echo "$REV"
|
|
||||||
}
|
|
||||||
|
|
||||||
## parse last found tag, extract it PARTS
|
|
||||||
function parse_last(){
|
|
||||||
local position=$(($1-1))
|
|
||||||
|
|
||||||
# two parts found only
|
|
||||||
local SUBS=( ${PARTS[$position]//-/ } )
|
|
||||||
#echo ${SUBS[@]}, size: ${#SUBS}
|
|
||||||
|
|
||||||
# found NUMBER
|
|
||||||
PARTS[$position]=${SUBS[0]}
|
|
||||||
#echo ${PARTS[@]}
|
|
||||||
|
|
||||||
# found SUFFIX
|
|
||||||
if [[ ${#SUBS} -ge 1 ]]; then
|
|
||||||
PARTS[4]=${SUBS[1],,} #lowercase
|
|
||||||
#echo ${PARTS[@]}, ${SUBS[@]}
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
## increment REVISION part, don't touch STAGE
|
|
||||||
function increment_revision(){
|
|
||||||
PARTS[3]=$(( PARTS[3] + 1 ))
|
|
||||||
IS_DIRTY=1
|
|
||||||
}
|
|
||||||
|
|
||||||
## increment PATCH part, reset all other lower PARTS, don't touch STAGE
|
|
||||||
function increment_patch(){
|
|
||||||
PARTS[2]=$(( PARTS[2] + 1 ))
|
|
||||||
PARTS[3]=0
|
|
||||||
IS_DIRTY=1
|
|
||||||
}
|
|
||||||
|
|
||||||
## increment MINOR part, reset all other lower PARTS, don't touch STAGE
|
|
||||||
function increment_minor(){
|
|
||||||
PARTS[1]=$(( PARTS[1] + 1 ))
|
|
||||||
PARTS[2]=0
|
|
||||||
PARTS[3]=0
|
|
||||||
IS_DIRTY=1
|
|
||||||
}
|
|
||||||
|
|
||||||
## increment MAJOR part, reset all other lower PARTS, don't touch STAGE
|
|
||||||
function incremet_major(){
|
|
||||||
PARTS[0]="v$(( PARTS[0] + 1 ))"
|
|
||||||
PARTS[1]=0
|
|
||||||
PARTS[2]=0
|
|
||||||
PARTS[3]=0
|
|
||||||
IS_DIRTY=1
|
|
||||||
}
|
|
||||||
|
|
||||||
## increment the number only of last found PART: REVISION --> PATCH --> MINOR. don't touch STAGE
|
|
||||||
function increment_last_found(){
|
|
||||||
if [[ "${#PARTS[3]}" == 0 || "${PARTS[3]}" == "0" ]]; then
|
|
||||||
if [[ "${#PARTS[2]}" == 0 || "${PARTS[2]}" == "0" ]]; then
|
|
||||||
increment_minor
|
|
||||||
else
|
|
||||||
increment_patch
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
increment_revision
|
|
||||||
fi
|
|
||||||
|
|
||||||
# stage part is not EMPTY
|
|
||||||
if [[ "${#PARTS[4]}" != 0 ]]; then
|
|
||||||
IS_SHIFT=1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
## compose version from PARTS
|
|
||||||
function compose(){
|
|
||||||
MAJOR="${PARTS[0]}"
|
|
||||||
MINOR=".${PARTS[1]}"
|
|
||||||
PATCH=".${PARTS[2]}"
|
|
||||||
REVISION=".${PARTS[3]}"
|
|
||||||
SUFFIX="-${PARTS[4]}"
|
|
||||||
|
|
||||||
if [[ "${#PATCH}" == 1 ]]; then # if empty {PATCH}
|
|
||||||
PATCH=""
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "${#REVISION}" == 1 ]]; then # if empty {REVISION}
|
|
||||||
REVISION=""
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "${PARTS[3]}" == "0" ]]; then # if revision is ZERO
|
|
||||||
REVISION=""
|
|
||||||
fi
|
|
||||||
|
|
||||||
# shrink patch and revision
|
|
||||||
if [[ -z "${REVISION// }" ]]; then
|
|
||||||
if [[ "${PARTS[2]}" == "0" ]]; then
|
|
||||||
PATCH=".0"
|
|
||||||
fi
|
|
||||||
else # revision is not EMPTY
|
|
||||||
if [[ "${#PATCH}" == 0 ]]; then
|
|
||||||
PATCH=".0"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# remove suffix if we don't have a alpha/beta/rc
|
|
||||||
if [[ "${#SUFFIX}" == 1 ]]; then
|
|
||||||
SUFFIX=""
|
|
||||||
fi
|
|
||||||
|
|
||||||
|
|
||||||
echo "${MAJOR}${MINOR}${PATCH}${REVISION}${SUFFIX}" #full format
|
|
||||||
}
|
|
||||||
|
|
||||||
# initial version used for repository without tags
|
|
||||||
INIT_VERSION=0.0.0.0-alpha
|
|
||||||
|
|
||||||
# do GIT data extracting
|
|
||||||
TAG=$(highest_tag)
|
|
||||||
REVISION=$(latest_revision)
|
|
||||||
BRANCH=$(current_branch)
|
|
||||||
TAG_HASH=$(tag_hash $TAG)
|
|
||||||
HEAD_HASH=$(head_hash)
|
|
||||||
|
|
||||||
# if tag and branch commit hashes are different, than print info about that
|
|
||||||
#echo $HEAD_HASH vs $TAG_HASH
|
|
||||||
if [[ "$@" == "" ]]; then
|
|
||||||
if [[ "$TAG_HASH" == "$HEAD_HASH" ]]; then
|
|
||||||
echo "Tag $TAG and HEAD are aligned. We will stay on the TAG version."
|
|
||||||
echo ""
|
|
||||||
NO_ARGS_VALUE='--stay'
|
|
||||||
else
|
|
||||||
PATTERN="^[0-9]+.[0-9]+(.[0-9]+)*(-(alpha|beta|rc))*$"
|
|
||||||
|
|
||||||
if [[ "$BRANCH" =~ $PATTERN ]]; then
|
|
||||||
echo "Detected version branch '$BRANCH'. We will auto-increment the last version PART."
|
|
||||||
echo ""
|
|
||||||
NO_ARGS_VALUE='--default'
|
|
||||||
else
|
|
||||||
echo "Detected branch name '$BRANCH' than does not match version pattern. We will increase MINOR."
|
|
||||||
echo ""
|
|
||||||
NO_ARGS_VALUE='--minor'
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
#
|
|
||||||
# {MAJOR}.{MINOR}[.{PATCH}[.{REVISION}][-(.*)]
|
|
||||||
#
|
|
||||||
# Suffix: alpha, beta, rc
|
|
||||||
# No Suffix --> {NEW_VERSION}-alpha
|
|
||||||
# alpha --> beta
|
|
||||||
# beta --> rc
|
|
||||||
# rc --> {VERSION}
|
|
||||||
#
|
|
||||||
PARTS=( ${TAG//./ } )
|
|
||||||
parse_last ${#PARTS[@]} # array size as argument
|
|
||||||
#echo ${PARTS[@]}
|
|
||||||
|
|
||||||
# if no parameters than emulate --default parameter
|
|
||||||
if [[ "$@" == "" ]]; then
|
|
||||||
set -- $NO_ARGS_VALUE
|
|
||||||
fi
|
|
||||||
|
|
||||||
# parse input parameters
|
|
||||||
for i in "$@"
|
|
||||||
do
|
|
||||||
key="$i"
|
|
||||||
|
|
||||||
case $key in
|
|
||||||
-a|--alpha) # switched to ALPHA
|
|
||||||
PARTS[4]="alpha"
|
|
||||||
IS_SHIFT=1
|
|
||||||
;;
|
|
||||||
-b|--beta) # switched to BETA
|
|
||||||
PARTS[4]="beta"
|
|
||||||
IS_SHIFT=1
|
|
||||||
;;
|
|
||||||
-c|--release-candidate) # switched to RC
|
|
||||||
PARTS[4]="rc"
|
|
||||||
IS_SHIFT=1
|
|
||||||
;;
|
|
||||||
-r|--release) # switched to RELEASE
|
|
||||||
PARTS[4]=""
|
|
||||||
IS_SHIFT=1
|
|
||||||
;;
|
|
||||||
-p|--patch) # increment of PATCH
|
|
||||||
increment_patch
|
|
||||||
;;
|
|
||||||
-e|--revision) # increment of REVISION
|
|
||||||
increment_revision
|
|
||||||
;;
|
|
||||||
-g|--git-revision) # use git revision number as a revision part§
|
|
||||||
PARTS[3]=$(( REVISION ))
|
|
||||||
IS_DIRTY=1
|
|
||||||
;;
|
|
||||||
-i|--minor) # increment of MINOR by default
|
|
||||||
increment_minor
|
|
||||||
;;
|
|
||||||
--default) # stay on the same stage, but increment only last found PART of version code
|
|
||||||
increment_last_found
|
|
||||||
;;
|
|
||||||
-m|--major) # increment of MAJOR
|
|
||||||
incremet_major
|
|
||||||
;;
|
|
||||||
-s|--stay) # extract version info
|
|
||||||
IS_DIRTY=1
|
|
||||||
NO_APPLY_MSG=1
|
|
||||||
;;
|
|
||||||
-t|--tag-only) # extract version info
|
|
||||||
TAG_ONLY=1
|
|
||||||
;;
|
|
||||||
--apply)
|
|
||||||
DO_APPLY=1
|
|
||||||
;;
|
|
||||||
-h|--help)
|
|
||||||
help
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
shift
|
|
||||||
done
|
|
||||||
|
|
||||||
# detected shift, but no increment
|
|
||||||
if [[ "$IS_SHIFT" == "1" ]]; then
|
|
||||||
# temporary disable stage shift
|
|
||||||
stage=${PARTS[4]}
|
|
||||||
PARTS[4]=''
|
|
||||||
|
|
||||||
# detect first run on repository, INIT_VERSION was used
|
|
||||||
if [[ "$(compose)" == "0.0" ]]; then
|
|
||||||
increment_minor
|
|
||||||
fi
|
|
||||||
|
|
||||||
PARTS[4]=$stage
|
|
||||||
fi
|
|
||||||
|
|
||||||
# no increment applied yet and no shift of state, do minor increase
|
|
||||||
if [[ "$IS_DIRTY$IS_SHIFT" == "" ]]; then
|
|
||||||
increment_minor
|
|
||||||
fi
|
|
||||||
|
|
||||||
compose
|
|
||||||
|
|
||||||
# is proposed tag in conflict with any other TAG
|
|
||||||
PROPOSED_HASH=$(tag_hash $(compose))
|
|
||||||
if [[ "${#PROPOSED_HASH}" -gt 0 && "$NO_APPLY_MSG" == "" ]]; then
|
|
||||||
echo -e "\033[31mERROR:\033[0m "
|
|
||||||
echo -e "\033[31mERROR:\033[0m Found conflict with existing tag \033[32m$(compose)\033[0m / $PROPOSED_HASH"
|
|
||||||
echo -e "\033[31mERROR:\033[0m Only manual resolving is possible now."
|
|
||||||
echo -e "\033[31mERROR:\033[0m "
|
|
||||||
echo -e "\033[31mERROR:\033[0m To Resolve try to add --revision or --patch modifier."
|
|
||||||
echo -e "\033[31mERROR:\033[0m "
|
|
||||||
echo ""
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
2
.gitignore
vendored
@@ -44,7 +44,7 @@ docs/.vitepress/data/releaseMetaData
|
|||||||
docs/.vitepress/data/categoriesData.json
|
docs/.vitepress/data/categoriesData.json
|
||||||
docs/.vitepress/data/iconDetails
|
docs/.vitepress/data/iconDetails
|
||||||
docs/.vitepress/data/relatedIcons.json
|
docs/.vitepress/data/relatedIcons.json
|
||||||
|
docs/.vitepress/data/brandStopwords.json
|
||||||
docs/.vercel
|
docs/.vercel
|
||||||
docs/.nitro
|
docs/.nitro
|
||||||
.gitignore
|
.gitignore
|
||||||
|
|
||||||
|
|||||||
149
brand-stopwords.json
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
{
|
||||||
|
"adobe": "Adobe",
|
||||||
|
"airplay": "AirPlay",
|
||||||
|
"amazon": "Amazon",
|
||||||
|
"angular": "Angular",
|
||||||
|
"aws": "AWS",
|
||||||
|
"azure": "Azure",
|
||||||
|
"bandcamp": "Bandcamp",
|
||||||
|
"behance": "Behance",
|
||||||
|
"bitbucket": "Bitbucket",
|
||||||
|
"blender": "Blender",
|
||||||
|
"bluesky": "BlueSky",
|
||||||
|
"bootstrap": "Bootstrap",
|
||||||
|
"brave": "Brave",
|
||||||
|
"chakra": "Chakra UI",
|
||||||
|
"chrome": "Chrome",
|
||||||
|
"codepen": "Codepen",
|
||||||
|
"codesandbox": "CodeSandbox",
|
||||||
|
"csharp": "C#",
|
||||||
|
"cypress": "Cypress",
|
||||||
|
"dart": "Dart",
|
||||||
|
"deezer": "Deezer",
|
||||||
|
"deno": "Deno",
|
||||||
|
"discord": "Discord",
|
||||||
|
"docker": "Docker",
|
||||||
|
"dribbble": "Dribbble",
|
||||||
|
"dropbox": "Dropbox",
|
||||||
|
"edge": "Edge",
|
||||||
|
"ember": "Ember",
|
||||||
|
"epic": "Epic Games",
|
||||||
|
"erlang": "Erlang",
|
||||||
|
"esbuild": "esbuild",
|
||||||
|
"eslint": "ESLint",
|
||||||
|
"facebook": "Facebook",
|
||||||
|
"figjam": "FigJam",
|
||||||
|
"figma": "Figma",
|
||||||
|
"firebase": "Firebase",
|
||||||
|
"firefox": "Firefox",
|
||||||
|
"framer": "Framer",
|
||||||
|
"gatsby": "Gatsby",
|
||||||
|
"gcp": "Google Cloud",
|
||||||
|
"github": "GitHub",
|
||||||
|
"gitlab": "GitLab",
|
||||||
|
"golang": "GoLang",
|
||||||
|
"google": "Google",
|
||||||
|
"gmail": "Gmail",
|
||||||
|
"gravatar": "Gravatar",
|
||||||
|
"haskell": "Haskell",
|
||||||
|
"instagram": "Instagram",
|
||||||
|
"java": "Java",
|
||||||
|
"javascript": "JavaScript",
|
||||||
|
"jest": "Jest",
|
||||||
|
"jira": "Jira",
|
||||||
|
"kotlin": "Kotlin",
|
||||||
|
"kubernetes": "Kubernetes",
|
||||||
|
"less": "Less",
|
||||||
|
"leetcode": "LeetCode",
|
||||||
|
"leet-code": "LeetCode",
|
||||||
|
"line": "LINE",
|
||||||
|
"linkedin": "LinkedIn",
|
||||||
|
"lua": "Lua",
|
||||||
|
"mariadb": "MariaDB",
|
||||||
|
"mcp": "MCP",
|
||||||
|
"messenger": "Messenger",
|
||||||
|
"microsoft": "Microsoft",
|
||||||
|
"mongodb": "MongoDB",
|
||||||
|
"mui": "Material UI",
|
||||||
|
"mysql": "MySQL",
|
||||||
|
"nestjs": "NestJS",
|
||||||
|
"netflix": "Netflix",
|
||||||
|
"netlify": "Netlify",
|
||||||
|
"next": "Next.js",
|
||||||
|
"nodejs": "Node.js",
|
||||||
|
"notion": "Notion",
|
||||||
|
"nostr": "Nostr",
|
||||||
|
"npm": "npm",
|
||||||
|
"nuxt": "Nuxt",
|
||||||
|
"opera": "Opera",
|
||||||
|
"oracle": "Oracle",
|
||||||
|
"patreon": "Patreon",
|
||||||
|
"paypal": "PayPal",
|
||||||
|
"perl": "Perl",
|
||||||
|
"php": "PHP",
|
||||||
|
"pinterest": "Pinterest",
|
||||||
|
"pix": "PiX",
|
||||||
|
"playstation": "PlayStation",
|
||||||
|
"playwright": "Playwright",
|
||||||
|
"pnpm": "pnpm",
|
||||||
|
"postcss": "PostCSS",
|
||||||
|
"postgresql": "PostgreSQL",
|
||||||
|
"prettier": "Prettier",
|
||||||
|
"prisma": "Prisma",
|
||||||
|
"python": "Python",
|
||||||
|
"qwik": "Qwik",
|
||||||
|
"react": "React",
|
||||||
|
"reddit": "Reddit",
|
||||||
|
"redis": "Redis",
|
||||||
|
"rollup": "Rollup",
|
||||||
|
"rust": "Rust",
|
||||||
|
"safari": "Safari",
|
||||||
|
"sass": "Sass",
|
||||||
|
"scala": "Scala",
|
||||||
|
"scss": "Sass",
|
||||||
|
"semantic": "Semantic UI",
|
||||||
|
"shopify": "Shopify",
|
||||||
|
"skype": "Skype",
|
||||||
|
"slack": "Slack",
|
||||||
|
"solid": "SolidJS",
|
||||||
|
"soundcloud": "SoundCloud",
|
||||||
|
"spotify": "Spotify",
|
||||||
|
"sqlite": "SQLite",
|
||||||
|
"squarespace": "Squarespace",
|
||||||
|
"steam": "Steam",
|
||||||
|
"stripe": "Stripe",
|
||||||
|
"substack": "Substack",
|
||||||
|
"supabase": "Supabase",
|
||||||
|
"surge": "Surge",
|
||||||
|
"svelte": "Svelte",
|
||||||
|
"swift": "Swift",
|
||||||
|
"tailwind": "Tailwind CSS",
|
||||||
|
"telegram": "Telegram",
|
||||||
|
"terraform": "Terraform",
|
||||||
|
"tesla": "Tesla",
|
||||||
|
"tidal": "Tidal",
|
||||||
|
"tiktok": "TikTok",
|
||||||
|
"trello": "Trello",
|
||||||
|
"twitch": "Twitch",
|
||||||
|
"twitter": "Twitter",
|
||||||
|
"typescript": "TypeScript",
|
||||||
|
"unity": "Unity",
|
||||||
|
"unreal": "Unreal Engine",
|
||||||
|
"vercel": "Vercel",
|
||||||
|
"vimeo": "Vimeo",
|
||||||
|
"vite": "Vite",
|
||||||
|
"vitest": "Vitest",
|
||||||
|
"vue": "Vue",
|
||||||
|
"webpack": "Webpack",
|
||||||
|
"wechat": "WeChat",
|
||||||
|
"whatsapp": "WhatsApp",
|
||||||
|
"windows": "Windows",
|
||||||
|
"wix": "Wix",
|
||||||
|
"x.com": "X.com",
|
||||||
|
"x-social": "X.com",
|
||||||
|
"xbox": "Xbox",
|
||||||
|
"yarn": "Yarn",
|
||||||
|
"youtube": "YouTube",
|
||||||
|
"zig": "Zig",
|
||||||
|
"zoom": "Zoom"
|
||||||
|
}
|
||||||
@@ -9,6 +9,7 @@ if (process.env.NODE_ENV === 'development') {
|
|||||||
|
|
||||||
wasm = fs.readFileSync(require.resolve('@resvg/resvg-wasm/index_bg.wasm'));
|
wasm = fs.readFileSync(require.resolve('@resvg/resvg-wasm/index_bg.wasm'));
|
||||||
} else {
|
} else {
|
||||||
|
// @ts-ignore
|
||||||
wasm = resvg_wasm;
|
wasm = resvg_wasm;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
55
docs/.vitepress/api/gh-icon/symmetry/[...data].get.ts
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import { eventHandler, setResponseHeader, defaultContentType } from 'h3';
|
||||||
|
import { renderToString } from 'react-dom/server';
|
||||||
|
import { createElement } from 'react';
|
||||||
|
import Diff from '../../../lib/SvgPreview/Diff.tsx';
|
||||||
|
|
||||||
|
export default eventHandler((event) => {
|
||||||
|
const { params } = event.context;
|
||||||
|
|
||||||
|
const pathData = params.data.split('/');
|
||||||
|
const data = pathData.at(-1).slice(0, -4);
|
||||||
|
const [operation] = pathData;
|
||||||
|
|
||||||
|
const newSrc = Buffer.from(data, 'base64')
|
||||||
|
.toString('utf8')
|
||||||
|
.replaceAll('\n', '')
|
||||||
|
.replace(/<svg[^>]*>|<\/svg>/g, '');
|
||||||
|
|
||||||
|
const width = parseInt(
|
||||||
|
(newSrc.includes('<svg ') ? newSrc.match(/width="(\d+)"/)?.[1] : null) ?? '24',
|
||||||
|
);
|
||||||
|
const height = parseInt(
|
||||||
|
(newSrc.includes('<svg ') ? newSrc.match(/height="(\d+)"/)?.[1] : null) ?? '24',
|
||||||
|
);
|
||||||
|
|
||||||
|
const children = [];
|
||||||
|
|
||||||
|
let oldSrc = '';
|
||||||
|
if (operation.startsWith('rotate-')) {
|
||||||
|
const degrees = parseInt(operation.replace('rotate-', ''));
|
||||||
|
if (isNaN(degrees)) return '';
|
||||||
|
oldSrc = `<g transform="rotate(${degrees} ${width / 2} ${height / 2})">${newSrc}</g>`;
|
||||||
|
} else if (operation === 'flip-horizontal') {
|
||||||
|
oldSrc = `<g transform="scale(1, -1) translate(0, -${height})">${newSrc}</g>`;
|
||||||
|
} else if (operation === 'flip-vertical') {
|
||||||
|
oldSrc = `<g transform="scale(-1, 1) translate(-${width}, 0)">${newSrc}</g>`;
|
||||||
|
} else if (operation === 'flip-backslash') {
|
||||||
|
oldSrc = `<g transform="rotate(90, ${width / 2}, ${height / 2}) scale(1, -1) translate(0, -${height})">${newSrc}</g>`;
|
||||||
|
} else if (operation === 'flip-slash') {
|
||||||
|
oldSrc = `<g transform="rotate(90, ${width / 2}, ${height / 2}) scale(-1, 1) translate(-${width}, 0)">${newSrc}</g>`;
|
||||||
|
} else {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
const svg = Buffer.from(
|
||||||
|
// We can't use jsx here, is not supported here by nitro.
|
||||||
|
renderToString(
|
||||||
|
createElement(Diff, { oldSrc, newSrc, showGrid: true, height, width }, children),
|
||||||
|
),
|
||||||
|
).toString('utf8');
|
||||||
|
|
||||||
|
defaultContentType(event, 'image/svg+xml');
|
||||||
|
setResponseHeader(event, 'Cache-Control', 'public,max-age=31536000');
|
||||||
|
|
||||||
|
return svg;
|
||||||
|
});
|
||||||
@@ -95,9 +95,25 @@
|
|||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"lucide-angular": {
|
"@lucide/angular": {
|
||||||
"order": 6,
|
"order": 6,
|
||||||
"icon": "angular",
|
"icon": "angular",
|
||||||
|
"shields": [
|
||||||
|
{
|
||||||
|
"alt": "npm",
|
||||||
|
"src": "https://img.shields.io/npm/v/@lucide/angular",
|
||||||
|
"href": "https://www.npmjs.com/package/@lucide/angular"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"alt": "npm",
|
||||||
|
"src": "https://img.shields.io/npm/dw/@lucide/angular",
|
||||||
|
"href": "https://www.npmjs.com/package/@lucide/angular"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"lucide-angular": {
|
||||||
|
"order": 7,
|
||||||
|
"icon": "angular",
|
||||||
"shields": [
|
"shields": [
|
||||||
{
|
{
|
||||||
"alt": "npm",
|
"alt": "npm",
|
||||||
@@ -112,7 +128,7 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"lucide-preact": {
|
"lucide-preact": {
|
||||||
"order": 7,
|
"order": 8,
|
||||||
"icon": "preact",
|
"icon": "preact",
|
||||||
"shields": [
|
"shields": [
|
||||||
{
|
{
|
||||||
@@ -130,7 +146,7 @@
|
|||||||
"@lucide/astro": {
|
"@lucide/astro": {
|
||||||
"docsAlias": "lucide-astro",
|
"docsAlias": "lucide-astro",
|
||||||
"packageDirname": "astro",
|
"packageDirname": "astro",
|
||||||
"order": 8,
|
"order": 9,
|
||||||
"icon": "astro",
|
"icon": "astro",
|
||||||
"iconDark": "astro-dark",
|
"iconDark": "astro-dark",
|
||||||
"shields": [
|
"shields": [
|
||||||
@@ -147,7 +163,7 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"lucide-static": {
|
"lucide-static": {
|
||||||
"order": 9,
|
"order": 10,
|
||||||
"icon": "svg",
|
"icon": "svg",
|
||||||
"shields": [
|
"shields": [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { bundledLanguages, getHighlighter, type ThemeRegistration } from 'shikiji';
|
import { bundledLanguages, createHighlighter, type ThemeRegistration } from 'shiki';
|
||||||
|
|
||||||
type CodeExampleType = {
|
type CodeExampleType = {
|
||||||
title: string;
|
title: string;
|
||||||
@@ -9,14 +9,20 @@ type CodeExampleType = {
|
|||||||
const getIconCodes = (): CodeExampleType => {
|
const getIconCodes = (): CodeExampleType => {
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
language: 'js',
|
language: 'html',
|
||||||
title: 'Vanilla',
|
title: 'Vanilla',
|
||||||
code: `\
|
code: `\
|
||||||
import { createIcons, icons } from 'lucide';
|
<script>
|
||||||
|
import { createIcons, $CamelCase } from 'lucide';
|
||||||
|
|
||||||
createIcons({ icons });
|
createIcons({
|
||||||
|
icons: {
|
||||||
|
$CamelCase
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
document.body.append('<i data-lucide="$Name"></i>');\
|
<i data-lucide="$Name"></i>\
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -112,7 +118,7 @@ export type ThemeOptions =
|
|||||||
| { light: ThemeRegistration; dark: ThemeRegistration };
|
| { light: ThemeRegistration; dark: ThemeRegistration };
|
||||||
|
|
||||||
const highLightCode = async (code: string, lang: string, active?: boolean) => {
|
const highLightCode = async (code: string, lang: string, active?: boolean) => {
|
||||||
const highlighter = await getHighlighter({
|
const highlighter = await createHighlighter({
|
||||||
themes: ['github-light', 'github-dark'],
|
themes: ['github-light', 'github-dark'],
|
||||||
langs: Object.keys(bundledLanguages),
|
langs: Object.keys(bundledLanguages),
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { bundledLanguages, type ThemeRegistration } from 'shikiji';
|
import { bundledLanguages, type ThemeRegistration } from 'shiki';
|
||||||
import { getHighlighter } from 'shikiji';
|
import { createHighlighter } from 'shiki';
|
||||||
|
|
||||||
type CodeExampleType = {
|
type CodeExampleType = {
|
||||||
title: string;
|
title: string;
|
||||||
@@ -10,10 +10,11 @@ type CodeExampleType = {
|
|||||||
const getIconCodes = (): CodeExampleType => {
|
const getIconCodes = (): CodeExampleType => {
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
language: 'js',
|
language: 'html',
|
||||||
title: 'Vanilla',
|
title: 'Vanilla',
|
||||||
code: `\
|
code: `\
|
||||||
import { createIcons, icons } from 'lucide';
|
<script>
|
||||||
|
import { createIcons } from 'lucide';
|
||||||
import { $CamelCase } from '@lucide/lab';
|
import { $CamelCase } from '@lucide/lab';
|
||||||
|
|
||||||
createIcons({
|
createIcons({
|
||||||
@@ -21,8 +22,9 @@ createIcons({
|
|||||||
$CamelCase
|
$CamelCase
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
document.body.append('<i data-lucide="$Name"></i>');\
|
<i data-lucide="$Name"></i>\
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -119,7 +121,7 @@ export type ThemeOptions =
|
|||||||
| { light: ThemeRegistration; dark: ThemeRegistration };
|
| { light: ThemeRegistration; dark: ThemeRegistration };
|
||||||
|
|
||||||
const highLightCode = async (code: string, lang: string, active?: boolean) => {
|
const highLightCode = async (code: string, lang: string, active?: boolean) => {
|
||||||
const highlighter = await getHighlighter({
|
const highlighter = await createHighlighter({
|
||||||
themes: ['github-light', 'github-dark'],
|
themes: ['github-light', 'github-dark'],
|
||||||
langs: Object.keys(bundledLanguages),
|
langs: Object.keys(bundledLanguages),
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
import { bundledLanguages, type ThemeRegistration } from 'shikiji';
|
import { bundledLanguages, type ThemeRegistration } from 'shiki';
|
||||||
import { getHighlighter } from 'shikiji';
|
import { createHighlighter } from 'shiki';
|
||||||
|
|
||||||
export type ThemeOptions =
|
export type ThemeOptions =
|
||||||
| ThemeRegistration
|
| ThemeRegistration
|
||||||
| { light: ThemeRegistration; dark: ThemeRegistration };
|
| { light: ThemeRegistration; dark: ThemeRegistration };
|
||||||
|
|
||||||
const highLightCode = async (code: string, lang: string, active?: boolean) => {
|
const highLightCode = async (code: string, lang: string, active?: boolean) => {
|
||||||
const highlighter = await getHighlighter({
|
const highlighter = await createHighlighter({
|
||||||
themes: ['github-light', 'github-dark'],
|
themes: ['github-light', 'github-dark'],
|
||||||
langs: Object.keys(bundledLanguages),
|
langs: Object.keys(bundledLanguages),
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { createLucideIcon } from 'lucide-react/src/lucide-react';
|
import { createLucideIcon } from 'lucide-react/src/lucide-react';
|
||||||
import { type LucideProps, type IconNode } from 'lucide-react/src/createLucideIcon';
|
import { type LucideProps, type IconNode } from 'lucide-react/src/types';
|
||||||
import { IconEntity } from '../theme/types';
|
import { IconEntity } from '../theme/types';
|
||||||
import { renderToStaticMarkup } from 'react-dom/server';
|
import { renderToStaticMarkup } from 'react-dom/server';
|
||||||
import { IconContent } from './generateZip';
|
import { IconContent } from './generateZip';
|
||||||
|
|||||||
@@ -69,39 +69,39 @@ const sidebar: UserConfig<DefaultTheme.Config>['themeConfig']['sidebar'] = {
|
|||||||
link: '/guide/packages/lucide',
|
link: '/guide/packages/lucide',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide React',
|
text: 'React',
|
||||||
link: '/guide/packages/lucide-react',
|
link: '/guide/packages/lucide-react',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide Vue',
|
text: 'Vue',
|
||||||
link: '/guide/packages/lucide-vue-next',
|
link: '/guide/packages/lucide-vue-next',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide Svelte',
|
text: 'Svelte',
|
||||||
link: '/guide/packages/lucide-svelte',
|
link: '/guide/packages/lucide-svelte',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide Solid',
|
text: 'Solid',
|
||||||
link: '/guide/packages/lucide-solid',
|
link: '/guide/packages/lucide-solid',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide React Native',
|
text: 'React Native',
|
||||||
link: '/guide/packages/lucide-react-native',
|
link: '/guide/packages/lucide-react-native',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide Angular',
|
text: 'Angular',
|
||||||
link: '/guide/packages/lucide-angular',
|
link: '/guide/packages/angular',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide Preact',
|
text: 'Preact',
|
||||||
link: '/guide/packages/lucide-preact',
|
link: '/guide/packages/lucide-preact',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide Astro',
|
text: 'Astro',
|
||||||
link: '/guide/packages/lucide-astro',
|
link: '/guide/packages/lucide-astro',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: 'Lucide Static',
|
text: 'Static',
|
||||||
link: '/guide/packages/lucide-static',
|
link: '/guide/packages/lucide-static',
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -1,23 +1,31 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { useData } from 'vitepress';
|
||||||
import { computed } from 'vue';
|
import { computed } from 'vue';
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
modelValue: string
|
modelValue: string;
|
||||||
id: string
|
id: string;
|
||||||
}>()
|
}>();
|
||||||
|
|
||||||
const emit = defineEmits(['update:modelValue'])
|
const { isDark } = useData();
|
||||||
|
|
||||||
|
const emit = defineEmits(['update:modelValue']);
|
||||||
|
|
||||||
const value = computed({
|
const value = computed({
|
||||||
get: () => props.modelValue,
|
get: () => {
|
||||||
set: (val) => emit('update:modelValue', val)
|
if (props.modelValue == null || props.modelValue === 'currentColor') {
|
||||||
})
|
return isDark.value ? '#ffffff' : '#000000';
|
||||||
|
}
|
||||||
|
|
||||||
|
return props.modelValue;
|
||||||
|
},
|
||||||
|
set: (val) => emit('update:modelValue', val),
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="color-picker">
|
<div class="color-picker">
|
||||||
<div class="color-input-wrapper">
|
<div class="color-input-wrapper">
|
||||||
<!-- TODO: Add currentColor div if value is currentColor -->
|
|
||||||
<input
|
<input
|
||||||
type="color"
|
type="color"
|
||||||
:id="id"
|
:id="id"
|
||||||
@@ -33,6 +41,7 @@ const value = computed({
|
|||||||
class="color-input-text"
|
class="color-input-text"
|
||||||
aria-label="Color picker input"
|
aria-label="Color picker input"
|
||||||
v-model="value"
|
v-model="value"
|
||||||
|
placeholder="[default]"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -45,27 +54,33 @@ const value = computed({
|
|||||||
top: -5px;
|
top: -5px;
|
||||||
left: -5px;
|
left: -5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-input-wrapper {
|
.color-input-wrapper {
|
||||||
height: 24px;
|
height: 24px;
|
||||||
width: 24px;
|
width: 24px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
position: relative;
|
position: relative;
|
||||||
border-radius: 12px;
|
border-radius: 4px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-picker {
|
.color-picker {
|
||||||
background: var(--color-picker-bg, var(--vp-c-bg-alt));
|
background: var(--color-picker-bg, var(--vp-c-bg-alt));
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
color: var(--vp-c-text-2);
|
color: var(--vp-c-text-2);
|
||||||
padding: 4px 8px;
|
padding: 3px 8px 3px 3px;
|
||||||
height: auto;
|
height: auto;
|
||||||
font-size: 14px;
|
font-size: 13px;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
cursor: text;
|
cursor: text;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 2px;
|
gap: 2px;
|
||||||
|
transition:
|
||||||
|
color 0.25s,
|
||||||
|
border-color 0.25s,
|
||||||
|
background-color 0.25s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-input-text {
|
.color-input-text {
|
||||||
@@ -75,19 +90,22 @@ const value = computed({
|
|||||||
border: none;
|
border: none;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
color: var(--vp-c-text-1);
|
color: var(--vp-c-text-1);
|
||||||
font-size: 14px;
|
font-size: 13px;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
cursor: text;
|
cursor: text;
|
||||||
transition: border-color 0.25s, background 0.4s ease;
|
transition:
|
||||||
|
border-color 0.25s,
|
||||||
|
background 0.4s ease;
|
||||||
|
letter-spacing: 1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-picker:hover, .color-picker:focus {
|
.color-picker:hover,
|
||||||
|
.color-picker:focus {
|
||||||
border-color: var(--vp-c-brand);
|
border-color: var(--vp-c-brand);
|
||||||
background: var(--vp-c-bg-alt);
|
background: var(--vp-c-bg-alt);
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-input[value="currentColor"] {
|
.color-input[value='currentColor'] {
|
||||||
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,22 +1,27 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import createLucideIcon from 'lucide-vue-next/src/createLucideIcon'
|
import Icon from 'lucide-vue-next/src/Icon';
|
||||||
import { search } from '../../../data/iconNodes'
|
import { search } from '../../../data/iconNodes';
|
||||||
|
|
||||||
const SearchIcon = createLucideIcon('search', search)
|
|
||||||
|
|
||||||
defineProps({
|
defineProps({
|
||||||
shortcut: {
|
shortcut: {
|
||||||
type: String,
|
type: String,
|
||||||
required: false
|
required: false,
|
||||||
}
|
},
|
||||||
})
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<button class="fake-input">
|
<button class="fake-input">
|
||||||
<component :is="SearchIcon" class="search-icon"/>
|
<Icon
|
||||||
<slot/>
|
:iconNode="search"
|
||||||
<kbd v-if="shortcut" class="shortcut">{{ shortcut }}</kbd>
|
class="search-icon"
|
||||||
|
/>
|
||||||
|
<slot />
|
||||||
|
<kbd
|
||||||
|
v-if="shortcut"
|
||||||
|
class="shortcut"
|
||||||
|
>{{ shortcut }}</kbd
|
||||||
|
>
|
||||||
</button>
|
</button>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -34,10 +39,14 @@ defineProps({
|
|||||||
cursor: text;
|
cursor: text;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
transition: color 0.25s, border-color 0.25s, background-color 0.25s;
|
transition:
|
||||||
|
color 0.25s,
|
||||||
|
border-color 0.25s,
|
||||||
|
background-color 0.25s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fake-input:hover, .fake-input:focus {
|
.fake-input:hover,
|
||||||
|
.fake-input:focus {
|
||||||
border-color: var(--vp-c-brand);
|
border-color: var(--vp-c-brand);
|
||||||
background: var(--vp-c-bg-alt);
|
background: var(--vp-c-bg-alt);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|
||||||
.icon-button {
|
.icon-button {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
@@ -30,9 +29,9 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.icon-button:active {
|
.icon-button:active {
|
||||||
border-color: var(--vp-button-alt-active-border);
|
border-color: var(--vp-button-alt-active-border);
|
||||||
color: var(--vp-button-alt-active-text);
|
color: var(--vp-button-alt-active-text);
|
||||||
background-color: var(--vp-button-alt-active-bg);
|
background-color: var(--vp-button-alt-active-bg);
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon-button.active {
|
.icon-button.active {
|
||||||
|
|||||||
@@ -1,60 +1,90 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
export default {
|
export default {
|
||||||
inheritAttrs: false,
|
inheritAttrs: false,
|
||||||
}
|
};
|
||||||
|
|
||||||
export interface InputProps {
|
export interface InputProps {
|
||||||
type: string
|
type: string;
|
||||||
modelValue: string
|
modelValue: string;
|
||||||
shortcut?: string
|
shortcut?: string;
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, onMounted, nextTick, watch } from 'vue'
|
import { ref, onMounted, nextTick, watch } from 'vue';
|
||||||
|
import Icon from 'lucide-vue-next/src/Icon';
|
||||||
|
import { x } from '../../../data/iconNodes';
|
||||||
|
import IconButton from './IconButton.vue';
|
||||||
|
|
||||||
const props = withDefaults(defineProps<InputProps>(), {
|
const props = withDefaults(defineProps<InputProps>(), {
|
||||||
type: 'text'
|
type: 'text',
|
||||||
})
|
});
|
||||||
|
|
||||||
const input = ref()
|
const input = ref();
|
||||||
const wrapperEl = ref()
|
const wrapperEl = ref();
|
||||||
const shortcutEl = ref()
|
const shortcutEl = ref();
|
||||||
|
|
||||||
defineEmits(['change', 'input', 'update:modelValue'])
|
const emit = defineEmits(['change', 'input', 'update:modelValue']);
|
||||||
|
|
||||||
const updateShortcutSpacing = () => {
|
const updateShortcutSpacing = () => {
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
if (shortcutEl.value && wrapperEl.value) {
|
if (shortcutEl.value && wrapperEl.value) {
|
||||||
const shortcutWidth = shortcutEl.value.offsetWidth
|
const shortcutWidth = shortcutEl.value.offsetWidth;
|
||||||
wrapperEl.value.style.setProperty('--shortcut-width', `${shortcutWidth}px`)
|
wrapperEl.value.style.setProperty('--shortcut-width', `${shortcutWidth}px`);
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
}
|
};
|
||||||
|
|
||||||
onMounted(updateShortcutSpacing)
|
onMounted(updateShortcutSpacing);
|
||||||
watch(() => props.shortcut, updateShortcutSpacing)
|
watch(() => props.shortcut, updateShortcutSpacing);
|
||||||
|
|
||||||
|
function onClear() {
|
||||||
|
emit('update:modelValue', '');
|
||||||
|
input.value.focus();
|
||||||
|
}
|
||||||
|
|
||||||
defineExpose({
|
defineExpose({
|
||||||
focus: () => {
|
focus: () => {
|
||||||
input.value.focus()
|
input.value.focus();
|
||||||
}
|
},
|
||||||
})
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="input-wrapper" ref="wrapperEl">
|
<div
|
||||||
<slot name="icon" class="icon" />
|
class="input-wrapper"
|
||||||
|
ref="wrapperEl"
|
||||||
|
>
|
||||||
|
<slot
|
||||||
|
name="icon"
|
||||||
|
class="icon"
|
||||||
|
/>
|
||||||
<input
|
<input
|
||||||
:type="type"
|
:type="type"
|
||||||
class="input"
|
class="input"
|
||||||
:class="{'has-icon': $slots.icon, 'has-shortcut': shortcut}"
|
:class="{ 'has-icon': $slots.icon, 'has-shortcut': shortcut }"
|
||||||
ref="input"
|
ref="input"
|
||||||
:value="modelValue"
|
:value="modelValue"
|
||||||
v-bind="$attrs"
|
v-bind="$attrs"
|
||||||
@input="$emit('update:modelValue', $event.target.value)"
|
@input="$emit('update:modelValue', $event.target.value)"
|
||||||
/>
|
/>
|
||||||
<kbd v-if="shortcut" class="shortcut" ref="shortcutEl">{{ shortcut }}</kbd>
|
<IconButton
|
||||||
|
@click="onClear"
|
||||||
|
v-if="type === 'search' && modelValue"
|
||||||
|
class="clear-button"
|
||||||
|
aria-label="Clear input"
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
:iconNode="x"
|
||||||
|
:size="20"
|
||||||
|
/>
|
||||||
|
</IconButton>
|
||||||
|
<kbd
|
||||||
|
v-if="shortcut"
|
||||||
|
class="shortcut"
|
||||||
|
ref="shortcutEl"
|
||||||
|
>{{ shortcut }}</kbd
|
||||||
|
>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -62,6 +92,7 @@ defineExpose({
|
|||||||
.input-wrapper {
|
.input-wrapper {
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
.input {
|
.input {
|
||||||
justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
@@ -71,13 +102,18 @@ defineExpose({
|
|||||||
height: 40px;
|
height: 40px;
|
||||||
background-color: var(--vp-c-bg-alt);
|
background-color: var(--vp-c-bg-alt);
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
|
transition:
|
||||||
|
color 0.25s,
|
||||||
|
border-color 0.25s,
|
||||||
|
background-color 0.25s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.input.has-shortcut {
|
.input.has-shortcut {
|
||||||
padding-right: calc(var(--shortcut-width, 40px) + 22px);
|
padding-right: calc(var(--shortcut-width, 40px) + 22px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.input:hover, .input:focus {
|
.input:hover,
|
||||||
|
.input:focus {
|
||||||
border-color: var(--vp-c-brand);
|
border-color: var(--vp-c-brand);
|
||||||
background: var(--vp-c-bg-alt);
|
background: var(--vp-c-bg-alt);
|
||||||
}
|
}
|
||||||
@@ -86,6 +122,14 @@ defineExpose({
|
|||||||
padding-left: 52px;
|
padding-left: 52px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.clear-button {
|
||||||
|
position: absolute;
|
||||||
|
right: 56px;
|
||||||
|
top: 9px;
|
||||||
|
padding: 4px;
|
||||||
|
transition: background-color .25s;
|
||||||
|
}
|
||||||
|
|
||||||
.shortcut {
|
.shortcut {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
right: 12px;
|
right: 12px;
|
||||||
@@ -111,7 +155,7 @@ defineExpose({
|
|||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.input-wrapper svg {
|
.input-wrapper > svg {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
left: 16px;
|
left: 16px;
|
||||||
top: 12px;
|
top: 12px;
|
||||||
|
|||||||
@@ -1,38 +1,36 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
export default {
|
export default {
|
||||||
inheritAttrs: false,
|
inheritAttrs: false,
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, ref } from 'vue'
|
import { computed, ref } from 'vue';
|
||||||
import Input from './Input.vue'
|
import Input from './Input.vue';
|
||||||
import createLucideIcon from 'lucide-vue-next/src/createLucideIcon'
|
import Icon from 'lucide-vue-next/src/Icon';
|
||||||
import { search } from '../../../data/iconNodes'
|
import { search } from '../../../data/iconNodes';
|
||||||
|
|
||||||
const SearchIcon = createLucideIcon('search', search)
|
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
modelValue: string
|
modelValue: string;
|
||||||
shortcut?: string
|
shortcut?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const props = defineProps<Props>()
|
const props = defineProps<Props>();
|
||||||
|
|
||||||
const input = ref()
|
const input = ref();
|
||||||
|
|
||||||
const emit = defineEmits(['update:modelValue'])
|
const emit = defineEmits(['update:modelValue']);
|
||||||
|
|
||||||
defineExpose({
|
defineExpose({
|
||||||
focus: () => {
|
focus: () => {
|
||||||
input.value.focus()
|
input.value.focus();
|
||||||
}
|
},
|
||||||
})
|
});
|
||||||
|
|
||||||
const value = computed({
|
const value = computed({
|
||||||
get: () => props.modelValue,
|
get: () => props.modelValue,
|
||||||
set: (val) => emit('update:modelValue', val)
|
set: (val) => emit('update:modelValue', val),
|
||||||
})
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -46,7 +44,10 @@ const value = computed({
|
|||||||
class="input-wrapper"
|
class="input-wrapper"
|
||||||
>
|
>
|
||||||
<template #icon>
|
<template #icon>
|
||||||
<component :is="SearchIcon" class="search-icon" />
|
<Icon
|
||||||
|
:iconNode="search"
|
||||||
|
class="search-icon"
|
||||||
|
/>
|
||||||
</template>
|
</template>
|
||||||
</Input>
|
</Input>
|
||||||
</template>
|
</template>
|
||||||
@@ -62,7 +63,8 @@ const value = computed({
|
|||||||
background-color: var(--vp-c-bg-alt);
|
background-color: var(--vp-c-bg-alt);
|
||||||
}
|
}
|
||||||
|
|
||||||
.input:hover, .input:focus {
|
.input:hover,
|
||||||
|
.input:focus {
|
||||||
border-color: var(--vp-c-brand);
|
border-color: var(--vp-c-brand);
|
||||||
background: var(--vp-c-bg-alt);
|
background: var(--vp-c-bg-alt);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,15 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { rotateCw } from '../../../data/iconNodes'
|
import { rotateCw } from '../../../data/iconNodes';
|
||||||
import createLucideIcon from 'lucide-vue-next/src/createLucideIcon'
|
import Icon from 'lucide-vue-next/src/Icon';
|
||||||
import IconButton from "./IconButton.vue";
|
import IconButton from './IconButton.vue';
|
||||||
|
|
||||||
const RotateIcon = createLucideIcon('RotateIcon', rotateCw)
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<IconButton class="reset-button">
|
<IconButton class="reset-button">
|
||||||
<RotateIcon :size="20"/>
|
<Icon
|
||||||
|
:size="20"
|
||||||
|
:iconNode="rotateCw"
|
||||||
|
/>
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -32,6 +33,7 @@ const RotateIcon = createLucideIcon('RotateIcon', rotateCw)
|
|||||||
0% {
|
0% {
|
||||||
transform: rotate(0deg);
|
transform: rotate(0deg);
|
||||||
}
|
}
|
||||||
|
|
||||||
100% {
|
100% {
|
||||||
transform: rotate(359deg);
|
transform: rotate(359deg);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,18 +22,18 @@ export default {
|
|||||||
logo: '/framework-logos/svelte.svg',
|
logo: '/framework-logos/svelte.svg',
|
||||||
label: 'Lucide documentation for Svelte',
|
label: 'Lucide documentation for Svelte',
|
||||||
},
|
},
|
||||||
{
|
|
||||||
name: 'lucide-preact',
|
|
||||||
logo: '/framework-logos/preact.svg',
|
|
||||||
label: 'Lucide documentation for Preact',
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
name: 'lucide-solid',
|
name: 'lucide-solid',
|
||||||
logo: '/framework-logos/solid.svg',
|
logo: '/framework-logos/solid.svg',
|
||||||
label: 'Lucide documentation for Solid',
|
label: 'Lucide documentation for Solid',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'lucide-angular',
|
name: 'lucide-preact',
|
||||||
|
logo: '/framework-logos/preact.svg',
|
||||||
|
label: 'Lucide documentation for Preact',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'angular',
|
||||||
logo: '/framework-logos/angular.svg',
|
logo: '/framework-logos/angular.svg',
|
||||||
label: 'Lucide documentation for Angular',
|
label: 'Lucide documentation for Angular',
|
||||||
},
|
},
|
||||||
@@ -48,11 +48,6 @@ export default {
|
|||||||
logo: '/framework-logos/react-native.svg',
|
logo: '/framework-logos/react-native.svg',
|
||||||
label: 'Lucide documentation for React Native',
|
label: 'Lucide documentation for React Native',
|
||||||
},
|
},
|
||||||
{
|
|
||||||
name: 'lucide-flutter',
|
|
||||||
logo: '/framework-logos/flutter.svg',
|
|
||||||
label: 'Lucide documentation for Flutter',
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -2,45 +2,48 @@
|
|||||||
import { useData } from 'vitepress';
|
import { useData } from 'vitepress';
|
||||||
import { useSessionStorage } from '@vueuse/core';
|
import { useSessionStorage } from '@vueuse/core';
|
||||||
import IconButton from '../base/IconButton.vue';
|
import IconButton from '../base/IconButton.vue';
|
||||||
import VPDocAsideCarbonAds from 'vitepress/dist/client/theme-default/components/VPDocAsideCarbonAds.vue'
|
import VPDocAsideCarbonAds from 'vitepress/dist/client/theme-default/components/VPDocAsideCarbonAds.vue';
|
||||||
import { x } from '../../../data/iconNodes'
|
import { x } from '../../../data/iconNodes';
|
||||||
import createLucideIcon from 'lucide-vue-next/src/createLucideIcon';
|
import Icon from 'lucide-vue-next/src/Icon';
|
||||||
import { onMounted, ref } from 'vue';
|
import { onMounted, ref } from 'vue';
|
||||||
|
|
||||||
const { theme } = useData()
|
const { theme } = useData();
|
||||||
const showAd = useSessionStorage('show-carbon-ads', true)
|
const showAd = useSessionStorage('show-carbon-ads', true);
|
||||||
const carbonLoaded = ref(true)
|
const carbonLoaded = ref(true);
|
||||||
|
|
||||||
defineProps<{
|
defineProps<{
|
||||||
drawerOpen: boolean
|
drawerOpen: boolean;
|
||||||
}>()
|
}>();
|
||||||
|
|
||||||
const CloseIcon = createLucideIcon('Close', x)
|
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
if (window?._carbonads == null) {
|
if (window?._carbonads == null) {
|
||||||
carbonLoaded.value = false
|
carbonLoaded.value = false;
|
||||||
}
|
}
|
||||||
}, 5000)
|
}, 5000);
|
||||||
})
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
:class="{
|
:class="{
|
||||||
'drawer-open': drawerOpen,
|
'drawer-open': drawerOpen,
|
||||||
'hide-ad': !(showAd && carbonLoaded)
|
'hide-ad': !(showAd && carbonLoaded),
|
||||||
}"
|
}"
|
||||||
class="floating-ad"
|
class="floating-ad"
|
||||||
v-if="theme.carbonAds"
|
v-if="theme.carbonAds"
|
||||||
>
|
>
|
||||||
<IconButton @click="showAd = false" class="hide-button">
|
<IconButton
|
||||||
<component :is="CloseIcon" :size="20" absoluteStrokeWidth />
|
@click="showAd = false"
|
||||||
|
class="hide-button"
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
:iconNode="x"
|
||||||
|
:size="20"
|
||||||
|
absoluteStrokeWidth
|
||||||
|
/>
|
||||||
</IconButton>
|
</IconButton>
|
||||||
<VPDocAsideCarbonAds
|
<VPDocAsideCarbonAds :carbon-ads="theme.carbonAds" />
|
||||||
:carbon-ads="theme.carbonAds"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -51,7 +54,9 @@ onMounted(() => {
|
|||||||
bottom: 32px;
|
bottom: 32px;
|
||||||
width: 224px;
|
width: 224px;
|
||||||
right: 32px;
|
right: 32px;
|
||||||
transition: opacity 0.5s, transform 0.25s ease;
|
transition:
|
||||||
|
opacity 0.5s,
|
||||||
|
transform 0.25s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.floating-ad.drawer-open {
|
.floating-ad.drawer-open {
|
||||||
@@ -67,8 +72,11 @@ onMounted(() => {
|
|||||||
transform: translateY(-288px) translateX(224px);
|
transform: translateY(-288px) translateX(224px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.floating-ad.drawer-open, .floating-ad.hide-ad {
|
.floating-ad.drawer-open,
|
||||||
transition: opacity 0.25s, transform 0.5s cubic-bezier(0.19, 1, 0.22, 1);
|
.floating-ad.hide-ad {
|
||||||
|
transition:
|
||||||
|
opacity 0.25s,
|
||||||
|
transform 0.5s cubic-bezier(0.19, 1, 0.22, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 1280px) {
|
@media (min-width: 1280px) {
|
||||||
|
|||||||
@@ -1,70 +1,68 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref } from 'vue';
|
import { ref } from 'vue';
|
||||||
import ButtonMenu from '../base/ButtonMenu.vue'
|
import ButtonMenu from '../base/ButtonMenu.vue';
|
||||||
import { useIconStyleContext } from '../../composables/useIconStyle';
|
import { useIconStyleContext } from '../../composables/useIconStyle';
|
||||||
import useConfetti from '../../composables/useConfetti';
|
import useConfetti from '../../composables/useConfetti';
|
||||||
import getSVGIcon from '../../utils/getSVGIcon';
|
import getSVGIcon from '../../utils/getSVGIcon';
|
||||||
import downloadData from '../../utils/downloadData';
|
import downloadData from '../../utils/downloadData';
|
||||||
|
|
||||||
const downloadText = 'Download!'
|
const downloadText = 'Download!';
|
||||||
const copiedText = 'Copied!'
|
const copiedText = 'Copied!';
|
||||||
const confettiText = ref(copiedText)
|
const confettiText = ref(copiedText);
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
name: string
|
name: string;
|
||||||
popoverPosition?: 'top' | 'bottom'
|
popoverPosition?: 'top' | 'bottom';
|
||||||
}>()
|
}>();
|
||||||
|
|
||||||
const { size } = useIconStyleContext()
|
const { size } = useIconStyleContext();
|
||||||
|
|
||||||
const { animate, confetti } = useConfetti()
|
const { animate, confetti } = useConfetti();
|
||||||
|
|
||||||
function copySVG() {
|
function copySVG() {
|
||||||
confettiText.value = copiedText
|
confettiText.value = copiedText;
|
||||||
const svgString = getSVGIcon()
|
const svgString = getSVGIcon();
|
||||||
|
|
||||||
navigator.clipboard.writeText(svgString)
|
navigator.clipboard.writeText(svgString);
|
||||||
|
|
||||||
confetti()
|
confetti();
|
||||||
}
|
}
|
||||||
|
|
||||||
function copyDataUrl() {
|
function copyDataUrl() {
|
||||||
confettiText.value = copiedText
|
confettiText.value = copiedText;
|
||||||
const svgString = getSVGIcon()
|
const svgString = getSVGIcon();
|
||||||
|
|
||||||
// Create SVG data url
|
// Create SVG data url
|
||||||
const dataUrl = `data:image/svg+xml;base64,${btoa(svgString)}`
|
const dataUrl = `data:image/svg+xml;base64,${btoa(svgString)}`;
|
||||||
navigator.clipboard.writeText(dataUrl)
|
navigator.clipboard.writeText(dataUrl);
|
||||||
|
|
||||||
confetti()
|
confetti();
|
||||||
}
|
}
|
||||||
|
|
||||||
function downloadSVG() {
|
function downloadSVG() {
|
||||||
confettiText.value = downloadText
|
confettiText.value = downloadText;
|
||||||
const svgString = getSVGIcon()
|
const svgString = getSVGIcon();
|
||||||
|
|
||||||
downloadData(`${props.name}.svg`, `data:image/svg+xml;base64,${btoa(svgString)}`)
|
downloadData(`${props.name}.svg`, `data:image/svg+xml;base64,${btoa(svgString)}`);
|
||||||
confetti()
|
confetti();
|
||||||
}
|
}
|
||||||
|
|
||||||
function downloadPNG() {
|
function downloadPNG() {
|
||||||
confettiText.value = downloadText
|
confettiText.value = downloadText;
|
||||||
const svgString = getSVGIcon()
|
const svgString = getSVGIcon();
|
||||||
|
|
||||||
const canvas = document.createElement('canvas');
|
const canvas = document.createElement('canvas');
|
||||||
canvas.width = size.value;
|
canvas.width = size.value;
|
||||||
canvas.height = size.value;
|
canvas.height = size.value;
|
||||||
const ctx = canvas.getContext("2d");
|
const ctx = canvas.getContext('2d');
|
||||||
|
|
||||||
const image = new Image();
|
const image = new Image();
|
||||||
image.src = `data:image/svg+xml;base64,${btoa(svgString)}`;
|
image.src = `data:image/svg+xml;base64,${btoa(svgString)}`;
|
||||||
image.onload = function() {
|
image.onload = function () {
|
||||||
ctx.drawImage(image, 0, 0);
|
ctx.drawImage(image, 0, 0);
|
||||||
downloadData(`${props.name}.png`, canvas.toDataURL('image/png'))
|
downloadData(`${props.name}.png`, canvas.toDataURL('image/png'));
|
||||||
confetti()
|
confetti();
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -75,10 +73,10 @@ function downloadPNG() {
|
|||||||
:data-confetti-text="confettiText"
|
:data-confetti-text="confettiText"
|
||||||
:popoverPosition="popoverPosition"
|
:popoverPosition="popoverPosition"
|
||||||
:options="[
|
:options="[
|
||||||
{ text: 'Copy SVG' , onClick: copySVG },
|
{ text: 'Copy SVG', onClick: copySVG },
|
||||||
{ text: 'Copy Data URL' , onClick: copyDataUrl },
|
{ text: 'Copy Data URL', onClick: copyDataUrl },
|
||||||
{ text: 'Download SVG' , onClick: downloadSVG },
|
{ text: 'Download SVG', onClick: downloadSVG },
|
||||||
{ text: 'Download PNG' , onClick: downloadPNG },
|
{ text: 'Download PNG', onClick: downloadPNG },
|
||||||
]"
|
]"
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,41 +1,44 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, useSlots } from 'vue';
|
import { computed, useSlots } from 'vue';
|
||||||
import createLucideIcon from 'lucide-vue-next/src/createLucideIcon'
|
import { copy } from '../../../data/iconNodes';
|
||||||
import { copy } from '../../../data/iconNodes'
|
|
||||||
import useConfetti from '../../composables/useConfetti';
|
import useConfetti from '../../composables/useConfetti';
|
||||||
const { animate, confetti } = useConfetti()
|
import Icon from 'lucide-vue-next/src/Icon';
|
||||||
const slots = useSlots()
|
const { animate, confetti } = useConfetti();
|
||||||
|
const slots = useSlots();
|
||||||
|
|
||||||
const copiedText = computed(() => slots.default?.()[0].children)
|
const copiedText = computed(() => slots.default?.()[0].children);
|
||||||
|
|
||||||
function copyText() {
|
function copyText() {
|
||||||
navigator.clipboard.writeText(copiedText.value)
|
navigator.clipboard.writeText(copiedText.value);
|
||||||
|
|
||||||
confetti()
|
confetti();
|
||||||
}
|
}
|
||||||
|
|
||||||
const Copy = createLucideIcon('ChevronUp', copy)
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<h1
|
<h1
|
||||||
class="icon-name confetti-button"
|
class="icon-name confetti-button"
|
||||||
:class="{animate}"
|
:class="{ animate }"
|
||||||
data-confetti-text="Copied!"
|
data-confetti-text="Copied!"
|
||||||
@click="copyText"
|
@click="copyText"
|
||||||
>
|
>
|
||||||
<slot />
|
<slot />
|
||||||
<Copy :size="20" class="copy-icon"/>
|
<Icon
|
||||||
|
:iconNode="copy"
|
||||||
|
:size="20"
|
||||||
|
class="copy-icon"
|
||||||
|
/>
|
||||||
</h1>
|
</h1>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
@import './confetti.css';
|
@import './confetti.css';
|
||||||
|
|
||||||
.icon-name {
|
.icon-name {
|
||||||
font-size: 24px;
|
font-size: 24px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
line-height: 32px;
|
line-height: 32px;
|
||||||
transition: background ease-in .15s;;
|
transition: background ease-in 0.15s;
|
||||||
padding: 2px 8px;
|
padding: 2px 8px;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
width: auto;
|
width: auto;
|
||||||
@@ -48,7 +51,7 @@ const Copy = createLucideIcon('ChevronUp', copy)
|
|||||||
}
|
}
|
||||||
|
|
||||||
.icon-name:hover .copy-icon {
|
.icon-name:hover .copy-icon {
|
||||||
opacity: .9;
|
opacity: 0.9;
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon-name:before,
|
.icon-name:before,
|
||||||
@@ -65,10 +68,10 @@ const Copy = createLucideIcon('ChevronUp', copy)
|
|||||||
opacity: 0;
|
opacity: 0;
|
||||||
margin-left: 12px;
|
margin-left: 12px;
|
||||||
margin-top: 6px;
|
margin-top: 6px;
|
||||||
transition:ease .3s opacity;
|
transition: ease 0.3s opacity;
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon-name:hover .copy-icon:hover {
|
.icon-name:hover .copy-icon:hover {
|
||||||
opacity: .6;
|
opacity: 0.6;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,19 +1,19 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, computed, defineAsyncComponent, onMounted, watch, watchEffect } from 'vue';
|
import { ref, computed, defineAsyncComponent, onMounted } from 'vue';
|
||||||
import type { IconEntity, Category } from '../../types';
|
import type { IconEntity, Category } from '../../types';
|
||||||
import useSearch from '../../composables/useSearch';
|
import useSearch from '../../composables/useSearch';
|
||||||
import InputSearch from '../base/InputSearch.vue';
|
import InputSearch from '../base/InputSearch.vue';
|
||||||
import useSearchInput from '../../composables/useSearchInput';
|
import useSearchInput from '../../composables/useSearchInput';
|
||||||
import useSearchShortcut from '../../utils/useSearchShortcut';
|
import useSearchShortcut from '../../utils/useSearchShortcut';
|
||||||
import StickyBar from './StickyBar.vue';
|
import StickyBar from './StickyBar.vue';
|
||||||
import IconsCategory from './IconsCategory.vue';
|
import IconsCategory, { CategoryRow } from './IconsCategory.vue';
|
||||||
import useFetchTags from '../../composables/useFetchTags';
|
import useFetchTags from '../../composables/useFetchTags';
|
||||||
import useFetchCategories from '../../composables/useFetchCategories';
|
import useFetchCategories from '../../composables/useFetchCategories';
|
||||||
import { useElementSize, useEventListener, useVirtualList } from '@vueuse/core';
|
import { useElementSize, useEventListener, useVirtualList } from '@vueuse/core';
|
||||||
import chunkArray from '../../utils/chunkArray';
|
import chunkArray from '../../utils/chunkArray';
|
||||||
import { CategoryRow } from './IconsCategory.vue';
|
|
||||||
import useScrollToCategory from '../../composables/useScrollToCategory';
|
import useScrollToCategory from '../../composables/useScrollToCategory';
|
||||||
import CarbonAdOverlay from './CarbonAdOverlay.vue';
|
import CarbonAdOverlay from './CarbonAdOverlay.vue';
|
||||||
|
import useSearchPlaceholder from '../../utils/useSearchPlaceholder.ts';
|
||||||
|
|
||||||
const ICON_SIZE = 56;
|
const ICON_SIZE = 56;
|
||||||
const ICON_GRID_GAP = 8;
|
const ICON_GRID_GAP = 8;
|
||||||
@@ -40,10 +40,10 @@ const { execute: fetchTags, data: tags } = useFetchTags();
|
|||||||
const { execute: fetchCategories, data: categoriesMap } = useFetchCategories();
|
const { execute: fetchCategories, data: categoriesMap } = useFetchCategories();
|
||||||
|
|
||||||
const overviewEl = ref<HTMLElement | null>(null);
|
const overviewEl = ref<HTMLElement | null>(null);
|
||||||
const { width: containerWidth } = useElementSize(overviewEl)
|
const { width: containerWidth } = useElementSize(overviewEl);
|
||||||
|
|
||||||
const columnSize = computed(() => {
|
const columnSize = computed(() => {
|
||||||
return Math.floor((containerWidth.value) / ((ICON_SIZE + ICON_GRID_GAP)));
|
return Math.floor(containerWidth.value / (ICON_SIZE + ICON_GRID_GAP));
|
||||||
});
|
});
|
||||||
|
|
||||||
const mappedIcons = computed(() => {
|
const mappedIcons = computed(() => {
|
||||||
@@ -71,26 +71,27 @@ const searchResults = useSearch(searchQueryDebounced, mappedIcons, [
|
|||||||
const categories = computed(() => {
|
const categories = computed(() => {
|
||||||
if (!props.categories?.length || !props.icons?.length) return [];
|
if (!props.categories?.length || !props.icons?.length) return [];
|
||||||
|
|
||||||
return props.categories
|
return props.categories.map(({ name, title }) => {
|
||||||
.map(({ name, title }) => {
|
const categoryIcons = props.icons.filter((icon) => {
|
||||||
const categoryIcons = props.icons.filter((icon) => {
|
const iconCategories = icon?.externalLibrary
|
||||||
const iconCategories = icon?.externalLibrary ? icon.categories : props.iconCategories[icon.name]
|
? icon.categories
|
||||||
|
: props.iconCategories[icon.name];
|
||||||
|
|
||||||
return iconCategories?.includes(name);
|
return iconCategories?.includes(name);
|
||||||
});
|
});
|
||||||
|
|
||||||
const searchedCategoryIcons = isSearching
|
const searchedCategoryIcons = isSearching
|
||||||
? categoryIcons.filter((icon) =>
|
? categoryIcons.filter((icon) =>
|
||||||
searchResults.value.some((item) => item?.name === icon?.name)
|
searchResults.value.some((item) => item?.name === icon?.name),
|
||||||
)
|
)
|
||||||
: categoryIcons;
|
: categoryIcons;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
title,
|
title,
|
||||||
name,
|
name,
|
||||||
icons: searchedCategoryIcons,
|
icons: searchedCategoryIcons,
|
||||||
};
|
};
|
||||||
})
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const categoriesList = computed(() => {
|
const categoriesList = computed(() => {
|
||||||
@@ -107,26 +108,24 @@ const categoriesList = computed(() => {
|
|||||||
return acc;
|
return acc;
|
||||||
}, []);
|
}, []);
|
||||||
});
|
});
|
||||||
|
const searchPlaceholder = useSearchPlaceholder(searchQuery, searchResults);
|
||||||
|
|
||||||
const { list, containerProps, wrapperProps, scrollTo } = useVirtualList(
|
const { list, containerProps, wrapperProps, scrollTo } = useVirtualList(categoriesList, {
|
||||||
categoriesList,
|
itemHeight: ICON_SIZE + ICON_GRID_GAP,
|
||||||
{
|
overscan: 10,
|
||||||
itemHeight: ICON_SIZE + ICON_GRID_GAP,
|
});
|
||||||
overscan: 10
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
useScrollToCategory({
|
useScrollToCategory({
|
||||||
categories,
|
categories,
|
||||||
categoriesList,
|
categoriesList,
|
||||||
scrollTo,
|
scrollTo,
|
||||||
searchQueryDebounced,
|
searchQueryDebounced,
|
||||||
})
|
});
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
containerProps.ref.value = document.documentElement;
|
containerProps.ref.value = document.documentElement;
|
||||||
useEventListener(window, 'scroll', containerProps.onScroll)
|
useEventListener(window, 'scroll', containerProps.onScroll);
|
||||||
})
|
});
|
||||||
|
|
||||||
function onFocusSearchInput() {
|
function onFocusSearchInput() {
|
||||||
if (tags.value == null) {
|
if (tags.value == null) {
|
||||||
@@ -145,16 +144,13 @@ function handleCloseDrawer() {
|
|||||||
|
|
||||||
window.history.pushState({}, '', '/icons/categories');
|
window.history.pushState({}, '', '/icons/categories');
|
||||||
}
|
}
|
||||||
|
|
||||||
watchEffect(() => {
|
|
||||||
|
|
||||||
console.log(props.icons.find((icon) => icon.name === 'burger'));
|
|
||||||
|
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div ref="overviewEl" class="overview-container">
|
<div
|
||||||
|
ref="overviewEl"
|
||||||
|
class="overview-container"
|
||||||
|
>
|
||||||
<StickyBar class="category-search">
|
<StickyBar class="category-search">
|
||||||
<InputSearch
|
<InputSearch
|
||||||
:placeholder="`Search ${icons.length} icons ...`"
|
:placeholder="`Search ${icons.length} icons ...`"
|
||||||
@@ -166,8 +162,9 @@ watchEffect(() => {
|
|||||||
/>
|
/>
|
||||||
</StickyBar>
|
</StickyBar>
|
||||||
<NoResults
|
<NoResults
|
||||||
v-if="categories.length === 0"
|
v-if="searchPlaceholder.isNoResults"
|
||||||
:searchQuery="searchQuery"
|
:searchQuery="searchPlaceholder.query"
|
||||||
|
:isBrandSearch="searchPlaceholder.isBrand"
|
||||||
@clear="searchQuery = ''"
|
@clear="searchQuery = ''"
|
||||||
/>
|
/>
|
||||||
<div v-bind="wrapperProps">
|
<div v-bind="wrapperProps">
|
||||||
@@ -208,8 +205,4 @@ watchEffect(() => {
|
|||||||
.icons {
|
.icons {
|
||||||
margin-bottom: 8px;
|
margin-bottom: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.overview-container {
|
|
||||||
padding-bottom: 288px;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, computed, defineAsyncComponent, onMounted, onBeforeUnmount, watch } from 'vue';
|
import { ref, computed, defineAsyncComponent, onMounted, watch } from 'vue';
|
||||||
import type { IconEntity } from '../../types';
|
import type { IconEntity } from '../../types';
|
||||||
import { useElementSize, useEventListener, useVirtualList } from '@vueuse/core';
|
import { useElementSize, useEventListener, useVirtualList } from '@vueuse/core';
|
||||||
import { useRoute } from 'vitepress';
|
import { useRoute } from 'vitepress';
|
||||||
@@ -13,17 +13,18 @@ import useFetchTags from '../../composables/useFetchTags';
|
|||||||
import useFetchCategories from '../../composables/useFetchCategories';
|
import useFetchCategories from '../../composables/useFetchCategories';
|
||||||
import chunkArray from '../../utils/chunkArray';
|
import chunkArray from '../../utils/chunkArray';
|
||||||
import CarbonAdOverlay from './CarbonAdOverlay.vue';
|
import CarbonAdOverlay from './CarbonAdOverlay.vue';
|
||||||
|
import useSearchPlaceholder from '../../utils/useSearchPlaceholder.ts';
|
||||||
|
|
||||||
const ICON_SIZE = 56;
|
const ICON_SIZE = 56;
|
||||||
const ICON_GRID_GAP = 8;
|
const ICON_GRID_GAP = 8;
|
||||||
|
|
||||||
const initialGridItems = computed(() => {
|
const initialGridItems = computed(() => {
|
||||||
if (containerWidth.value === 0) return 120;
|
if (containerWidth.value === 0) return 120;
|
||||||
|
|
||||||
const itemsPerRow = columnSize.value || 10;
|
const itemsPerRow = columnSize.value || 10;
|
||||||
const visibleRows = Math.ceil(window.innerHeight / (ICON_SIZE + ICON_GRID_GAP));
|
const visibleRows = Math.ceil(window.innerHeight / (ICON_SIZE + ICON_GRID_GAP));
|
||||||
|
|
||||||
return Math.min(itemsPerRow * (visibleRows + 2), 200);
|
return Math.min(itemsPerRow * (visibleRows + 2), 200);
|
||||||
});
|
});
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
@@ -36,10 +37,10 @@ const { execute: fetchTags, data: tags } = useFetchTags();
|
|||||||
const { execute: fetchCategories, data: categories } = useFetchCategories();
|
const { execute: fetchCategories, data: categories } = useFetchCategories();
|
||||||
|
|
||||||
const overviewEl = ref<HTMLElement | null>(null);
|
const overviewEl = ref<HTMLElement | null>(null);
|
||||||
const { width: containerWidth } = useElementSize(overviewEl)
|
const { width: containerWidth } = useElementSize(overviewEl);
|
||||||
|
|
||||||
const columnSize = computed(() => {
|
const columnSize = computed(() => {
|
||||||
return Math.floor((containerWidth.value) / ((ICON_SIZE + ICON_GRID_GAP)));
|
return Math.floor(containerWidth.value / (ICON_SIZE + ICON_GRID_GAP));
|
||||||
});
|
});
|
||||||
|
|
||||||
const mappedIcons = computed(() => {
|
const mappedIcons = computed(() => {
|
||||||
@@ -71,29 +72,27 @@ const searchResults = useSearch(searchQueryDebounced, mappedIcons, [
|
|||||||
{ name: 'tags', weight: 2 },
|
{ name: 'tags', weight: 2 },
|
||||||
{ name: 'categories', weight: 1 },
|
{ name: 'categories', weight: 1 },
|
||||||
]);
|
]);
|
||||||
|
const searchPlaceholder = useSearchPlaceholder(searchQuery, searchResults);
|
||||||
|
|
||||||
const chunkedIcons = computed(() => {
|
const chunkedIcons = computed(() => {
|
||||||
return chunkArray(searchResults.value, columnSize.value);
|
return chunkArray(searchResults.value, columnSize.value);
|
||||||
});
|
});
|
||||||
|
|
||||||
const { list, containerProps, wrapperProps, scrollTo } = useVirtualList(
|
const { list, containerProps, wrapperProps, scrollTo } = useVirtualList(chunkedIcons, {
|
||||||
chunkedIcons,
|
itemHeight: ICON_SIZE + ICON_GRID_GAP,
|
||||||
{
|
overscan: 10,
|
||||||
itemHeight: ICON_SIZE + ICON_GRID_GAP,
|
});
|
||||||
overscan: 10
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
containerProps.ref.value = document.documentElement;
|
containerProps.ref.value = document.documentElement;
|
||||||
useEventListener(window, 'scroll', containerProps.onScroll)
|
useEventListener(window, 'scroll', containerProps.onScroll);
|
||||||
|
|
||||||
// Check if we should focus the search input from URL parameter
|
// Check if we should focus the search input from URL parameter
|
||||||
const route = useRoute()
|
const route = useRoute();
|
||||||
if (route.data?.relativePath && window.location.search.includes('focus')) {
|
if (route.data?.relativePath && window.location.search.includes('focus')) {
|
||||||
searchInput.value?.focus()
|
searchInput.value?.focus();
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
|
|
||||||
function setActiveIconName(name: string) {
|
function setActiveIconName(name: string) {
|
||||||
activeIconName.value = name;
|
activeIconName.value = name;
|
||||||
@@ -113,8 +112,8 @@ const NoResults = defineAsyncComponent(() => import('./NoResults.vue'));
|
|||||||
const IconDetailOverlay = defineAsyncComponent(() => import('./IconDetailOverlay.vue'));
|
const IconDetailOverlay = defineAsyncComponent(() => import('./IconDetailOverlay.vue'));
|
||||||
|
|
||||||
watch(searchQueryDebounced, () => {
|
watch(searchQueryDebounced, () => {
|
||||||
scrollTo(0)
|
scrollTo(0);
|
||||||
})
|
});
|
||||||
|
|
||||||
function handleCloseDrawer() {
|
function handleCloseDrawer() {
|
||||||
setActiveIconName('');
|
setActiveIconName('');
|
||||||
@@ -124,7 +123,10 @@ function handleCloseDrawer() {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div ref="overviewEl" class="overview-container">
|
<div
|
||||||
|
ref="overviewEl"
|
||||||
|
class="overview-container"
|
||||||
|
>
|
||||||
<StickyBar>
|
<StickyBar>
|
||||||
<InputSearch
|
<InputSearch
|
||||||
:placeholder="`Search ${icons.length} icons ...`"
|
:placeholder="`Search ${icons.length} icons ...`"
|
||||||
@@ -136,8 +138,9 @@ function handleCloseDrawer() {
|
|||||||
/>
|
/>
|
||||||
</StickyBar>
|
</StickyBar>
|
||||||
<NoResults
|
<NoResults
|
||||||
v-if="searchResults.length === 0 && searchQuery !== ''"
|
v-if="searchPlaceholder.isNoResults"
|
||||||
:searchQuery="searchQuery"
|
:searchQuery="searchPlaceholder.query"
|
||||||
|
:isBrandSearch="searchPlaceholder.isBrand"
|
||||||
@clear="searchQuery = ''"
|
@clear="searchQuery = ''"
|
||||||
/>
|
/>
|
||||||
<IconGrid
|
<IconGrid
|
||||||
@@ -183,8 +186,4 @@ function handleCloseDrawer() {
|
|||||||
.input-wrapper {
|
.input-wrapper {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.overview-container {
|
|
||||||
padding-bottom: 288px;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,56 +1,218 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, onMounted, computed } from 'vue'
|
import { ref, onMounted, computed, markRaw, shallowReadonly, watch } from 'vue';
|
||||||
import { bird, squirrel, rabbit } from '../../../data/iconNodes'
|
import {
|
||||||
import createLucideIcon from 'lucide-vue-next/src/createLucideIcon'
|
bird,
|
||||||
import {useEventListener} from '@vueuse/core'
|
squirrel,
|
||||||
import VPButton from 'vitepress/dist/client/theme-default/components/VPButton.vue'
|
rabbit,
|
||||||
import { IconNode } from '../../types'
|
ghost,
|
||||||
|
castle,
|
||||||
|
drama,
|
||||||
|
dog,
|
||||||
|
cat,
|
||||||
|
wandSparkles,
|
||||||
|
save,
|
||||||
|
snowflake,
|
||||||
|
cake,
|
||||||
|
fish,
|
||||||
|
turtle,
|
||||||
|
rat,
|
||||||
|
worm,
|
||||||
|
testTubeDiagonal,
|
||||||
|
sword,
|
||||||
|
} from '../../../data/iconNodes';
|
||||||
|
import createLucideIcon from 'lucide-vue-next/src/createLucideIcon';
|
||||||
|
import { useEventListener } from '@vueuse/core';
|
||||||
|
import VPButton from 'vitepress/dist/client/theme-default/components/VPButton.vue';
|
||||||
|
import { IconNode } from '../../types';
|
||||||
|
|
||||||
defineProps<{
|
const { searchQuery, isBrandSearch } = defineProps<{
|
||||||
searchQuery: string
|
searchQuery: string;
|
||||||
}>()
|
isBrandSearch: boolean;
|
||||||
|
}>();
|
||||||
|
|
||||||
defineEmits(['clear'])
|
defineEmits(['clear']);
|
||||||
|
|
||||||
const animalIcon = ref<HTMLElement>()
|
interface Placeholder {
|
||||||
const randomAnimal = computed<IconNode>(() => {
|
title: string;
|
||||||
return Math.random() > 0.5 ? squirrel : Math.random() > 0.5 ? rabbit : bird
|
message: string;
|
||||||
})
|
icon: IconNode;
|
||||||
const animalComponent = computed(() => createLucideIcon('animal', randomAnimal.value))
|
finePrint?: string;
|
||||||
const flip = ref(false)
|
}
|
||||||
|
|
||||||
|
const brandPlaceholders: Placeholder[] = shallowReadonly([
|
||||||
|
{
|
||||||
|
title: 'Boooo! What a scary brand logo!',
|
||||||
|
message:
|
||||||
|
'[name] and its friends often haunt this search box, but you won’t ever find them here.',
|
||||||
|
icon: markRaw(ghost),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Thank You Mario!',
|
||||||
|
message: 'But [name] is in another castle!',
|
||||||
|
icon: markRaw(castle),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '[name] did audition for our icon set',
|
||||||
|
message: '...but didn’t make the callback.',
|
||||||
|
icon: markRaw(drama),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Such Search. Very [name].',
|
||||||
|
message: 'Much not here. So Wow.',
|
||||||
|
icon: markRaw(dog),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'I Can Has [name]?',
|
||||||
|
message: 'No [name] for you in here.',
|
||||||
|
icon: markRaw(cat),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Loading [name]...',
|
||||||
|
message: 'Fatal error: our cartridge contains only open-source pixels.',
|
||||||
|
icon: markRaw(save),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '[name] Shall Not Pass',
|
||||||
|
message: 'Do not look to its coming at first light of any day.',
|
||||||
|
icon: markRaw(wandSparkles),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Winter is coming',
|
||||||
|
message: 'But [name] sure isn’t.',
|
||||||
|
icon: markRaw(snowflake),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'The cake is a lie',
|
||||||
|
message: 'And so is the promise of an icon for [name] at Lucide.',
|
||||||
|
icon: markRaw(cake),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'It’s not a bug',
|
||||||
|
message: 'Having no [name] icon around is a feature.',
|
||||||
|
icon: markRaw(worm),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'The lab exploded',
|
||||||
|
message: 'We tried mixing [name] with open-source icons.',
|
||||||
|
icon: markRaw(testTubeDiagonal),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'It’s Dangerous to Go Alone',
|
||||||
|
message: 'Take this icon instead — it’s not [name], but it might help.',
|
||||||
|
icon: markRaw(sword),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const notFoundPlaceholders: Omit<Placeholder, 'title'>[] = shallowReadonly([
|
||||||
|
{
|
||||||
|
message: 'We’ve looked for this icon for a bird’s eye view, but could not find it.',
|
||||||
|
icon: markRaw(bird),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: 'We checked every tree. Only acorns, no results.',
|
||||||
|
icon: markRaw(squirrel),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: 'You’ve gone too deep into the rabbit hole — this icon doesn’t exist.',
|
||||||
|
icon: markRaw(rabbit),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: 'This icon seems to have slipped through the net.',
|
||||||
|
icon: markRaw(fish),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: 'This icon might exist in the future… but it hasn’t arrived yet.',
|
||||||
|
icon: markRaw(turtle),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: 'Rats! This icon seems to have slipped through the cracks.',
|
||||||
|
icon: markRaw(rat),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
function randomItem<T>(arr: T[]): T {
|
||||||
|
return arr[Math.floor(Math.random() * arr.length)];
|
||||||
|
}
|
||||||
|
|
||||||
|
const placeholderIcon = ref<HTMLElement>();
|
||||||
|
const placeholder = ref<Placeholder>();
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => isBrandSearch,
|
||||||
|
() => {
|
||||||
|
placeholder.value = isBrandSearch
|
||||||
|
? {
|
||||||
|
...randomItem(brandPlaceholders),
|
||||||
|
finePrint:
|
||||||
|
'Lucide does not accept brand logos, and we do not plan to add them in the future. This is due to a combination of legal restrictions, design consistency concerns, and practical maintenance reasons.',
|
||||||
|
}
|
||||||
|
: {
|
||||||
|
title: `No results for “[name]”`,
|
||||||
|
finePrint:
|
||||||
|
'This icon doesn’t seem to exist… yet. Try searching similar terms, browsing existing requests, or opening a new one.',
|
||||||
|
...randomItem(notFoundPlaceholders),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
{ immediate: true },
|
||||||
|
);
|
||||||
|
const iconComponent = computed(() => createLucideIcon('placeholder', placeholder.value.icon));
|
||||||
|
const flip = ref(false);
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
useEventListener(document, 'mousemove', (mouseEvent) => {
|
useEventListener(document, 'mousemove', (mouseEvent) => {
|
||||||
const {width, height, x, y} = animalIcon.value.getBoundingClientRect()
|
const { width, x } = placeholderIcon.value.getBoundingClientRect();
|
||||||
|
|
||||||
const centerX = (width / 2) + x
|
const centerX = width / 2 + x;
|
||||||
|
|
||||||
flip.value = mouseEvent.x < centerX
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
|
flip.value = !isBrandSearch && mouseEvent.x < centerX;
|
||||||
|
});
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="no-results">
|
<div class="no-results">
|
||||||
<component
|
<component
|
||||||
:is="animalComponent"
|
:is="iconComponent"
|
||||||
class="animal-icon"
|
class="placeholder-icon"
|
||||||
ref="animalIcon"
|
ref="placeholderIcon"
|
||||||
:class="{ flip }"
|
:class="{ flip }"
|
||||||
:strokeWidth="1"
|
:strokeWidth="1"
|
||||||
/>
|
/>
|
||||||
<h2 class="no-results-text">
|
<h2 class="no-results-text">{{ placeholder.title.replace('[name]', searchQuery) }}</h2>
|
||||||
No icons found for '{{ searchQuery }}'
|
<p class="no-results-message">
|
||||||
</h2>
|
{{ placeholder.message.replace('[name]', searchQuery) }}
|
||||||
|
</p>
|
||||||
|
<div class="divider"></div>
|
||||||
|
<p
|
||||||
|
v-if="placeholder.finePrint"
|
||||||
|
class="no-results-fine-print"
|
||||||
|
>
|
||||||
|
{{ placeholder.finePrint }}
|
||||||
|
</p>
|
||||||
<VPButton
|
<VPButton
|
||||||
text="Clear your search and try again"
|
v-if="isBrandSearch"
|
||||||
theme="alt"
|
text="Head over to Simple Icons"
|
||||||
|
theme="brand"
|
||||||
|
:href="`https://simpleicons.org/?q=${searchQuery}`"
|
||||||
|
target="_blank"
|
||||||
|
/>
|
||||||
|
<VPButton
|
||||||
|
v-else
|
||||||
|
text="Clear search & try again"
|
||||||
|
theme="brand"
|
||||||
@click="$emit('clear')"
|
@click="$emit('clear')"
|
||||||
/>
|
/>
|
||||||
<span class="text-divider">or</span>
|
<span class="text-divider">or</span>
|
||||||
<VPButton
|
<VPButton
|
||||||
text="Search on Github issues"
|
v-if="isBrandSearch"
|
||||||
|
text="Read our statement on brand logos"
|
||||||
|
theme="alt"
|
||||||
|
href="https://github.com/lucide-icons/lucide/blob/main/BRAND_LOGOS_STATEMENT.md"
|
||||||
|
target="_blank"
|
||||||
|
/>
|
||||||
|
<VPButton
|
||||||
|
v-else
|
||||||
|
text="Search GitHub issues"
|
||||||
theme="alt"
|
theme="alt"
|
||||||
:href="`https://github.com/lucide-icons/lucide/issues?q=is%3Aopen+${searchQuery}`"
|
:href="`https://github.com/lucide-icons/lucide/issues?q=is%3Aopen+${searchQuery}`"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
@@ -63,33 +225,38 @@ onMounted(() => {
|
|||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
text-align: center;
|
||||||
|
padding-block: 48px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.animal-icon {
|
.placeholder-icon {
|
||||||
width: 160px;
|
width: 96px;
|
||||||
height: 160px;
|
height: 96px;
|
||||||
color: var(--vp-c-neutral);
|
color: var(--vp-c-text-1);
|
||||||
opacity: 0.8;
|
|
||||||
margin-top: 72px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.animal-icon.flip {
|
.placeholder-icon.flip {
|
||||||
transform: rotateY(180deg);
|
transform: rotateY(180deg);
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 960px) {
|
|
||||||
.animal-icon {
|
|
||||||
width: 240px;
|
|
||||||
height: 240px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.no-results-text {
|
.no-results-text {
|
||||||
line-height: 40px;
|
line-height: 1.35;
|
||||||
font-size: 24px;
|
font-size: 24px;
|
||||||
margin-top: 24px;
|
margin-top: 24px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
text-wrap: balance;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-results-message {
|
||||||
|
text-wrap: balance;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-results-fine-print {
|
||||||
|
max-inline-size: 60ch;
|
||||||
|
font-size: 14px;
|
||||||
margin-bottom: 32px;
|
margin-bottom: 32px;
|
||||||
text-align: center;
|
color: var(--vp-c-text-2);
|
||||||
|
text-wrap: balance;
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-divider {
|
.text-divider {
|
||||||
@@ -97,4 +264,10 @@ onMounted(() => {
|
|||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
color: var(--vp-c-neutral);
|
color: var(--vp-c-neutral);
|
||||||
}
|
}
|
||||||
|
.divider {
|
||||||
|
margin: 24px auto 18px;
|
||||||
|
width: 64px;
|
||||||
|
height: 1px;
|
||||||
|
background-color: var(--vp-c-divider);
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,75 +1,72 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { shallowRef, type Ref, watch, computed } from 'vue'
|
import { shallowRef, type Ref, watch, computed } from 'vue';
|
||||||
import { useCssVar, syncRef } from '@vueuse/core'
|
import { useCssVar, syncRef } from '@vueuse/core';
|
||||||
import { STYLE_DEFAULTS, useIconStyleContext } from '../../composables/useIconStyle'
|
import { STYLE_DEFAULTS, useIconStyleContext } from '../../composables/useIconStyle';
|
||||||
import RangeSlider from '../base/RangeSlider.vue'
|
import RangeSlider from '../base/RangeSlider.vue';
|
||||||
import InputField from '../base/InputField.vue'
|
import InputField from '../base/InputField.vue';
|
||||||
import ColorPicker from '../base/ColorPicker.vue'
|
import ColorPicker from '../base/ColorPicker.vue';
|
||||||
import ResetButton from '../base/ResetButton.vue'
|
import ResetButton from '../base/ResetButton.vue';
|
||||||
import Switch from '../base/Switch.vue'
|
import Switch from '../base/Switch.vue';
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
rootEl?: Ref<HTMLElement>
|
rootEl?: Ref<HTMLElement>;
|
||||||
}>()
|
}>();
|
||||||
|
|
||||||
const { color, strokeWidth, size, absoluteStrokeWidth } = useIconStyleContext()
|
const { color, strokeWidth, size, absoluteStrokeWidth } = useIconStyleContext();
|
||||||
const documentRef = shallowRef<HTMLElement | undefined>(typeof document !== 'undefined' ? document?.documentElement : undefined)
|
const documentRef = shallowRef<HTMLElement | undefined>(
|
||||||
|
typeof document !== 'undefined' ? document?.documentElement : undefined,
|
||||||
|
);
|
||||||
|
|
||||||
const colorCssVar = useCssVar(
|
const colorCssVar = useCssVar('--customize-color', props.rootEl?.value ?? documentRef.value, {
|
||||||
'--customize-color',
|
initialValue: `${STYLE_DEFAULTS.color}`,
|
||||||
props.rootEl?.value ?? documentRef.value,
|
});
|
||||||
{
|
|
||||||
initialValue: `${STYLE_DEFAULTS.color}`
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
const strokeWidthCssVar = useCssVar(
|
const strokeWidthCssVar = useCssVar(
|
||||||
'--customize-strokeWidth',
|
'--customize-strokeWidth',
|
||||||
props.rootEl?.value ?? documentRef.value,
|
props.rootEl?.value ?? documentRef.value,
|
||||||
{
|
{
|
||||||
initialValue: `${STYLE_DEFAULTS.strokeWidth}`
|
initialValue: `${STYLE_DEFAULTS.strokeWidth}`,
|
||||||
}
|
},
|
||||||
)
|
);
|
||||||
|
|
||||||
const sizeCssVar = useCssVar(
|
const sizeCssVar = useCssVar('--customize-size', props.rootEl?.value ?? documentRef.value, {
|
||||||
'--customize-size',
|
initialValue: `${STYLE_DEFAULTS.size}`,
|
||||||
props.rootEl?.value ?? documentRef.value,
|
});
|
||||||
{
|
|
||||||
initialValue: `${STYLE_DEFAULTS.size}`
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
syncRef(color, colorCssVar, { direction: 'ltr' })
|
syncRef(color, colorCssVar, { direction: 'ltr' });
|
||||||
syncRef(strokeWidth, strokeWidthCssVar, { direction: 'ltr' })
|
syncRef(strokeWidth, strokeWidthCssVar, { direction: 'ltr' });
|
||||||
syncRef(size, sizeCssVar, { direction: 'ltr' })
|
syncRef(size, sizeCssVar, { direction: 'ltr' });
|
||||||
|
|
||||||
function resetStyle () {
|
function resetStyle() {
|
||||||
color.value = STYLE_DEFAULTS.color
|
color.value = STYLE_DEFAULTS.color;
|
||||||
strokeWidth.value = STYLE_DEFAULTS.strokeWidth
|
strokeWidth.value = STYLE_DEFAULTS.strokeWidth;
|
||||||
size.value = STYLE_DEFAULTS.size
|
size.value = STYLE_DEFAULTS.size;
|
||||||
absoluteStrokeWidth.value = STYLE_DEFAULTS.absoluteStrokeWidth
|
absoluteStrokeWidth.value = STYLE_DEFAULTS.absoluteStrokeWidth;
|
||||||
}
|
}
|
||||||
|
|
||||||
watch(absoluteStrokeWidth, (enabled) => {
|
watch(absoluteStrokeWidth, (enabled) => {
|
||||||
const htmlEl = document.documentElement
|
const htmlEl = document.documentElement;
|
||||||
|
|
||||||
htmlEl.classList.toggle('absolute-stroke-width', enabled)
|
htmlEl.classList.toggle('absolute-stroke-width', enabled);
|
||||||
})
|
});
|
||||||
|
|
||||||
const customizingActive = computed(() => {
|
const customizingActive = computed(() => {
|
||||||
return color.value !== STYLE_DEFAULTS.color
|
return (
|
||||||
|| strokeWidth.value !== STYLE_DEFAULTS.strokeWidth
|
color.value !== STYLE_DEFAULTS.color ||
|
||||||
|| size.value !== STYLE_DEFAULTS.size
|
strokeWidth.value !== STYLE_DEFAULTS.strokeWidth ||
|
||||||
|| absoluteStrokeWidth.value !== STYLE_DEFAULTS.absoluteStrokeWidth
|
size.value !== STYLE_DEFAULTS.size ||
|
||||||
})
|
absoluteStrokeWidth.value !== STYLE_DEFAULTS.absoluteStrokeWidth
|
||||||
|
);
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="customizer-card" :class="{ customized: customizingActive }">
|
<div
|
||||||
|
class="customizer-card"
|
||||||
|
:class="{ customized: customizingActive }"
|
||||||
|
>
|
||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
<h2 class="card-title">
|
<h2 class="card-title">Customizer</h2>
|
||||||
Customizer
|
|
||||||
</h2>
|
|
||||||
<ResetButton @click="resetStyle"></ResetButton>
|
<ResetButton @click="resetStyle"></ResetButton>
|
||||||
</div>
|
</div>
|
||||||
<InputField
|
<InputField
|
||||||
@@ -77,7 +74,11 @@ const customizingActive = computed(() => {
|
|||||||
label="Color"
|
label="Color"
|
||||||
>
|
>
|
||||||
<template #display>
|
<template #display>
|
||||||
<ColorPicker v-model="color" id="icon-color" class="color-picker"/>
|
<ColorPicker
|
||||||
|
v-model="color"
|
||||||
|
id="icon-color"
|
||||||
|
class="color-picker"
|
||||||
|
/>
|
||||||
</template>
|
</template>
|
||||||
</InputField>
|
</InputField>
|
||||||
|
|
||||||
@@ -117,7 +118,7 @@ const customizingActive = computed(() => {
|
|||||||
|
|
||||||
<InputField
|
<InputField
|
||||||
id="absolute-stroke-width"
|
id="absolute-stroke-width"
|
||||||
label="Absolute Stroke width"
|
label="Absolute stroke width"
|
||||||
>
|
>
|
||||||
<Switch
|
<Switch
|
||||||
id="absolute-stroke-width"
|
id="absolute-stroke-width"
|
||||||
@@ -143,6 +144,7 @@ const customizingActive = computed(() => {
|
|||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
/* margin-bottom: 12px; */
|
/* margin-bottom: 12px; */
|
||||||
}
|
}
|
||||||
|
|
||||||
.customizer-card {
|
.customizer-card {
|
||||||
background: var(--vp-c-bg);
|
background: var(--vp-c-bg);
|
||||||
padding: 12px 24px 24px;
|
padding: 12px 24px 24px;
|
||||||
@@ -151,7 +153,7 @@ const customizingActive = computed(() => {
|
|||||||
position: relative;
|
position: relative;
|
||||||
z-index: 0;
|
z-index: 0;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
transition: border-color .4s ease-in-out;
|
transition: border-color 0.4s ease-in-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
.customizer-card.customized {
|
.customizer-card.customized {
|
||||||
|
|||||||
41
docs/.vitepress/theme/utils/useSearchPlaceholder.ts
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import { ref, Ref, watch } from 'vue';
|
||||||
|
import BRAND_STOPWORDS from '../../data/brandStopwords.json' with { type: 'json' };
|
||||||
|
|
||||||
|
export default function useSearchPlaceholder(
|
||||||
|
searchQuery: Ref<string, string>,
|
||||||
|
results: Ref<{ name: string }[]>,
|
||||||
|
) {
|
||||||
|
const state = ref({
|
||||||
|
isNoResults: false,
|
||||||
|
isBrand: false,
|
||||||
|
query: '',
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(
|
||||||
|
results,
|
||||||
|
() => {
|
||||||
|
const query = searchQuery.value;
|
||||||
|
const searchResults = results.value;
|
||||||
|
if (query.length > 0 && searchResults.length === 0) {
|
||||||
|
for (const stopword of Object.keys(BRAND_STOPWORDS)) {
|
||||||
|
if (stopword.startsWith(query)) {
|
||||||
|
state.value = {
|
||||||
|
isNoResults: true,
|
||||||
|
isBrand: true,
|
||||||
|
query: BRAND_STOPWORDS[stopword],
|
||||||
|
};
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
state.value = {
|
||||||
|
isNoResults: query in BRAND_STOPWORDS && searchResults.length === 0 && query !== '',
|
||||||
|
isBrand: query in BRAND_STOPWORDS,
|
||||||
|
query: BRAND_STOPWORDS[query] ?? query,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
{ immediate: true },
|
||||||
|
);
|
||||||
|
|
||||||
|
return state;
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { IconNode } from 'lucide-vue-next/src/createLucideIcon';
|
import { type IconNode } from 'lucide-vue-next/src/types';
|
||||||
import Vue from 'vue';
|
import Vue from 'vue';
|
||||||
|
|
||||||
declare module '*.vue' {
|
declare module '*.vue' {
|
||||||
@@ -20,5 +20,6 @@ declare module 'node:module' {
|
|||||||
}
|
}
|
||||||
|
|
||||||
declare module '*.node.json' {
|
declare module '*.node.json' {
|
||||||
export default IconNode;
|
const value: IconNode;
|
||||||
|
export default value;
|
||||||
}
|
}
|
||||||
@@ -29,7 +29,7 @@ However, not everyone can understand them easily. Read more about [how to use Lu
|
|||||||
|
|
||||||
## Official Packages
|
## Official Packages
|
||||||
|
|
||||||
Lucide's official packages are designed to work on different platforms, making it easier for users to integrate icons into their projects. The packages are available for various technologies, including [Web (Vanilla)](https://lucide.dev/guide/packages/lucide), [React](https://lucide.dev/guide/packages/lucide-react), [React Native](https://lucide.dev/guide/packages/lucide-react-native), [Vue](https://lucide.dev/guide/packages/lucide-vue), [Vue 3](https://lucide.dev/guide/packages/lucide-vue-next), [Svelte](https://lucide.dev/guide/packages/lucide-svelte), [Preact](https://lucide.dev/guide/packages/lucide-preact), [Solid](https://lucide.dev/guide/packages/lucide-solid), [Angular](https://lucide.dev/guide/packages/lucide-angular), [Astro](https://lucide.dev/guide/packages/lucide-astro), and [NodeJS](https://lucide.dev/guide/packages/lucide-static#nodejs).
|
Lucide's official packages are designed to work on different platforms, making it easier for users to integrate icons into their projects. The packages are available for various technologies, including [Web (Vanilla)](https://lucide.dev/guide/packages/lucide), [React](https://lucide.dev/guide/packages/lucide-react), [React Native](https://lucide.dev/guide/packages/lucide-react-native), [Vue](https://lucide.dev/guide/packages/lucide-vue), [Vue 3](https://lucide.dev/guide/packages/lucide-vue-next), [Svelte](https://lucide.dev/guide/packages/lucide-svelte), [Preact](https://lucide.dev/guide/packages/lucide-preact), [Solid](https://lucide.dev/guide/packages/lucide-solid), [Angular](https://lucide.dev/guide/packages/angular), [Astro](https://lucide.dev/guide/packages/lucide-astro), and [NodeJS](https://lucide.dev/guide/packages/lucide-static#nodejs).
|
||||||
|
|
||||||
## Community
|
## Community
|
||||||
|
|
||||||
|
|||||||
277
docs/guide/packages/angular.md
Normal file
@@ -0,0 +1,277 @@
|
|||||||
|
# `@lucide/angular`
|
||||||
|
|
||||||
|
::: warning
|
||||||
|
This documentation is for `@lucide/angular`.
|
||||||
|
|
||||||
|
To learn about our legacy package for Angular, please refer to [`lucide-angular`](./lucide-angular).
|
||||||
|
:::
|
||||||
|
|
||||||
|
A standalone, signal-based, zoneless implementation of Lucide icons for Angular.
|
||||||
|
|
||||||
|
**What you can accomplish:**
|
||||||
|
- Use icons as standalone Angular components with full dependency injection support
|
||||||
|
- Configure icons globally through modern Angular providers
|
||||||
|
- Integrate with Angular's reactive forms and data binding
|
||||||
|
- Build scalable applications with tree-shaken icons and lazy loading support
|
||||||
|
|
||||||
|
## Prerequisites
|
||||||
|
|
||||||
|
This package requires Angular 17+ and uses standalone components, signals, and zoneless change detection.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
::: code-group
|
||||||
|
|
||||||
|
```sh [pnpm]
|
||||||
|
pnpm add @lucide/angular
|
||||||
|
```
|
||||||
|
|
||||||
|
```sh [yarn]
|
||||||
|
yarn add @lucide/angular
|
||||||
|
```
|
||||||
|
|
||||||
|
```sh [npm]
|
||||||
|
npm install @lucide/angular
|
||||||
|
```
|
||||||
|
|
||||||
|
```sh [bun]
|
||||||
|
bun add @lucide/angular
|
||||||
|
```
|
||||||
|
|
||||||
|
:::
|
||||||
|
|
||||||
|
## How to use
|
||||||
|
|
||||||
|
### Standalone icons
|
||||||
|
|
||||||
|
Every icon can be imported as a ready-to-use standalone component:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<svg lucideFileText></svg>
|
||||||
|
```
|
||||||
|
|
||||||
|
```ts{2,7}
|
||||||
|
import { Component } from '@angular/core';
|
||||||
|
import { LucideFileText } from '@lucide/angular';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-foobar',
|
||||||
|
templateUrl: './foobar.html',
|
||||||
|
imports: [LucideFileText],
|
||||||
|
})
|
||||||
|
export class Foobar { }
|
||||||
|
```
|
||||||
|
|
||||||
|
::: tip
|
||||||
|
Standalone icon components use the selector `svg[lucide{PascalCaseIconName}]`.
|
||||||
|
|
||||||
|
This ensures minimal bloating of the DOM and the ability to directly manipulate all attributes of the resulting SVG element.
|
||||||
|
:::
|
||||||
|
|
||||||
|
### Dynamic icon component
|
||||||
|
|
||||||
|
You may also use the dynamic `LucideIcon` component to dynamically render icons.
|
||||||
|
|
||||||
|
#### With tree-shaken imports
|
||||||
|
|
||||||
|
You may pass imported icons directly to the component:
|
||||||
|
|
||||||
|
```html{3}
|
||||||
|
@for (item of items) {
|
||||||
|
<a navbarItem [routerLink]="item.routerLink">
|
||||||
|
<svg [lucideIcon]="item.icon"></svg>
|
||||||
|
{{ item.title }}
|
||||||
|
</a>
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```ts{2,8,14,19}
|
||||||
|
import { Component } from '@angular/core';
|
||||||
|
import { LucideIcon, LucideHouse, LucideUsersRound } from '@lucide/angular';
|
||||||
|
import { NavbarItem, NavbarItemModel } from './navbar-item';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-navbar',
|
||||||
|
templateUrl: './navbar.html',
|
||||||
|
imports: [LucideIcon, NavbarItem],
|
||||||
|
})
|
||||||
|
export class Navbar {
|
||||||
|
readonly items: NavbarItemModel[] = [
|
||||||
|
{
|
||||||
|
title: 'Home',
|
||||||
|
icon: LucideHouse,
|
||||||
|
routerLink: [''],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Users',
|
||||||
|
icon: LucideUsersRound,
|
||||||
|
routerLink: ['admin/users'],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### With icons provided via dependency injection
|
||||||
|
|
||||||
|
Alternatively, the component also accepts string inputs.
|
||||||
|
|
||||||
|
To use icons this way, first, you have to provide icons via `provideLucideIcons`:
|
||||||
|
|
||||||
|
:::code-group
|
||||||
|
```ts{7-10} [app.config.ts]
|
||||||
|
import { ApplicationConfig } from '@angular/core';
|
||||||
|
import { provideLucideIcons, LucideCircleCheck, LucideCircleX } from '@lucide/angular';
|
||||||
|
|
||||||
|
export const appConfig: ApplicationConfig = {
|
||||||
|
providers: [
|
||||||
|
// ...
|
||||||
|
provideLucideIcons([
|
||||||
|
LucideCircleCheck,
|
||||||
|
LucideCircleX,
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
```html [foobar.html]
|
||||||
|
<svg lucideIcon="circle-check"></svg>
|
||||||
|
```
|
||||||
|
|
||||||
|
```ts{7} [foobar.ts]
|
||||||
|
import { Component } from '@angular/core';
|
||||||
|
import { LucideIcon } from '@lucide/angular';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-foobar',
|
||||||
|
templateUrl: './template-url',
|
||||||
|
imports: [LucideIcon],
|
||||||
|
})
|
||||||
|
export class Foobar { }
|
||||||
|
```
|
||||||
|
:::
|
||||||
|
|
||||||
|
::: tip
|
||||||
|
For optimal bundle size, provide icons at the highest appropriate level in your application.
|
||||||
|
|
||||||
|
Providing all icons at the root level may increase your initial bundle size, while providing them at feature module level enables better code splitting.
|
||||||
|
:::
|
||||||
|
|
||||||
|
::: warning
|
||||||
|
While you may provide your icons at any level of the dependency injection tree, be aware that [Angular's DI system is hierarchical](https://angular.dev/guide/di/defining-dependency-providers#injector-hierarchy-in-angular): `LucideIcon` will only have access to the icons provided closest to it in the tree.
|
||||||
|
:::
|
||||||
|
|
||||||
|
## Accessible labels
|
||||||
|
|
||||||
|
You can use the `title` input property to set the [accessible name element](https://developer.mozilla.org/en-US/docs/Web/SVG/Reference/Element/title) on the SVG:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<svg lucideIcon="house" title="Go to dashboard"></svg>
|
||||||
|
```
|
||||||
|
|
||||||
|
This will result in the following output:
|
||||||
|
|
||||||
|
```html{2}
|
||||||
|
<svg class="lucide lucide-house" ...>
|
||||||
|
<title>Go to dashboard</title>
|
||||||
|
<!-- SVG paths -->
|
||||||
|
</svg>
|
||||||
|
```
|
||||||
|
|
||||||
|
## Props
|
||||||
|
|
||||||
|
You can pass additional props to adjust the icon appearance.
|
||||||
|
|
||||||
|
| name | type | default |
|
||||||
|
|-----------------------|-----------|--------------|
|
||||||
|
| `size` | *number* | 24 |
|
||||||
|
| `color` | *string* | currentColor |
|
||||||
|
| `strokeWidth` | *number* | 2 |
|
||||||
|
| `absoluteStrokeWidth` | *boolean* | false |
|
||||||
|
|
||||||
|
```html
|
||||||
|
<svg lucideHouse size="48" color="red" strokeWidth="1"></svg>
|
||||||
|
```
|
||||||
|
|
||||||
|
## Global configuration
|
||||||
|
|
||||||
|
You can use `provideLucideConfig` to configure the default property values as defined above:
|
||||||
|
|
||||||
|
```ts{2,7-9}
|
||||||
|
import { ApplicationConfig } from '@angular/core';
|
||||||
|
import { provideLucideConfig } from '@lucide/angular';
|
||||||
|
|
||||||
|
export const appConfig: ApplicationConfig = {
|
||||||
|
providers: [
|
||||||
|
// ...
|
||||||
|
provideLucideConfig({
|
||||||
|
strokeWidth: 1.5
|
||||||
|
}),
|
||||||
|
]
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
## Styling via CSS
|
||||||
|
|
||||||
|
Icons can also be styled by using custom CSS classes:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<svg lucideHousePlus class="my-icon"></svg>
|
||||||
|
```
|
||||||
|
|
||||||
|
```css
|
||||||
|
svg.my-icon {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
stroke-width: 3;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## With Lucide lab or custom icons
|
||||||
|
|
||||||
|
[Lucide lab](https://github.com/lucide-icons/lucide-lab) is a collection of icons that are not part of the Lucide main library.
|
||||||
|
|
||||||
|
While they aren't provided as standalone components, they can be still be passed to the `LucideIcon` component the same way as official icons:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<!-- Directly as LucideIconData: -->
|
||||||
|
<svg [lucideIcon]="CoconutIcon"></svg>
|
||||||
|
|
||||||
|
<!-- As a provided icon by name: -->
|
||||||
|
<svg lucideIcon="coconut"></svg>
|
||||||
|
```
|
||||||
|
|
||||||
|
```ts{2,6-7,11-12}
|
||||||
|
import { provideLucideIcons } from '@lucide/angular';
|
||||||
|
import { coconut } from '@lucide/lab';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
templateUrl: './foobar.html',
|
||||||
|
// For using by name via provider:
|
||||||
|
providers: [provideLucideIcons({ coconut })],
|
||||||
|
imports: [LucideIcon]
|
||||||
|
})
|
||||||
|
export class Foobar {
|
||||||
|
// For passing directly as LucideIconData:
|
||||||
|
readonly CoconutIcon = coconut;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### The icon is not being displayed
|
||||||
|
If using per-icon-components:
|
||||||
|
1. Ensure that the icon component is being imported, if using per-icon-components
|
||||||
|
2. Check that the icon name matches exactly (case-sensitive)
|
||||||
|
|
||||||
|
If using the dynamic component:
|
||||||
|
1. Ensure the icon is provided via `provideLucideIcons()` if using string names
|
||||||
|
2. Verify the icon is imported from `@lucide/angular` and not the legacy package
|
||||||
|
|
||||||
|
### TypeScript errors?
|
||||||
|
Make sure you're importing from `@lucide/angular` and not `lucide-angular`.
|
||||||
|
|
||||||
|
### Icons render with wrong defaults
|
||||||
|
Ensure `provideLucideConfig()` is used at the right level.
|
||||||
|
|
||||||
|
## Migration guide
|
||||||
|
Migrating from `lucide-angular`? Read our [comprehensive migration guide](https://github.com/lucide-icons/lucide/blob/main/packages/angular/MIGRATION.md).
|
||||||
@@ -1,5 +1,11 @@
|
|||||||
# Lucide Angular
|
# Lucide Angular
|
||||||
|
|
||||||
|
::: warning
|
||||||
|
This documentation if for our legacy package for Angular.
|
||||||
|
|
||||||
|
For our modern, standalone-first implementation, please refer to [`@lucide/angular`](./angular).
|
||||||
|
:::
|
||||||
|
|
||||||
Angular components and services for Lucide icons that integrate with Angular's dependency injection and component system. Provides both traditional module-based and modern standalone component approaches for maximum flexibility in Angular applications.
|
Angular components and services for Lucide icons that integrate with Angular's dependency injection and component system. Provides both traditional module-based and modern standalone component approaches for maximum flexibility in Angular applications.
|
||||||
|
|
||||||
**What you can accomplish:**
|
**What you can accomplish:**
|
||||||
|
|||||||
@@ -102,10 +102,16 @@ The example below imports all ES Modules, so exercise caution when using it. Imp
|
|||||||
|
|
||||||
### Icon Component Example
|
### Icon Component Example
|
||||||
|
|
||||||
```jsx
|
```tsx
|
||||||
import { icons } from 'lucide-react-native';
|
import * as icons from 'lucide-react-native/icons';
|
||||||
|
|
||||||
const Icon = ({ name, color, size }) => {
|
interface IconProps {
|
||||||
|
name: keyof typeof icons;
|
||||||
|
color?: string;
|
||||||
|
size?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
const Icon = ({ name, color, size }: IconProps) => {
|
||||||
const LucideIcon = icons[name];
|
const LucideIcon = icons[name];
|
||||||
|
|
||||||
return <LucideIcon color={color} size={size} />;
|
return <LucideIcon color={color} size={size} />;
|
||||||
@@ -116,11 +122,11 @@ export default Icon;
|
|||||||
|
|
||||||
#### Using the Icon Component
|
#### Using the Icon Component
|
||||||
|
|
||||||
```jsx
|
```tsx
|
||||||
import Icon from './Icon';
|
import Icon from './Icon';
|
||||||
|
|
||||||
const App = () => {
|
const App = () => {
|
||||||
return <Icon name="house" />;
|
return <Icon name="House" />;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default App;
|
export default App;
|
||||||
|
|||||||
@@ -9,13 +9,14 @@
|
|||||||
"docs:build": "pnpm run /^prebuild:.*/ && vitepress build",
|
"docs:build": "pnpm run /^prebuild:.*/ && vitepress build",
|
||||||
"docs:preview": "vitepress preview",
|
"docs:preview": "vitepress preview",
|
||||||
"build:docs": "vitepress build",
|
"build:docs": "vitepress build",
|
||||||
"prebuild:iconNodes": "node --experimental-strip-types ./scripts/writeIconNodes.mjs",
|
"prebuild:iconNodes": "node ./scripts/writeIconNodes.mjs",
|
||||||
"prebuild:metaJson": "node --experimental-strip-types ./scripts/writeIconMetaIndex.mjs",
|
"prebuild:metaJson": "node ./scripts/writeIconMetaIndex.mjs",
|
||||||
"prebuild:releaseJson": "node --experimental-strip-types ./scripts/writeReleaseMetadata.mjs",
|
"prebuild:releaseJson": "node ./scripts/writeReleaseMetadata.mjs",
|
||||||
"prebuild:categoriesJson": "node --experimental-strip-types ./scripts/writeCategoriesMetadata.mjs",
|
"prebuild:categoriesJson": "node ./scripts/writeCategoriesMetadata.mjs",
|
||||||
"prebuild:relatedIcons": "node --experimental-strip-types ./scripts/writeIconRelatedIcons.mjs",
|
"prebuild:relatedIcons": "node ./scripts/writeIconRelatedIcons.mjs",
|
||||||
"prebuild:iconDetails": "node --experimental-strip-types ./scripts/writeIconDetails.mjs",
|
"prebuild:iconDetails": "node ./scripts/writeIconDetails.mjs",
|
||||||
"postbuild:vercelJson": "node --experimental-strip-types ./scripts/writeVercelOutput.mjs",
|
"prebuild:brandStopwords": "node ./scripts/writeBrandStopwords.mjs",
|
||||||
|
"postbuild:vercelJson": "node ./scripts/writeVercelOutput.mjs",
|
||||||
"dev": "npx nitropack dev",
|
"dev": "npx nitropack dev",
|
||||||
"prebuild:api": "npx nitropack prepare",
|
"prebuild:api": "npx nitropack prepare",
|
||||||
"build:api": "npx nitropack build",
|
"build:api": "npx nitropack build",
|
||||||
@@ -28,35 +29,35 @@
|
|||||||
"@lucide/build-icons": "workspace:*",
|
"@lucide/build-icons": "workspace:*",
|
||||||
"@lucide/helpers": "workspace:*",
|
"@lucide/helpers": "workspace:*",
|
||||||
"@lucide/shared": "workspace:*",
|
"@lucide/shared": "workspace:*",
|
||||||
"@rollup/plugin-replace": "^6.0.2",
|
"@rollup/plugin-replace": "^6.0.3",
|
||||||
"@types/semver": "^7.5.3",
|
"@types/semver": "^7.7.1",
|
||||||
"nitropack": "2.8.1",
|
"nitropack": "2.8.1",
|
||||||
"rollup-plugin-copy": "^3.5.0",
|
"rollup-plugin-copy": "^3.5.0",
|
||||||
"vitepress": "^1.6.3",
|
"svg-path-commander": "^2.1.11",
|
||||||
"svg-path-commander": "^2.1.11"
|
"vitepress": "^1.6.4"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@floating-ui/vue": "^1.0.3",
|
"@floating-ui/vue": "^1.1.9",
|
||||||
"@headlessui/vue": "^1.7.17",
|
"@headlessui/vue": "^1.7.23",
|
||||||
"@resvg/resvg-wasm": "^2.6.2",
|
"@resvg/resvg-wasm": "^2.6.2",
|
||||||
"@vueuse/components": "^12.0.0",
|
"@vueuse/components": "^14.0.0",
|
||||||
"@vueuse/core": "^12.0.0",
|
"@vueuse/core": "^14.0.0",
|
||||||
"element-to-path": "^1.2.1",
|
"element-to-path": "^1.2.1",
|
||||||
"fuse.js": "^6.5.3",
|
"fuse.js": "^7.1.0",
|
||||||
"jszip": "^3.7.0",
|
"jszip": "^3.10.1",
|
||||||
"lodash": "^4.17.20",
|
"lodash": "^4.17.21",
|
||||||
"lodash-es": "^4.17.21",
|
"lodash-es": "^4.17.21",
|
||||||
"lucide-react": "workspace:*",
|
"lucide-react": "workspace:*",
|
||||||
"lucide-vue-next": "workspace:*",
|
"lucide-vue-next": "workspace:*",
|
||||||
"react": "^18.2.0",
|
"react": "^18.3.1",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.3.1",
|
||||||
"sandpack-vue3": "3.1.11",
|
"sandpack-vue3": "3.1.11",
|
||||||
"semver": "^7.5.2",
|
"semver": "^7.7.3",
|
||||||
"shikiji": "^0.7.4",
|
"shiki": "^3.15.0",
|
||||||
"simple-git": "^3.18.0",
|
"simple-git": "^3.30.0",
|
||||||
"sitemap": "^7.1.1",
|
"sitemap": "^7.1.2",
|
||||||
"svg-pathdata": "^6.0.3",
|
"svg-pathdata": "^6.0.3",
|
||||||
"svgson": "^5.2.1",
|
"svgson": "^5.3.1",
|
||||||
"vue": "^3.5.18"
|
"vue": "^3.5.24"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
15
docs/scripts/writeBrandStopwords.mjs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import fs from 'fs/promises';
|
||||||
|
import path from 'path';
|
||||||
|
|
||||||
|
const currentDir = process.cwd();
|
||||||
|
const dataDirectory = path.resolve(currentDir, '.vitepress/data');
|
||||||
|
const stopwordsSource = path.resolve(currentDir, `../brand-stopwords.json`);
|
||||||
|
const stopwordsFile = path.resolve(dataDirectory, `brandStopwords.json`);
|
||||||
|
|
||||||
|
fs.copyFile(stopwordsSource, stopwordsFile)
|
||||||
|
.then(() => {
|
||||||
|
console.log('Successfully copied brandStopwords.json file');
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
throw new Error(`Something went wrong generating the brandStopwords.json file,\n ${error}`);
|
||||||
|
});
|
||||||
22
icons/balloon.json
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"peteruithoven"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"party",
|
||||||
|
"festival",
|
||||||
|
"congratulations",
|
||||||
|
"celebration",
|
||||||
|
"decoration",
|
||||||
|
"colorful",
|
||||||
|
"floating",
|
||||||
|
"fun",
|
||||||
|
"birthday",
|
||||||
|
"event",
|
||||||
|
"entertainment"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"emoji"
|
||||||
|
]
|
||||||
|
}
|
||||||
15
icons/balloon.svg
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M12 16v1a2 2 0 0 0 2 2h1a2 2 0 0 1 2 2v1" />
|
||||||
|
<path d="M12 6a2 2 0 0 1 2 2" />
|
||||||
|
<path d="M18 8c0 4-3.5 8-6 8s-6-4-6-8a6 6 0 0 1 12 0" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 358 B |
25
icons/book-search.json
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"karsa-mistmere",
|
||||||
|
"Muhammad-Aqib-Bashir"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"reading",
|
||||||
|
"library",
|
||||||
|
"study",
|
||||||
|
"education",
|
||||||
|
"research",
|
||||||
|
"knowledge",
|
||||||
|
"discover",
|
||||||
|
"browsing",
|
||||||
|
"lookup",
|
||||||
|
"finding",
|
||||||
|
"scanning"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"text",
|
||||||
|
"development",
|
||||||
|
"gaming"
|
||||||
|
]
|
||||||
|
}
|
||||||
16
icons/book-search.svg
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M11 22H5.5a1 1 0 0 1 0-5h4.501" />
|
||||||
|
<path d="m21 22-1.879-1.878" />
|
||||||
|
<path d="M3 19.5v-15A2.5 2.5 0 0 1 5.5 2H18a1 1 0 0 1 1 1v8" />
|
||||||
|
<circle cx="17" cy="18" r="3" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 389 B |
@@ -1,7 +1,8 @@
|
|||||||
{
|
{
|
||||||
"$schema": "../icon.schema.json",
|
"$schema": "../icon.schema.json",
|
||||||
"contributors": [
|
"contributors": [
|
||||||
"karsa-mistmere"
|
"karsa-mistmere",
|
||||||
|
"jguddas"
|
||||||
],
|
],
|
||||||
"tags": [
|
"tags": [
|
||||||
"cleaning",
|
"cleaning",
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="m16 22-1-4" />
|
<path d="m16 22-1-4" />
|
||||||
<path d="M19 13.99a1 1 0 0 0 1-1V12a2 2 0 0 0-2-2h-3a1 1 0 0 1-1-1V4a2 2 0 0 0-4 0v5a1 1 0 0 1-1 1H6a2 2 0 0 0-2 2v.99a1 1 0 0 0 1 1" />
|
<path d="M19 14a1 1 0 0 0 1-1v-1a2 2 0 0 0-2-2h-3a1 1 0 0 1-1-1V4a2 2 0 0 0-4 0v5a1 1 0 0 1-1 1H6a2 2 0 0 0-2 2v1a1 1 0 0 0 1 1" />
|
||||||
<path d="M5 14h14l1.973 6.767A1 1 0 0 1 20 22H4a1 1 0 0 1-.973-1.233z" />
|
<path d="M19 14H5l-1.973 6.767A1 1 0 0 0 4 22h16a1 1 0 0 0 .973-1.233z" />
|
||||||
<path d="m8 22 1-4" />
|
<path d="m8 22 1-4" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 474 B After Width: | Height: | Size: 470 B |
@@ -2,7 +2,8 @@
|
|||||||
"$schema": "../icon.schema.json",
|
"$schema": "../icon.schema.json",
|
||||||
"contributors": [
|
"contributors": [
|
||||||
"vqh2602",
|
"vqh2602",
|
||||||
"jguddas"
|
"jguddas",
|
||||||
|
"karsa-mistmere"
|
||||||
],
|
],
|
||||||
"tags": [
|
"tags": [
|
||||||
"water",
|
"water",
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="M7.2 14.8a2 2 0 0 1 2 2" />
|
<path d="M7.001 15.085A1.5 1.5 0 0 1 9 16.5" />
|
||||||
<circle cx="18.5" cy="8.5" r="3.5" />
|
<circle cx="18.5" cy="8.5" r="3.5" />
|
||||||
<circle cx="7.5" cy="16.5" r="5.5" />
|
<circle cx="7.5" cy="16.5" r="5.5" />
|
||||||
<circle cx="7.5" cy="4.5" r="2.5" />
|
<circle cx="7.5" cy="4.5" r="2.5" />
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 366 B After Width: | Height: | Size: 377 B |
14
icons/cannabis-off.json
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"nickveles"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"cannabis",
|
||||||
|
"weed",
|
||||||
|
"leaf"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"nature"
|
||||||
|
]
|
||||||
|
}
|
||||||
18
icons/cannabis-off.svg
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M12 22v-4c1.5 1.5 3.5 3 6 3 0-1.5-.5-3.5-2-5" />
|
||||||
|
<path d="M13.988 8.327C13.902 6.054 13.365 3.82 12 2a9.3 9.3 0 0 0-1.445 2.9" />
|
||||||
|
<path d="M17.375 11.725C18.882 10.53 21 7.841 21 6c-2.324 0-5.08 1.296-6.662 2.684" />
|
||||||
|
<path d="m2 2 20 20" />
|
||||||
|
<path d="M21.024 15.378A15 15 0 0 0 22 15c-.426-1.279-2.67-2.557-4.25-2.907" />
|
||||||
|
<path d="M6.995 6.992C5.714 6.4 4.29 6 3 6c0 2 2.5 5 4 6-1.5 0-4.5 1.5-5 3 3.5 1.5 6 1 6 1-1.5 1.5-2 3.5-2 5 2.5 0 4.5-1.5 6-3" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 681 B |
29
icons/circle-pile.json
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"colebemis",
|
||||||
|
"nathan-de-pachtere"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"off",
|
||||||
|
"zero",
|
||||||
|
"record",
|
||||||
|
"shape",
|
||||||
|
"circle-pile",
|
||||||
|
"circle",
|
||||||
|
"pile",
|
||||||
|
"stack",
|
||||||
|
"layer",
|
||||||
|
"structure",
|
||||||
|
"form",
|
||||||
|
"group",
|
||||||
|
"collection",
|
||||||
|
"stock",
|
||||||
|
"inventory",
|
||||||
|
"materials",
|
||||||
|
"warehouse"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"shapes"
|
||||||
|
]
|
||||||
|
}
|
||||||
9
icons/circle-pile.svg
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor"
|
||||||
|
stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<circle cx="12" cy="19" r="2" />
|
||||||
|
<circle cx="12" cy="5" r="2" />
|
||||||
|
<circle cx="16" cy="12" r="2" />
|
||||||
|
<circle cx="20" cy="19" r="2" />
|
||||||
|
<circle cx="4" cy="19" r="2" />
|
||||||
|
<circle cx="8" cy="12" r="2" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 397 B |
33
icons/cloud-backup.json
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"ericfennis",
|
||||||
|
"jguddas",
|
||||||
|
"danielbayley",
|
||||||
|
"karsa-mistmere"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"storage",
|
||||||
|
"memory",
|
||||||
|
"bytes",
|
||||||
|
"servers",
|
||||||
|
"backup",
|
||||||
|
"timemachine",
|
||||||
|
"rotate",
|
||||||
|
"synchronize",
|
||||||
|
"synchronise",
|
||||||
|
"refresh",
|
||||||
|
"reconnect",
|
||||||
|
"transfer",
|
||||||
|
"data",
|
||||||
|
"security",
|
||||||
|
"upload",
|
||||||
|
"save",
|
||||||
|
"remote",
|
||||||
|
"safety"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"arrows",
|
||||||
|
"files"
|
||||||
|
]
|
||||||
|
}
|
||||||
15
icons/cloud-backup.svg
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M21 15.251A4.5 4.5 0 0 0 17.5 8h-1.79A7 7 0 1 0 3 13.607" />
|
||||||
|
<path d="M7 11v4h4" />
|
||||||
|
<path d="M8 19a5 5 0 0 0 9-3 4.5 4.5 0 0 0-4.5-4.5 4.82 4.82 0 0 0-3.41 1.41L7 15" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 393 B |
27
icons/cloud-sync.json
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"colebemis",
|
||||||
|
"csandman",
|
||||||
|
"ericfennis",
|
||||||
|
"karsa-mistmere"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"synchronize",
|
||||||
|
"synchronise",
|
||||||
|
"refresh",
|
||||||
|
"reconnect",
|
||||||
|
"transfer",
|
||||||
|
"backup",
|
||||||
|
"storage",
|
||||||
|
"upload",
|
||||||
|
"download",
|
||||||
|
"connection",
|
||||||
|
"network",
|
||||||
|
"data"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"arrows",
|
||||||
|
"files"
|
||||||
|
]
|
||||||
|
}
|
||||||
17
icons/cloud-sync.svg
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="m17 18-1.535 1.605a5 5 0 0 1-8-1.5" />
|
||||||
|
<path d="M17 22v-4h-4" />
|
||||||
|
<path d="M20.996 15.251A4.5 4.5 0 0 0 17.495 8h-1.79a7 7 0 1 0-12.709 5.607" />
|
||||||
|
<path d="M7 10v4h4" />
|
||||||
|
<path d="m7 14 1.535-1.605a5 5 0 0 1 8 1.5" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 442 B |
28
icons/fishing-hook.json
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"7ender",
|
||||||
|
"jguddas",
|
||||||
|
"karsa-mistmere",
|
||||||
|
"jamiemlaw"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"sea",
|
||||||
|
"boating",
|
||||||
|
"angler",
|
||||||
|
"bait",
|
||||||
|
"reel",
|
||||||
|
"tackle",
|
||||||
|
"marine",
|
||||||
|
"outdoors",
|
||||||
|
"fish",
|
||||||
|
"fishing",
|
||||||
|
"hook",
|
||||||
|
"sports",
|
||||||
|
"travel"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"sports",
|
||||||
|
"travel"
|
||||||
|
]
|
||||||
|
}
|
||||||
15
icons/fishing-hook.svg
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="m17.586 11.414-5.93 5.93a1 1 0 0 1-8-8l3.137-3.137a.707.707 0 0 1 1.207.5V10" />
|
||||||
|
<path d="M20.414 8.586 22 7" />
|
||||||
|
<circle cx="19" cy="10" r="2" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 369 B |
@@ -4,10 +4,17 @@
|
|||||||
"Andreto",
|
"Andreto",
|
||||||
"ericfennis",
|
"ericfennis",
|
||||||
"karsa-mistmere",
|
"karsa-mistmere",
|
||||||
"csandman"
|
"csandman",
|
||||||
|
"jamiemlaw"
|
||||||
],
|
],
|
||||||
"tags": [
|
"tags": [
|
||||||
"torch"
|
"torch",
|
||||||
|
"light",
|
||||||
|
"beam",
|
||||||
|
"emergency",
|
||||||
|
"safety",
|
||||||
|
"tool",
|
||||||
|
"bright"
|
||||||
],
|
],
|
||||||
"categories": [
|
"categories": [
|
||||||
"photography",
|
"photography",
|
||||||
|
|||||||
@@ -9,8 +9,9 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="M16 16v4a2 2 0 0 1-2 2h-4a2 2 0 0 1-2-2V10c0-2-2-2-2-4" />
|
<path d="M11.652 6H18" />
|
||||||
<path d="M7 2h11v4c0 2-2 2-2 4v1" />
|
<path d="M12 13v1" />
|
||||||
<line x1="11" x2="18" y1="6" y2="6" />
|
<path d="M16 16v4a2 2 0 0 1-2 2h-4a2 2 0 0 1-2-2v-8a4 4 0 0 0-.8-2.4l-.6-.8A3 3 0 0 1 6 7V6" />
|
||||||
<line x1="2" x2="22" y1="2" y2="22" />
|
<path d="m2 2 20 20" />
|
||||||
|
<path d="M7.649 2H17a1 1 0 0 1 1 1v4a3 3 0 0 1-.6 1.8l-.6.8a4 4 0 0 0-.55 1.007" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 399 B After Width: | Height: | Size: 470 B |
@@ -2,10 +2,17 @@
|
|||||||
"$schema": "../icon.schema.json",
|
"$schema": "../icon.schema.json",
|
||||||
"contributors": [
|
"contributors": [
|
||||||
"csandman",
|
"csandman",
|
||||||
"ericfennis"
|
"ericfennis",
|
||||||
|
"jamiemlaw"
|
||||||
],
|
],
|
||||||
"tags": [
|
"tags": [
|
||||||
"torch"
|
"torch",
|
||||||
|
"light",
|
||||||
|
"beam",
|
||||||
|
"emergency",
|
||||||
|
"safety",
|
||||||
|
"tool",
|
||||||
|
"bright"
|
||||||
],
|
],
|
||||||
"categories": [
|
"categories": [
|
||||||
"photography",
|
"photography",
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="M18 6c0 2-2 2-2 4v10a2 2 0 0 1-2 2h-4a2 2 0 0 1-2-2V10c0-2-2-2-2-4V2h12z" />
|
<path d="M12 13v1" />
|
||||||
<line x1="6" x2="18" y1="6" y2="6" />
|
<path d="M17 2a1 1 0 0 1 1 1v4a3 3 0 0 1-.6 1.8l-.6.8A4 4 0 0 0 16 12v8a2 2 0 0 1-2 2H10a2 2 0 0 1-2-2v-8a4 4 0 0 0-.8-2.4l-.6-.8A3 3 0 0 1 6 7V3a1 1 0 0 1 1-1z" />
|
||||||
<line x1="12" x2="12" y1="12" y2="12" />
|
<path d="M6 6h12" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 379 B After Width: | Height: | Size: 422 B |
21
icons/hd.json
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"ahtohbi4",
|
||||||
|
"jamiemlaw",
|
||||||
|
"karsa-mistmere",
|
||||||
|
"jguddas"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"tv",
|
||||||
|
"resolution",
|
||||||
|
"video",
|
||||||
|
"high definition",
|
||||||
|
"720p",
|
||||||
|
"1080p"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"devices",
|
||||||
|
"multimedia"
|
||||||
|
]
|
||||||
|
}
|
||||||
17
icons/hd.svg
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M10 12H6" />
|
||||||
|
<path d="M10 15V9" />
|
||||||
|
<path d="M14 14.5a.5.5 0 0 0 .5.5h1a2.5 2.5 0 0 0 2.5-2.5v-1A2.5 2.5 0 0 0 15.5 9h-1a.5.5 0 0 0-.5.5z" />
|
||||||
|
<path d="M6 15V9" />
|
||||||
|
<rect x="2" y="5" width="20" height="14" rx="2" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 440 B |
22
icons/layers-plus.json
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"juanisidoro",
|
||||||
|
"karsa-mistmere"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"stack",
|
||||||
|
"layers",
|
||||||
|
"add",
|
||||||
|
"new",
|
||||||
|
"increase",
|
||||||
|
"create",
|
||||||
|
"positive",
|
||||||
|
"copy",
|
||||||
|
"upgrade"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"design",
|
||||||
|
"layout"
|
||||||
|
]
|
||||||
|
}
|
||||||
17
icons/layers-plus.svg
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M12.83 2.18a2 2 0 0 0-1.66 0L2.6 6.08a1 1 0 0 0 0 1.83l8.58 3.91a2 2 0 0 0 .83.18 2 2 0 0 0 .83-.18l8.58-3.9a1 1 0 0 0 0-1.831z" />
|
||||||
|
<path d="M16 17h6" />
|
||||||
|
<path d="M19 14v6" />
|
||||||
|
<path d="M2 12a1 1 0 0 0 .58.91l8.6 3.91a2 2 0 0 0 .825.178" />
|
||||||
|
<path d="M2 17a1 1 0 0 0 .58.91l8.6 3.91a2 2 0 0 0 1.65 0l2.116-.962" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 539 B |
@@ -9,13 +9,15 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="M6 19v-3" />
|
<path d="M12 12v-2" />
|
||||||
<path d="M10 19v-3" />
|
<path d="M12 18v-2" />
|
||||||
<path d="M14 19v-3" />
|
<path d="M16 12v-2" />
|
||||||
<path d="M18 19v-3" />
|
<path d="M16 18v-2" />
|
||||||
<path d="M8 11V9" />
|
<path d="M2 11h1.5" />
|
||||||
<path d="M16 11V9" />
|
<path d="M20 18v-2" />
|
||||||
<path d="M12 11V9" />
|
<path d="M20.5 11H22" />
|
||||||
<path d="M2 15h20" />
|
<path d="M4 18v-2" />
|
||||||
<path d="M2 7a2 2 0 0 1 2-2h16a2 2 0 0 1 2 2v1.1a2 2 0 0 0 0 3.837V17a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2v-5.1a2 2 0 0 0 0-3.837Z" />
|
<path d="M8 12v-2" />
|
||||||
|
<path d="M8 18v-2" />
|
||||||
|
<rect x="2" y="6" width="20" height="10" rx="2" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 532 B After Width: | Height: | Size: 510 B |
@@ -9,15 +9,14 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
|
<path d="M10 12h4" />
|
||||||
|
<path d="M10 17h4" />
|
||||||
|
<path d="M10 7h4" />
|
||||||
<path d="M18 12h2" />
|
<path d="M18 12h2" />
|
||||||
<path d="M18 16h2" />
|
<path d="M18 18h2" />
|
||||||
<path d="M18 20h2" />
|
<path d="M18 6h2" />
|
||||||
<path d="M18 4h2" />
|
|
||||||
<path d="M18 8h2" />
|
|
||||||
<path d="M4 12h2" />
|
<path d="M4 12h2" />
|
||||||
<path d="M4 16h2" />
|
<path d="M4 18h2" />
|
||||||
<path d="M4 20h2" />
|
<path d="M4 6h2" />
|
||||||
<path d="M4 4h2" />
|
<rect x="6" y="2" width="12" height="20" rx="2" />
|
||||||
<path d="M4 8h2" />
|
|
||||||
<path d="M8 2a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2V4a2 2 0 0 0-2-2h-1.5c-.276 0-.494.227-.562.495a2 2 0 0 1-3.876 0C9.994 2.227 9.776 2 9.5 2z" />
|
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 598 B After Width: | Height: | Size: 471 B |
@@ -9,8 +9,8 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="m19 11-8-8-8.6 8.6a2 2 0 0 0 0 2.8l5.2 5.2c.8.8 2 .8 2.8 0L19 11Z" />
|
<path d="M11 7 6 2" />
|
||||||
<path d="m5 2 5 5" />
|
<path d="M18.992 12H2.041" />
|
||||||
<path d="M2 13h15" />
|
<path d="M21.145 18.38A3.34 3.34 0 0 1 20 16.5a3.3 3.3 0 0 1-1.145 1.88c-.575.46-.855 1.02-.855 1.595A2 2 0 0 0 20 22a2 2 0 0 0 2-2.025c0-.58-.285-1.13-.855-1.595" />
|
||||||
<path d="M22 20a2 2 0 1 1-4 0c0-1.6 1.7-2.4 2-4 .3 1.6 2 2.4 2 4Z" />
|
<path d="m8.5 4.5 2.148-2.148a1.205 1.205 0 0 1 1.704 0l7.296 7.296a1.205 1.205 0 0 1 0 1.704l-7.592 7.592a3.615 3.615 0 0 1-5.112 0l-3.888-3.888a3.615 3.615 0 0 1 0-5.112L5.67 7.33" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 409 B After Width: | Height: | Size: 622 B |
@@ -10,7 +10,11 @@
|
|||||||
"energy",
|
"energy",
|
||||||
"electronics",
|
"electronics",
|
||||||
"socket",
|
"socket",
|
||||||
"outlet"
|
"outlet",
|
||||||
|
"power",
|
||||||
|
"voltage",
|
||||||
|
"current",
|
||||||
|
"charger"
|
||||||
],
|
],
|
||||||
"categories": [
|
"categories": [
|
||||||
"devices",
|
"devices",
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="M12 22v-5" />
|
<path d="M12 22v-5" />
|
||||||
<path d="M9 8V2" />
|
|
||||||
<path d="M15 8V2" />
|
<path d="M15 8V2" />
|
||||||
<path d="M18 8v5a4 4 0 0 1-4 4h-4a4 4 0 0 1-4-4V8Z" />
|
<path d="M17 8a1 1 0 0 1 1 1v4a4 4 0 0 1-4 4h-4a4 4 0 0 1-4-4V9a1 1 0 0 1 1-1z" />
|
||||||
|
<path d="M9 8V2" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 335 B After Width: | Height: | Size: 363 B |
22
icons/scooter.json
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"Ahmed-Dghaies",
|
||||||
|
"karsa-mistmere"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"vehicle",
|
||||||
|
"drive",
|
||||||
|
"trip",
|
||||||
|
"journey",
|
||||||
|
"transport",
|
||||||
|
"electric",
|
||||||
|
"ride",
|
||||||
|
"urban",
|
||||||
|
"commute",
|
||||||
|
"speed"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"transportation"
|
||||||
|
]
|
||||||
|
}
|
||||||
16
icons/scooter.svg
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M21 4h-3.5l2 11.05" />
|
||||||
|
<path d="M6.95 17h5.142c.523 0 .95-.406 1.063-.916a6.5 6.5 0 0 1 5.345-5.009" />
|
||||||
|
<circle cx="19.5" cy="17.5" r="2.5" />
|
||||||
|
<circle cx="4.5" cy="17.5" r="2.5" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 405 B |
25
icons/search-alert.json
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"colebemis",
|
||||||
|
"ericfennis",
|
||||||
|
"jguddas",
|
||||||
|
"Veatec22"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"find",
|
||||||
|
"scan",
|
||||||
|
"magnifier",
|
||||||
|
"magnifying glass",
|
||||||
|
"stop",
|
||||||
|
"warning",
|
||||||
|
"alert",
|
||||||
|
"error",
|
||||||
|
"anomaly",
|
||||||
|
"lens"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"text",
|
||||||
|
"social"
|
||||||
|
]
|
||||||
|
}
|
||||||
16
icons/search-alert.svg
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<circle cx="11" cy="11" r="8" />
|
||||||
|
<path d="m21 21-4.3-4.3" />
|
||||||
|
<path d="M11 7v4" />
|
||||||
|
<path d="M11 15h.01" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 321 B |
@@ -1,7 +1,8 @@
|
|||||||
{
|
{
|
||||||
"$schema": "../icon.schema.json",
|
"$schema": "../icon.schema.json",
|
||||||
"contributors": [
|
"contributors": [
|
||||||
"danielbayley"
|
"danielbayley",
|
||||||
|
"eden881"
|
||||||
],
|
],
|
||||||
"tags": [
|
"tags": [
|
||||||
"cut",
|
"cut",
|
||||||
|
|||||||
@@ -9,12 +9,18 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<path d="M4 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h16a2 2 0 0 1 2 2v16a2 2 0 0 1-2 2" />
|
<line x1="5" y1="3" x2="19" y2="3" />
|
||||||
<path d="M10 22H8" />
|
<line x1="3" y1="5" x2="3" y2="19" />
|
||||||
<path d="M16 22h-2" />
|
<line x1="21" y1="5" x2="21" y2="19" />
|
||||||
<circle cx="8" cy="8" r="2" />
|
<line x1="9" y1="21" x2="10" y2="21" />
|
||||||
<path d="M9.414 9.414 12 12" />
|
<line x1="14" y1="21" x2="15" y2="21" />
|
||||||
<path d="M14.8 14.8 18 18" />
|
<path d="M 3 5 A2 2 0 0 1 5 3" />
|
||||||
<circle cx="8" cy="16" r="2" />
|
<path d="M 19 3 A2 2 0 0 1 21 5" />
|
||||||
<path d="m18 6-8.586 8.586" />
|
<path d="M 5 21 A2 2 0 0 1 3 19" />
|
||||||
|
<path d="M 21 19 A2 2 0 0 1 19 21" />
|
||||||
|
<circle cx="8.5" cy="8.5" r="1.5" />
|
||||||
|
<line x1="9.56066" y1="9.56066" x2="12" y2="12" />
|
||||||
|
<line x1="17" y1="17" x2="14.82" y2="14.82" />
|
||||||
|
<circle cx="8.5" cy="15.5" r="1.5" />
|
||||||
|
<line x1="9.56066" y1="14.43934" x2="17" y2="7" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 508 B After Width: | Height: | Size: 801 B |
@@ -1,7 +1,8 @@
|
|||||||
{
|
{
|
||||||
"$schema": "../icon.schema.json",
|
"$schema": "../icon.schema.json",
|
||||||
"contributors": [
|
"contributors": [
|
||||||
"danielbayley"
|
"danielbayley",
|
||||||
|
"eden881"
|
||||||
],
|
],
|
||||||
"tags": [
|
"tags": [
|
||||||
"cut",
|
"cut",
|
||||||
|
|||||||
@@ -9,10 +9,10 @@
|
|||||||
stroke-linecap="round"
|
stroke-linecap="round"
|
||||||
stroke-linejoin="round"
|
stroke-linejoin="round"
|
||||||
>
|
>
|
||||||
<rect width="20" height="20" x="2" y="2" rx="2" />
|
<rect width="18" height="18" x="3" y="3" rx="2" />
|
||||||
<circle cx="8" cy="8" r="2" />
|
<circle cx="8.5" cy="8.5" r="1.5" />
|
||||||
<path d="M9.414 9.414 12 12" />
|
<line x1="9.56066" y1="9.56066" x2="12" y2="12" />
|
||||||
<path d="M14.8 14.8 18 18" />
|
<line x1="17" y1="17" x2="14.82" y2="14.82" />
|
||||||
<circle cx="8" cy="16" r="2" />
|
<circle cx="8.5" cy="15.5" r="1.5" />
|
||||||
<path d="m18 6-8.586 8.586" />
|
<line x1="9.56066" y1="14.43934" x2="17" y2="7" />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 427 B After Width: | Height: | Size: 495 B |
24
icons/stone.json
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
"$schema": "../icon.schema.json",
|
||||||
|
"contributors": [
|
||||||
|
"Alportan",
|
||||||
|
"karsa-mistmere"
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"mineral",
|
||||||
|
"geology",
|
||||||
|
"nature",
|
||||||
|
"solid",
|
||||||
|
"pebble",
|
||||||
|
"crystal",
|
||||||
|
"ore",
|
||||||
|
"hard",
|
||||||
|
"coal",
|
||||||
|
"stone",
|
||||||
|
"rock",
|
||||||
|
"boulder"
|
||||||
|
],
|
||||||
|
"categories": [
|
||||||
|
"nature"
|
||||||
|
]
|
||||||
|
}
|
||||||