mirror of
https://github.com/iconify/iconify.git
synced 2025-01-23 07:08:34 +00:00
Support addIcon, addCollection and referencing icon by name in Svelte component
This commit is contained in:
parent
292ba1b7cb
commit
8b0c580eb2
@ -1,5 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import { Icon } from '@iconify/svelte';
|
import { Icon, addIcon } from '@iconify/svelte';
|
||||||
import adminCustomizer from '@iconify-icons/dashicons/admin-customizer';
|
import adminCustomizer from '@iconify-icons/dashicons/admin-customizer';
|
||||||
import bxUser from '@iconify-icons/bx/bx-user';
|
import bxUser from '@iconify-icons/bx/bx-user';
|
||||||
import bxError from '@iconify-icons/bx/bx-error';
|
import bxError from '@iconify-icons/bx/bx-error';
|
||||||
@ -17,7 +17,9 @@
|
|||||||
'<defs><path id="ssvg-id-1st-place-medala" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medald" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalf" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalh" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalj" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalm" d="M.93.01h120.55v58.36H.93z"/><path d="M52.849 78.373v-3.908c3.681-.359 6.25-.958 7.703-1.798c1.454-.84 2.54-2.828 3.257-5.962h4.021v40.385h-5.437V78.373h-9.544z" id="ssvg-id-1st-place-medalp"/><linearGradient x1="49.998%" y1="-13.249%" x2="49.998%" y2="90.002%" id="ssvg-id-1st-place-medalb"><stop stop-color="#1E88E5" offset="13.55%"/><stop stop-color="#1565C0" offset="93.8%"/></linearGradient><linearGradient x1="26.648%" y1="2.735%" x2="77.654%" y2="105.978%" id="ssvg-id-1st-place-medalk"><stop stop-color="#64B5F6" offset="13.55%"/><stop stop-color="#2196F3" offset="94.62%"/></linearGradient><radialGradient cx="22.368%" cy="12.5%" fx="22.368%" fy="12.5%" r="95.496%" id="ssvg-id-1st-place-medalo"><stop stop-color="#FFEB3B" offset="29.72%"/><stop stop-color="#FBC02D" offset="95.44%"/></radialGradient></defs><g fill="none" fill-rule="evenodd"><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medalc" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medala"/></mask><path fill="url(#ssvg-id-1st-place-medalb)" fill-rule="nonzero" mask="url(#ssvg-id-1st-place-medalc)" d="M45.44 42.18h31.43l30-48.43H75.44z"/></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medale" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medald"/></mask><g opacity=".2" mask="url(#ssvg-id-1st-place-medale)" fill="#424242" fill-rule="nonzero"><path d="M101.23-3L75.2 39H50.85L77.11-3h24.12zm5.64-3H75.44l-30 48h31.42l30.01-48z"/></g></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medalg" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalf"/></mask><path d="M79 30H43c-4.42 0-8 3.58-8 8v16.04c0 2.17 1.8 3.95 4.02 3.96h.01c2.23-.01 4.97-1.75 4.97-3.96V44c0-1.1.9-2 2-2h30c1.1 0 2 .9 2 2v9.93c0 1.98 2.35 3.68 4.22 4.04c.26.05.52.08.78.08c2.21 0 4-1.79 4-4V38c0-4.42-3.58-8-8-8z" fill="#FDD835" fill-rule="nonzero" mask="url(#ssvg-id-1st-place-medalg)"/></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medali" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalh"/></mask><g opacity=".2" mask="url(#ssvg-id-1st-place-medali)" fill="#424242" fill-rule="nonzero"><path d="M79 32c3.31 0 6 2.69 6 6v16.04A2.006 2.006 0 0 1 82.59 56c-1.18-.23-2.59-1.35-2.59-2.07V44c0-2.21-1.79-4-4-4H46c-2.21 0-4 1.79-4 4v10.04c0 .88-1.64 1.96-2.97 1.96c-1.12-.01-2.03-.89-2.03-1.96V38c0-3.31 2.69-6 6-6h36zm0-2H43c-4.42 0-8 3.58-8 8v16.04c0 2.17 1.8 3.95 4.02 3.96h.01c2.23-.01 4.97-1.75 4.97-3.96V44c0-1.1.9-2 2-2h30c1.1 0 2 .9 2 2v9.93c0 1.98 2.35 3.68 4.22 4.04c.26.05.52.08.78.08c2.21 0 4-1.79 4-4V38c0-4.42-3.58-8-8-8z"/></g></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medall" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalj"/></mask><path fill="url(#ssvg-id-1st-place-medalk)" fill-rule="nonzero" mask="url(#ssvg-id-1st-place-medall)" d="M76.87 42.18H45.44l-30-48.43h31.43z"/></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medaln" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalm"/></mask><g opacity=".2" mask="url(#ssvg-id-1st-place-medaln)" fill="#424242" fill-rule="nonzero"><path d="M45.1-3l26.35 42H47.1L20.86-3H45.1zm1.77-3H15.44l30 48h31.42L46.87-6z"/></g></g><circle fill="url(#ssvg-id-1st-place-medalo)" fill-rule="nonzero" cx="64" cy="86" r="38"/><path d="M64 51c19.3 0 35 15.7 35 35s-15.7 35-35 35s-35-15.7-35-35s15.7-35 35-35zm0-3c-20.99 0-38 17.01-38 38s17.01 38 38 38s38-17.01 38-38s-17.01-38-38-38z" opacity=".2" fill="#424242" fill-rule="nonzero"/><path d="M47.3 63.59h33.4v44.4H47.3z"/><use fill="#000" xlink:href="#ssvg-id-1st-place-medalp"/><use fill="#FFA000" xlink:href="#ssvg-id-1st-place-medalp"/></g>',
|
'<defs><path id="ssvg-id-1st-place-medala" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medald" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalf" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalh" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalj" d="M.93.01h120.55v58.36H.93z"/><path id="ssvg-id-1st-place-medalm" d="M.93.01h120.55v58.36H.93z"/><path d="M52.849 78.373v-3.908c3.681-.359 6.25-.958 7.703-1.798c1.454-.84 2.54-2.828 3.257-5.962h4.021v40.385h-5.437V78.373h-9.544z" id="ssvg-id-1st-place-medalp"/><linearGradient x1="49.998%" y1="-13.249%" x2="49.998%" y2="90.002%" id="ssvg-id-1st-place-medalb"><stop stop-color="#1E88E5" offset="13.55%"/><stop stop-color="#1565C0" offset="93.8%"/></linearGradient><linearGradient x1="26.648%" y1="2.735%" x2="77.654%" y2="105.978%" id="ssvg-id-1st-place-medalk"><stop stop-color="#64B5F6" offset="13.55%"/><stop stop-color="#2196F3" offset="94.62%"/></linearGradient><radialGradient cx="22.368%" cy="12.5%" fx="22.368%" fy="12.5%" r="95.496%" id="ssvg-id-1st-place-medalo"><stop stop-color="#FFEB3B" offset="29.72%"/><stop stop-color="#FBC02D" offset="95.44%"/></radialGradient></defs><g fill="none" fill-rule="evenodd"><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medalc" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medala"/></mask><path fill="url(#ssvg-id-1st-place-medalb)" fill-rule="nonzero" mask="url(#ssvg-id-1st-place-medalc)" d="M45.44 42.18h31.43l30-48.43H75.44z"/></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medale" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medald"/></mask><g opacity=".2" mask="url(#ssvg-id-1st-place-medale)" fill="#424242" fill-rule="nonzero"><path d="M101.23-3L75.2 39H50.85L77.11-3h24.12zm5.64-3H75.44l-30 48h31.42l30.01-48z"/></g></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medalg" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalf"/></mask><path d="M79 30H43c-4.42 0-8 3.58-8 8v16.04c0 2.17 1.8 3.95 4.02 3.96h.01c2.23-.01 4.97-1.75 4.97-3.96V44c0-1.1.9-2 2-2h30c1.1 0 2 .9 2 2v9.93c0 1.98 2.35 3.68 4.22 4.04c.26.05.52.08.78.08c2.21 0 4-1.79 4-4V38c0-4.42-3.58-8-8-8z" fill="#FDD835" fill-rule="nonzero" mask="url(#ssvg-id-1st-place-medalg)"/></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medali" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalh"/></mask><g opacity=".2" mask="url(#ssvg-id-1st-place-medali)" fill="#424242" fill-rule="nonzero"><path d="M79 32c3.31 0 6 2.69 6 6v16.04A2.006 2.006 0 0 1 82.59 56c-1.18-.23-2.59-1.35-2.59-2.07V44c0-2.21-1.79-4-4-4H46c-2.21 0-4 1.79-4 4v10.04c0 .88-1.64 1.96-2.97 1.96c-1.12-.01-2.03-.89-2.03-1.96V38c0-3.31 2.69-6 6-6h36zm0-2H43c-4.42 0-8 3.58-8 8v16.04c0 2.17 1.8 3.95 4.02 3.96h.01c2.23-.01 4.97-1.75 4.97-3.96V44c0-1.1.9-2 2-2h30c1.1 0 2 .9 2 2v9.93c0 1.98 2.35 3.68 4.22 4.04c.26.05.52.08.78.08c2.21 0 4-1.79 4-4V38c0-4.42-3.58-8-8-8z"/></g></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medall" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalj"/></mask><path fill="url(#ssvg-id-1st-place-medalk)" fill-rule="nonzero" mask="url(#ssvg-id-1st-place-medall)" d="M76.87 42.18H45.44l-30-48.43h31.43z"/></g><g transform="translate(3 4)"><mask id="ssvg-id-1st-place-medaln" fill="#fff"><use xlink:href="#ssvg-id-1st-place-medalm"/></mask><g opacity=".2" mask="url(#ssvg-id-1st-place-medaln)" fill="#424242" fill-rule="nonzero"><path d="M45.1-3l26.35 42H47.1L20.86-3H45.1zm1.77-3H15.44l30 48h31.42L46.87-6z"/></g></g><circle fill="url(#ssvg-id-1st-place-medalo)" fill-rule="nonzero" cx="64" cy="86" r="38"/><path d="M64 51c19.3 0 35 15.7 35 35s-15.7 35-35 35s-35-15.7-35-35s15.7-35 35-35zm0-3c-20.99 0-38 17.01-38 38s17.01 38 38 38s38-17.01 38-38s-17.01-38-38-38z" opacity=".2" fill="#424242" fill-rule="nonzero"/><path d="M47.3 63.59h33.4v44.4H47.3z"/><use fill="#000" xlink:href="#ssvg-id-1st-place-medalp"/><use fill="#FFA000" xlink:href="#ssvg-id-1st-place-medalp"/></g>',
|
||||||
width: 128,
|
width: 128,
|
||||||
height: 128,
|
height: 128,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
addIcon('demo', adminCustomizer);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
@ -89,6 +91,10 @@
|
|||||||
Simple icon:
|
Simple icon:
|
||||||
<Icon icon={bxUser} inline={true} />
|
<Icon icon={bxUser} inline={true} />
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
|
Icon refenced by name:
|
||||||
|
<Icon icon="demo" inline={true} />
|
||||||
|
</div>
|
||||||
<div class="alert">
|
<div class="alert">
|
||||||
<Icon icon={bxError} />
|
<Icon icon={bxError} />
|
||||||
Important notice with alert icon!
|
Important notice with alert icon!
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
"name": "@iconify/svelte",
|
"name": "@iconify/svelte",
|
||||||
"description": "Iconify icon component for Svelte.",
|
"description": "Iconify icon component for Svelte.",
|
||||||
"author": "Vjacheslav Trushkin",
|
"author": "Vjacheslav Trushkin",
|
||||||
"version": "1.0.4",
|
"version": "2.0.0-dev",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"bugs": "https://github.com/iconify/iconify/issues",
|
"bugs": "https://github.com/iconify/iconify/issues",
|
||||||
"homepage": "https://github.com/iconify/iconify",
|
"homepage": "https://github.com/iconify/iconify",
|
||||||
|
@ -10,7 +10,7 @@ import pkg from './package.json';
|
|||||||
try {
|
try {
|
||||||
fs.mkdirSync(__dirname + '/dist');
|
fs.mkdirSync(__dirname + '/dist');
|
||||||
} catch (err) {}
|
} catch (err) {}
|
||||||
['Icon.svelte'].forEach((file) => {
|
['OfflineIcon.svelte'].forEach((file) => {
|
||||||
fs.writeFileSync(
|
fs.writeFileSync(
|
||||||
__dirname + '/dist/' + file,
|
__dirname + '/dist/' + file,
|
||||||
fs.readFileSync(__dirname + '/src/' + file)
|
fs.readFileSync(__dirname + '/src/' + file)
|
||||||
@ -48,12 +48,12 @@ export default [
|
|||||||
commonjs(),
|
commonjs(),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
// Files included in Icon.svelte as bundles without dependencies
|
// Files included in OfflineIcon.svelte as bundles without dependencies
|
||||||
{
|
{
|
||||||
input: 'src/generate-icon.ts',
|
input: 'src/offline.ts',
|
||||||
output: [
|
output: [
|
||||||
{
|
{
|
||||||
file: 'dist/generate-icon.js',
|
file: 'dist/offline.js',
|
||||||
format: 'es',
|
format: 'es',
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import { generateIcon } from './generate-icon';
|
import { generateIcon } from './offline';
|
||||||
|
|
||||||
// Generated data
|
// Generated data
|
||||||
let data;
|
let data;
|
||||||
@ -9,6 +9,10 @@
|
|||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
{#if data === null}
|
||||||
|
<slot />
|
||||||
|
{:else}
|
||||||
<svg {...data.attributes}>
|
<svg {...data.attributes}>
|
||||||
{@html data.body}
|
{@html data.body}
|
||||||
</svg>
|
</svg>
|
||||||
|
{/if}
|
@ -1 +1,2 @@
|
|||||||
export { default as Icon } from './Icon.svelte';
|
export { default as Icon } from './OfflineIcon.svelte';
|
||||||
|
export { addIcon, addCollection } from './offline';
|
||||||
|
68
packages/svelte/src/offline.ts
Normal file
68
packages/svelte/src/offline.ts
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import type { IconifyIcon, IconifyJSON } from '@iconify/types';
|
||||||
|
import { fullIcon } from '@iconify/core/lib/icon';
|
||||||
|
import { parseIconSet } from '@iconify/core/lib/icon/icon-set';
|
||||||
|
import { render } from './render';
|
||||||
|
import type { RenderResult } from './render';
|
||||||
|
import type { IconProps } from './props';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Storage for icons referred by name
|
||||||
|
*/
|
||||||
|
const storage: Record<string, Required<IconifyIcon>> = Object.create(null);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate icon
|
||||||
|
*/
|
||||||
|
export function generateIcon(props: IconProps): RenderResult | null {
|
||||||
|
// Split properties
|
||||||
|
const icon =
|
||||||
|
typeof props.icon === 'string'
|
||||||
|
? storage[props.icon]
|
||||||
|
: typeof props.icon === 'object'
|
||||||
|
? fullIcon(props.icon)
|
||||||
|
: null;
|
||||||
|
|
||||||
|
// Validate icon object
|
||||||
|
if (
|
||||||
|
icon === null ||
|
||||||
|
typeof icon !== 'object' ||
|
||||||
|
typeof icon.body !== 'string'
|
||||||
|
) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return render(icon, props);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add icon to storage, allowing to call it by name
|
||||||
|
*
|
||||||
|
* @param name
|
||||||
|
* @param data
|
||||||
|
*/
|
||||||
|
export function addIcon(name: string, data: IconifyIcon): void {
|
||||||
|
storage[name] = fullIcon(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add collection to storage, allowing to call icons by name
|
||||||
|
*
|
||||||
|
* @param data Icon set
|
||||||
|
* @param prefix Optional prefix to add to icon names, true (default) if prefix from icon set should be used.
|
||||||
|
*/
|
||||||
|
export function addCollection(
|
||||||
|
data: IconifyJSON,
|
||||||
|
prefix?: string | boolean
|
||||||
|
): void {
|
||||||
|
const iconPrefix: string =
|
||||||
|
typeof prefix === 'string'
|
||||||
|
? prefix
|
||||||
|
: prefix !== false && typeof data.prefix === 'string'
|
||||||
|
? data.prefix + ':'
|
||||||
|
: '';
|
||||||
|
parseIconSet(data, (name, icon) => {
|
||||||
|
if (icon !== null) {
|
||||||
|
storage[iconPrefix + name] = icon;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
@ -1,3 +1,4 @@
|
|||||||
|
import type { IconifyIcon } from '@iconify/types';
|
||||||
import { defaults } from '@iconify/core/lib/customisations';
|
import { defaults } from '@iconify/core/lib/customisations';
|
||||||
import {
|
import {
|
||||||
flipFromString,
|
flipFromString,
|
||||||
@ -23,7 +24,7 @@ const svgDefaults = {
|
|||||||
/**
|
/**
|
||||||
* Result
|
* Result
|
||||||
*/
|
*/
|
||||||
export interface GenerateIconResult {
|
export interface RenderResult {
|
||||||
attributes: Record<string, unknown>;
|
attributes: Record<string, unknown>;
|
||||||
body: string;
|
body: string;
|
||||||
}
|
}
|
||||||
@ -31,15 +32,12 @@ export interface GenerateIconResult {
|
|||||||
/**
|
/**
|
||||||
* Generate icon from properties
|
* Generate icon from properties
|
||||||
*/
|
*/
|
||||||
export function generateIcon(props: IconProps): GenerateIconResult {
|
export function render(
|
||||||
let iconData = fullIcon(props.icon);
|
// Icon must be validated before calling this function
|
||||||
if (!iconData) {
|
icon: Required<IconifyIcon>,
|
||||||
return {
|
// Properties
|
||||||
attributes: svgDefaults,
|
props: IconProps
|
||||||
body: '',
|
): RenderResult {
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const customisations = merge(defaults, props as typeof defaults);
|
const customisations = merge(defaults, props as typeof defaults);
|
||||||
const componentProps = merge(svgDefaults) as Record<string, unknown>;
|
const componentProps = merge(svgDefaults) as Record<string, unknown>;
|
||||||
|
|
||||||
@ -100,7 +98,7 @@ export function generateIcon(props: IconProps): GenerateIconResult {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Generate icon
|
// Generate icon
|
||||||
const item = iconToSVG(iconData, customisations);
|
const item = iconToSVG(icon, customisations);
|
||||||
|
|
||||||
// Add icon stuff
|
// Add icon stuff
|
||||||
for (let key in item.attributes) {
|
for (let key in item.attributes) {
|
Loading…
x
Reference in New Issue
Block a user