valtech-components 2.0.434 → 2.0.435
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/atoms/fab/fab.component.mjs +5 -4
- package/esm2022/lib/components/molecules/comment-input/comment-input.component.mjs +13 -5
- package/esm2022/lib/components/molecules/comment-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +23 -4
- package/esm2022/lib/components/molecules/date-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/feedback-form/feedback-form.component.mjs +352 -0
- package/esm2022/lib/components/molecules/feedback-form/types.mjs +2 -0
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +7 -7
- package/esm2022/lib/components/molecules/file-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/number-from-to/number-from-to.component.mjs +23 -9
- package/esm2022/lib/components/molecules/number-from-to/types.mjs +2 -0
- package/esm2022/lib/components/molecules/pin-input/pin-input.component.mjs +2 -2
- package/esm2022/lib/components/molecules/pin-input/types.mjs +2 -0
- package/esm2022/lib/components/organisms/form/form.component.mjs +108 -30
- package/esm2022/lib/services/content/content-types/blog.mjs +275 -0
- package/esm2022/lib/services/content/content-types/documentation.mjs +303 -0
- package/esm2022/lib/services/content/content-types/news.mjs +277 -0
- package/esm2022/lib/services/content/index.mjs +51 -0
- package/esm2022/lib/services/content/transformer.mjs +265 -0
- package/esm2022/lib/services/content/types.mjs +41 -0
- package/esm2022/lib/services/feedback/config.mjs +49 -0
- package/esm2022/lib/services/feedback/feedback.service.mjs +174 -0
- package/esm2022/lib/services/feedback/index.mjs +44 -0
- package/esm2022/lib/services/feedback/types.mjs +30 -0
- package/esm2022/lib/services/firebase/index.mjs +3 -1
- package/esm2022/lib/services/firebase/shared-config.mjs +132 -0
- package/esm2022/public-api.mjs +14 -1
- package/fesm2022/valtech-components.mjs +2225 -177
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/fab/fab.component.d.ts +2 -0
- package/lib/components/molecules/comment-input/comment-input.component.d.ts +4 -4
- package/lib/components/molecules/comment-input/types.d.ts +59 -0
- package/lib/components/molecules/date-input/date-input.component.d.ts +4 -3
- package/lib/components/molecules/date-input/types.d.ts +74 -0
- package/lib/components/molecules/feedback-form/feedback-form.component.d.ts +56 -0
- package/lib/components/molecules/feedback-form/types.d.ts +54 -0
- package/lib/components/molecules/file-input/file-input.component.d.ts +5 -3
- package/lib/components/molecules/file-input/types.d.ts +72 -0
- package/lib/components/molecules/number-from-to/number-from-to.component.d.ts +8 -2
- package/lib/components/molecules/number-from-to/types.d.ts +76 -0
- package/lib/components/molecules/pin-input/pin-input.component.d.ts +4 -3
- package/lib/components/molecules/pin-input/types.d.ts +63 -0
- package/lib/components/organisms/form/form.component.d.ts +16 -2
- package/lib/services/content/content-types/blog.d.ts +148 -0
- package/lib/services/content/content-types/documentation.d.ts +183 -0
- package/lib/services/content/content-types/news.d.ts +162 -0
- package/lib/services/content/index.d.ts +49 -0
- package/lib/services/content/transformer.d.ts +96 -0
- package/lib/services/content/types.d.ts +220 -0
- package/lib/services/feedback/config.d.ts +35 -0
- package/lib/services/feedback/feedback.service.d.ts +76 -0
- package/lib/services/feedback/index.d.ts +40 -0
- package/lib/services/feedback/types.d.ts +107 -0
- package/lib/services/firebase/index.d.ts +1 -0
- package/lib/services/firebase/shared-config.d.ts +120 -0
- package/package.json +1 -1
- package/public-api.d.ts +9 -0
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Content Types System
|
|
3
|
+
*
|
|
4
|
+
* This module provides a content abstraction layer that transforms
|
|
5
|
+
* structured content documents (blog posts, documentation, news articles)
|
|
6
|
+
* into ArticleMetadata for rendering with val-article component.
|
|
7
|
+
*/
|
|
8
|
+
// =============================================================================
|
|
9
|
+
// TYPE GUARDS
|
|
10
|
+
// =============================================================================
|
|
11
|
+
/**
|
|
12
|
+
* Type guard to check if a block is a heading
|
|
13
|
+
*/
|
|
14
|
+
export function isHeadingBlock(block) {
|
|
15
|
+
return block.type === 'heading';
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Type guard to check if a block is a paragraph
|
|
19
|
+
*/
|
|
20
|
+
export function isParagraphBlock(block) {
|
|
21
|
+
return block.type === 'paragraph';
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Type guard to check if a block is a code block
|
|
25
|
+
*/
|
|
26
|
+
export function isCodeBlock(block) {
|
|
27
|
+
return block.type === 'code';
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Type guard to check if a block is a list
|
|
31
|
+
*/
|
|
32
|
+
export function isListBlock(block) {
|
|
33
|
+
return block.type === 'list';
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Type guard to check if a block is a callout
|
|
37
|
+
*/
|
|
38
|
+
export function isCalloutBlock(block) {
|
|
39
|
+
return block.type === 'callout';
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/content/types.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAwOH,gFAAgF;AAChF,cAAc;AACd,gFAAgF;AAEhF;;GAEG;AACH,MAAM,UAAU,cAAc,CAAC,KAAmB;IAChD,OAAO,KAAK,CAAC,IAAI,KAAK,SAAS,CAAC;AAClC,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,gBAAgB,CAAC,KAAmB;IAClD,OAAO,KAAK,CAAC,IAAI,KAAK,WAAW,CAAC;AACpC,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,WAAW,CAAC,KAAmB;IAC7C,OAAO,KAAK,CAAC,IAAI,KAAK,MAAM,CAAC;AAC/B,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,WAAW,CAAC,KAAmB;IAC7C,OAAO,KAAK,CAAC,IAAI,KAAK,MAAM,CAAC;AAC/B,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,cAAc,CAAC,KAAmB;IAChD,OAAO,KAAK,CAAC,IAAI,KAAK,SAAS,CAAC;AAClC,CAAC","sourcesContent":["/**\n * Content Types System\n *\n * This module provides a content abstraction layer that transforms\n * structured content documents (blog posts, documentation, news articles)\n * into ArticleMetadata for rendering with val-article component.\n */\n\n// =============================================================================\n// AUTHOR & META\n// =============================================================================\n\n/**\n * Author information for content documents\n */\nexport interface ContentAuthor {\n  /** Author's display name */\n  name: string;\n  /** URL to author's avatar image */\n  avatar?: string;\n  /** Author's role or title */\n  role?: string;\n  /** Link to author's profile */\n  profileUrl?: string;\n}\n\n/**\n * Common metadata for all content documents\n */\nexport interface ContentMeta {\n  /** Unique identifier */\n  id?: string;\n  /** URL-friendly slug */\n  slug?: string;\n  /** Creation date */\n  createdAt?: Date | string;\n  /** Last update date */\n  updatedAt?: Date | string;\n  /** Publication date */\n  publishedAt?: Date | string;\n  /** Content author */\n  author?: ContentAuthor;\n  /** Tags for categorization */\n  tags?: string[];\n  /** Primary category */\n  category?: string;\n  /** Content locale (e.g., 'es', 'en') */\n  locale?: string;\n}\n\n// =============================================================================\n// CONTENT BLOCKS\n// =============================================================================\n\n/**\n * Heading block - renders as title or subtitle\n */\nexport interface HeadingBlock {\n  type: 'heading';\n  /** Heading level: 1 = title, 2-3 = subtitle */\n  level: 1 | 2 | 3;\n  /** Heading text */\n  text: string;\n}\n\n/**\n * Paragraph block - renders as text\n */\nexport interface ParagraphBlock {\n  type: 'paragraph';\n  /** Paragraph text content */\n  text: string;\n  /** Apply emphasis styling */\n  emphasis?: boolean;\n}\n\n/**\n * Quote block - renders as blockquote\n */\nexport interface QuoteBlock {\n  type: 'quote';\n  /** Quote text */\n  text: string;\n  /** Quote author name */\n  author?: string;\n  /** Quote source */\n  source?: string;\n}\n\n/**\n * Code block - renders as syntax-highlighted code\n */\nexport interface CodeBlock {\n  type: 'code';\n  /** Code content */\n  code: string;\n  /** Programming language for syntax highlighting */\n  language?: string;\n  /** Code block title/filename */\n  title?: string;\n}\n\n/**\n * List block - renders as ordered, unordered, or checklist\n */\nexport interface ListBlock {\n  type: 'list';\n  /** List items (simple strings) */\n  items: string[];\n  /** Render as numbered list */\n  ordered?: boolean;\n  /** Render as checklist with checkmarks */\n  checklist?: boolean;\n}\n\n/**\n * Image block - renders as image with optional caption\n */\nexport interface ImageBlock {\n  type: 'image';\n  /** Image source URL */\n  src: string;\n  /** Alt text for accessibility */\n  alt: string;\n  /** Image caption */\n  caption?: string;\n  /** Image alignment */\n  alignment?: 'left' | 'center' | 'right';\n}\n\n/**\n * Callout block - renders as highlighted note/warning\n */\nexport interface CalloutBlock {\n  type: 'callout';\n  /** Callout text content */\n  text: string;\n  /** Callout variant determines styling */\n  variant: 'info' | 'warning' | 'success' | 'error';\n  /** Optional title/prefix */\n  title?: string;\n}\n\n/**\n * Divider block - renders as separator line\n */\nexport interface DividerBlock {\n  type: 'divider';\n  /** Divider style */\n  style?: 'line' | 'dots' | 'space';\n}\n\n/**\n * Button block - renders as call-to-action button\n */\nexport interface ButtonBlock {\n  type: 'button';\n  /** Button text */\n  text: string;\n  /** Link URL (for navigation) */\n  href?: string;\n  /** Action identifier (for event handling) */\n  action?: string;\n  /** Button color */\n  color?: 'primary' | 'secondary' | 'success' | 'warning' | 'danger';\n  /** Button alignment */\n  alignment?: 'left' | 'center' | 'right';\n}\n\n/**\n * Command block - renders as terminal/CLI command\n */\nexport interface CommandBlock {\n  type: 'command';\n  /** Command to display */\n  command: string;\n  /** Show copy button */\n  copyable?: boolean;\n}\n\n/**\n * Union type of all content blocks\n */\nexport type ContentBlock =\n  | HeadingBlock\n  | ParagraphBlock\n  | QuoteBlock\n  | CodeBlock\n  | ListBlock\n  | ImageBlock\n  | CalloutBlock\n  | DividerBlock\n  | ButtonBlock\n  | CommandBlock;\n\n// =============================================================================\n// CONTENT DOCUMENT\n// =============================================================================\n\n/**\n * Configuration options for content rendering\n */\nexport interface ContentConfig {\n  /** Article theme */\n  theme?: 'light' | 'dark' | 'auto';\n  /** Maximum width of the content container */\n  maxWidth?: string;\n  /** Show metadata (author, date, etc.) */\n  showMeta?: boolean;\n  /** Show table of contents */\n  showTableOfContents?: boolean;\n  /** Center the content container */\n  centered?: boolean;\n}\n\n/**\n * Base interface for all content documents.\n * Generic type T represents the document type discriminator.\n *\n * @example\n * ```typescript\n * interface BlogPost extends ContentDocument<'blog'> {\n *   title: string;\n *   excerpt: string;\n * }\n * ```\n */\nexport interface ContentDocument<T extends string = string> {\n  /** Document type discriminator */\n  type: T;\n  /** Document metadata */\n  meta: ContentMeta;\n  /** Array of content blocks */\n  content: ContentBlock[];\n  /** Rendering configuration */\n  config?: ContentConfig;\n}\n\n// =============================================================================\n// TYPE GUARDS\n// =============================================================================\n\n/**\n * Type guard to check if a block is a heading\n */\nexport function isHeadingBlock(block: ContentBlock): block is HeadingBlock {\n  return block.type === 'heading';\n}\n\n/**\n * Type guard to check if a block is a paragraph\n */\nexport function isParagraphBlock(block: ContentBlock): block is ParagraphBlock {\n  return block.type === 'paragraph';\n}\n\n/**\n * Type guard to check if a block is a code block\n */\nexport function isCodeBlock(block: ContentBlock): block is CodeBlock {\n  return block.type === 'code';\n}\n\n/**\n * Type guard to check if a block is a list\n */\nexport function isListBlock(block: ContentBlock): block is ListBlock {\n  return block.type === 'list';\n}\n\n/**\n * Type guard to check if a block is a callout\n */\nexport function isCalloutBlock(block: ContentBlock): block is CalloutBlock {\n  return block.type === 'callout';\n}\n"]}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { InjectionToken, makeEnvironmentProviders, } from '@angular/core';
|
|
2
|
+
/**
|
|
3
|
+
* Token de inyección para la configuración de Feedback.
|
|
4
|
+
*/
|
|
5
|
+
export const VALTECH_FEEDBACK_CONFIG = new InjectionToken('ValtechFeedbackConfig');
|
|
6
|
+
/**
|
|
7
|
+
* Configuración por defecto.
|
|
8
|
+
*/
|
|
9
|
+
export const DEFAULT_FEEDBACK_CONFIG = {
|
|
10
|
+
feedbackPrefix: '/v1/feedback',
|
|
11
|
+
maxAttachments: 5,
|
|
12
|
+
maxFileSize: 10 * 1024 * 1024, // 10MB
|
|
13
|
+
allowedFileTypes: ['image/*', 'video/*', 'application/pdf'],
|
|
14
|
+
storagePath: 'feedback',
|
|
15
|
+
};
|
|
16
|
+
/**
|
|
17
|
+
* Provee el servicio de feedback a la aplicación Angular.
|
|
18
|
+
*
|
|
19
|
+
* @param config - Configuración de feedback
|
|
20
|
+
* @returns EnvironmentProviders para usar en bootstrapApplication
|
|
21
|
+
*
|
|
22
|
+
* @example
|
|
23
|
+
* ```typescript
|
|
24
|
+
* // main.ts
|
|
25
|
+
* import { bootstrapApplication } from '@angular/platform-browser';
|
|
26
|
+
* import { provideValtechFeedback } from 'valtech-components';
|
|
27
|
+
* import { environment } from './environments/environment';
|
|
28
|
+
*
|
|
29
|
+
* bootstrapApplication(AppComponent, {
|
|
30
|
+
* providers: [
|
|
31
|
+
* provideValtechAuth({ apiUrl: environment.apiUrl }),
|
|
32
|
+
* provideValtechFeedback({
|
|
33
|
+
* apiUrl: environment.apiUrl,
|
|
34
|
+
* appId: 'my-app-name',
|
|
35
|
+
* }),
|
|
36
|
+
* ],
|
|
37
|
+
* });
|
|
38
|
+
* ```
|
|
39
|
+
*/
|
|
40
|
+
export function provideValtechFeedback(config) {
|
|
41
|
+
const mergedConfig = {
|
|
42
|
+
...DEFAULT_FEEDBACK_CONFIG,
|
|
43
|
+
...config,
|
|
44
|
+
};
|
|
45
|
+
return makeEnvironmentProviders([
|
|
46
|
+
{ provide: VALTECH_FEEDBACK_CONFIG, useValue: mergedConfig },
|
|
47
|
+
]);
|
|
48
|
+
}
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zZXJ2aWNlcy9mZWVkYmFjay9jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUVMLGNBQWMsRUFDZCx3QkFBd0IsR0FDekIsTUFBTSxlQUFlLENBQUM7QUFHdkI7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSx1QkFBdUIsR0FBRyxJQUFJLGNBQWMsQ0FDdkQsdUJBQXVCLENBQ3hCLENBQUM7QUFFRjs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLHVCQUF1QixHQUFtQztJQUNyRSxjQUFjLEVBQUUsY0FBYztJQUM5QixjQUFjLEVBQUUsQ0FBQztJQUNqQixXQUFXLEVBQUUsRUFBRSxHQUFHLElBQUksR0FBRyxJQUFJLEVBQUUsT0FBTztJQUN0QyxnQkFBZ0IsRUFBRSxDQUFDLFNBQVMsRUFBRSxTQUFTLEVBQUUsaUJBQWlCLENBQUM7SUFDM0QsV0FBVyxFQUFFLFVBQVU7Q0FDeEIsQ0FBQztBQUVGOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVCRztBQUNILE1BQU0sVUFBVSxzQkFBc0IsQ0FDcEMsTUFBNkI7SUFFN0IsTUFBTSxZQUFZLEdBQTBCO1FBQzFDLEdBQUcsdUJBQXVCO1FBQzFCLEdBQUcsTUFBTTtLQUNWLENBQUM7SUFFRixPQUFPLHdCQUF3QixDQUFDO1FBQzlCLEVBQUUsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFFBQVEsRUFBRSxZQUFZLEVBQUU7S0FDN0QsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEVudmlyb25tZW50UHJvdmlkZXJzLFxuICBJbmplY3Rpb25Ub2tlbixcbiAgbWFrZUVudmlyb25tZW50UHJvdmlkZXJzLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFZhbHRlY2hGZWVkYmFja0NvbmZpZyB9IGZyb20gJy4vdHlwZXMnO1xuXG4vKipcbiAqIFRva2VuIGRlIGlueWVjY2nDs24gcGFyYSBsYSBjb25maWd1cmFjacOzbiBkZSBGZWVkYmFjay5cbiAqL1xuZXhwb3J0IGNvbnN0IFZBTFRFQ0hfRkVFREJBQ0tfQ09ORklHID0gbmV3IEluamVjdGlvblRva2VuPFZhbHRlY2hGZWVkYmFja0NvbmZpZz4oXG4gICdWYWx0ZWNoRmVlZGJhY2tDb25maWcnXG4pO1xuXG4vKipcbiAqIENvbmZpZ3VyYWNpw7NuIHBvciBkZWZlY3RvLlxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9GRUVEQkFDS19DT05GSUc6IFBhcnRpYWw8VmFsdGVjaEZlZWRiYWNrQ29uZmlnPiA9IHtcbiAgZmVlZGJhY2tQcmVmaXg6ICcvdjEvZmVlZGJhY2snLFxuICBtYXhBdHRhY2htZW50czogNSxcbiAgbWF4RmlsZVNpemU6IDEwICogMTAyNCAqIDEwMjQsIC8vIDEwTUJcbiAgYWxsb3dlZEZpbGVUeXBlczogWydpbWFnZS8qJywgJ3ZpZGVvLyonLCAnYXBwbGljYXRpb24vcGRmJ10sXG4gIHN0b3JhZ2VQYXRoOiAnZmVlZGJhY2snLFxufTtcblxuLyoqXG4gKiBQcm92ZWUgZWwgc2VydmljaW8gZGUgZmVlZGJhY2sgYSBsYSBhcGxpY2FjacOzbiBBbmd1bGFyLlxuICpcbiAqIEBwYXJhbSBjb25maWcgLSBDb25maWd1cmFjacOzbiBkZSBmZWVkYmFja1xuICogQHJldHVybnMgRW52aXJvbm1lbnRQcm92aWRlcnMgcGFyYSB1c2FyIGVuIGJvb3RzdHJhcEFwcGxpY2F0aW9uXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIG1haW4udHNcbiAqIGltcG9ydCB7IGJvb3RzdHJhcEFwcGxpY2F0aW9uIH0gZnJvbSAnQGFuZ3VsYXIvcGxhdGZvcm0tYnJvd3Nlcic7XG4gKiBpbXBvcnQgeyBwcm92aWRlVmFsdGVjaEZlZWRiYWNrIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqIGltcG9ydCB7IGVudmlyb25tZW50IH0gZnJvbSAnLi9lbnZpcm9ubWVudHMvZW52aXJvbm1lbnQnO1xuICpcbiAqIGJvb3RzdHJhcEFwcGxpY2F0aW9uKEFwcENvbXBvbmVudCwge1xuICogICBwcm92aWRlcnM6IFtcbiAqICAgICBwcm92aWRlVmFsdGVjaEF1dGgoeyBhcGlVcmw6IGVudmlyb25tZW50LmFwaVVybCB9KSxcbiAqICAgICBwcm92aWRlVmFsdGVjaEZlZWRiYWNrKHtcbiAqICAgICAgIGFwaVVybDogZW52aXJvbm1lbnQuYXBpVXJsLFxuICogICAgICAgYXBwSWQ6ICdteS1hcHAtbmFtZScsXG4gKiAgICAgfSksXG4gKiAgIF0sXG4gKiB9KTtcbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZVZhbHRlY2hGZWVkYmFjayhcbiAgY29uZmlnOiBWYWx0ZWNoRmVlZGJhY2tDb25maWdcbik6IEVudmlyb25tZW50UHJvdmlkZXJzIHtcbiAgY29uc3QgbWVyZ2VkQ29uZmlnOiBWYWx0ZWNoRmVlZGJhY2tDb25maWcgPSB7XG4gICAgLi4uREVGQVVMVF9GRUVEQkFDS19DT05GSUcsXG4gICAgLi4uY29uZmlnLFxuICB9O1xuXG4gIHJldHVybiBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMoW1xuICAgIHsgcHJvdmlkZTogVkFMVEVDSF9GRUVEQkFDS19DT05GSUcsIHVzZVZhbHVlOiBtZXJnZWRDb25maWcgfSxcbiAgXSk7XG59XG4iXX0=
|
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
|
+
import { HttpClient } from '@angular/common/http';
|
|
3
|
+
import { firstValueFrom } from 'rxjs';
|
|
4
|
+
import { VALTECH_FEEDBACK_CONFIG } from './config';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
/**
|
|
7
|
+
* Servicio para gestionar feedback de usuarios.
|
|
8
|
+
*
|
|
9
|
+
* @example
|
|
10
|
+
* ```typescript
|
|
11
|
+
* @Component({...})
|
|
12
|
+
* export class MyComponent {
|
|
13
|
+
* private feedbackService = inject(FeedbackService);
|
|
14
|
+
*
|
|
15
|
+
* async submitFeedback() {
|
|
16
|
+
* const response = await this.feedbackService.createAsync(
|
|
17
|
+
* 'feedback',
|
|
18
|
+
* 'Mi comentario',
|
|
19
|
+
* 'Descripción detallada...'
|
|
20
|
+
* );
|
|
21
|
+
* console.log('Feedback enviado:', response.feedbackId);
|
|
22
|
+
* }
|
|
23
|
+
* }
|
|
24
|
+
* ```
|
|
25
|
+
*/
|
|
26
|
+
export class FeedbackService {
|
|
27
|
+
constructor() {
|
|
28
|
+
this.config = inject(VALTECH_FEEDBACK_CONFIG);
|
|
29
|
+
this.http = inject(HttpClient);
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* URL base para endpoints de feedback.
|
|
33
|
+
*/
|
|
34
|
+
get baseUrl() {
|
|
35
|
+
return `${this.config.apiUrl}${this.config.feedbackPrefix}`;
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Captura el contexto del dispositivo automáticamente.
|
|
39
|
+
*/
|
|
40
|
+
captureDeviceContext() {
|
|
41
|
+
const ua = navigator.userAgent;
|
|
42
|
+
return {
|
|
43
|
+
browser: this.detectBrowser(ua),
|
|
44
|
+
os: this.detectOS(ua),
|
|
45
|
+
viewport: `${window.innerWidth}x${window.innerHeight}`,
|
|
46
|
+
language: navigator.language,
|
|
47
|
+
userAgent: ua,
|
|
48
|
+
pageUrl: window.location.href,
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Crea un nuevo feedback.
|
|
53
|
+
*
|
|
54
|
+
* @param type - Tipo de feedback
|
|
55
|
+
* @param title - Título del feedback
|
|
56
|
+
* @param description - Descripción detallada
|
|
57
|
+
* @param attachments - URLs de archivos adjuntos (opcional)
|
|
58
|
+
* @param contentRef - Referencia a contenido específico (opcional)
|
|
59
|
+
* @returns Observable con la respuesta
|
|
60
|
+
*/
|
|
61
|
+
create(type, title, description, attachments = [], contentRef) {
|
|
62
|
+
const request = {
|
|
63
|
+
type,
|
|
64
|
+
title,
|
|
65
|
+
description,
|
|
66
|
+
attachments,
|
|
67
|
+
contentRef,
|
|
68
|
+
deviceContext: this.captureDeviceContext(),
|
|
69
|
+
appId: this.config.appId,
|
|
70
|
+
};
|
|
71
|
+
return this.http.post(this.baseUrl, request);
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Crea un nuevo feedback (versión async/await).
|
|
75
|
+
*/
|
|
76
|
+
async createAsync(type, title, description, attachments = [], contentRef) {
|
|
77
|
+
return firstValueFrom(this.create(type, title, description, attachments, contentRef));
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Obtiene un feedback por ID (solo el propietario).
|
|
81
|
+
*
|
|
82
|
+
* @param feedbackId - ID del feedback
|
|
83
|
+
* @returns Observable con la respuesta
|
|
84
|
+
*/
|
|
85
|
+
getById(feedbackId) {
|
|
86
|
+
return this.http.get(`${this.baseUrl}/${feedbackId}`);
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Obtiene un feedback por ID (versión async/await).
|
|
90
|
+
*/
|
|
91
|
+
async getByIdAsync(feedbackId) {
|
|
92
|
+
return firstValueFrom(this.getById(feedbackId));
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Valida si un archivo cumple con las restricciones.
|
|
96
|
+
*/
|
|
97
|
+
validateFile(file) {
|
|
98
|
+
// Verificar tamaño
|
|
99
|
+
if (file.size > this.config.maxFileSize) {
|
|
100
|
+
const maxSizeMB = Math.round(this.config.maxFileSize / (1024 * 1024));
|
|
101
|
+
return {
|
|
102
|
+
valid: false,
|
|
103
|
+
error: `El archivo excede el tamaño máximo de ${maxSizeMB}MB`,
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
// Verificar tipo
|
|
107
|
+
const allowedTypes = this.config.allowedFileTypes || [];
|
|
108
|
+
const isAllowed = allowedTypes.some((pattern) => {
|
|
109
|
+
if (pattern.endsWith('/*')) {
|
|
110
|
+
const baseType = pattern.replace('/*', '');
|
|
111
|
+
return file.type.startsWith(baseType);
|
|
112
|
+
}
|
|
113
|
+
return file.type === pattern;
|
|
114
|
+
});
|
|
115
|
+
if (!isAllowed) {
|
|
116
|
+
return {
|
|
117
|
+
valid: false,
|
|
118
|
+
error: 'Tipo de archivo no permitido',
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
return { valid: true };
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Obtiene la configuración actual del servicio.
|
|
125
|
+
*/
|
|
126
|
+
getConfig() {
|
|
127
|
+
return this.config;
|
|
128
|
+
}
|
|
129
|
+
// =========================================================================
|
|
130
|
+
// Helpers privados para detección de browser/OS
|
|
131
|
+
// =========================================================================
|
|
132
|
+
detectBrowser(ua) {
|
|
133
|
+
if (ua.includes('Edg/'))
|
|
134
|
+
return 'Edge';
|
|
135
|
+
if (ua.includes('Chrome/'))
|
|
136
|
+
return 'Chrome';
|
|
137
|
+
if (ua.includes('Firefox/'))
|
|
138
|
+
return 'Firefox';
|
|
139
|
+
if (ua.includes('Safari/') && !ua.includes('Chrome'))
|
|
140
|
+
return 'Safari';
|
|
141
|
+
if (ua.includes('Opera') || ua.includes('OPR/'))
|
|
142
|
+
return 'Opera';
|
|
143
|
+
return 'Unknown';
|
|
144
|
+
}
|
|
145
|
+
detectOS(ua) {
|
|
146
|
+
if (ua.includes('Windows NT 10'))
|
|
147
|
+
return 'Windows 10';
|
|
148
|
+
if (ua.includes('Windows NT 11'))
|
|
149
|
+
return 'Windows 11';
|
|
150
|
+
if (ua.includes('Windows'))
|
|
151
|
+
return 'Windows';
|
|
152
|
+
if (ua.includes('Mac OS X')) {
|
|
153
|
+
const match = ua.match(/Mac OS X (\d+[._]\d+)/);
|
|
154
|
+
if (match) {
|
|
155
|
+
return `macOS ${match[1].replace('_', '.')}`;
|
|
156
|
+
}
|
|
157
|
+
return 'macOS';
|
|
158
|
+
}
|
|
159
|
+
if (ua.includes('Android'))
|
|
160
|
+
return 'Android';
|
|
161
|
+
if (ua.includes('iPhone') || ua.includes('iPad'))
|
|
162
|
+
return 'iOS';
|
|
163
|
+
if (ua.includes('Linux'))
|
|
164
|
+
return 'Linux';
|
|
165
|
+
return 'Unknown';
|
|
166
|
+
}
|
|
167
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
168
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackService, providedIn: 'root' }); }
|
|
169
|
+
}
|
|
170
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackService, decorators: [{
|
|
171
|
+
type: Injectable,
|
|
172
|
+
args: [{ providedIn: 'root' }]
|
|
173
|
+
}] });
|
|
174
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"feedback.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/feedback/feedback.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAClD,OAAO,EAAc,cAAc,EAAE,MAAM,MAAM,CAAC;AAClD,OAAO,EAAE,uBAAuB,EAAE,MAAM,UAAU,CAAC;;AAUnD;;;;;;;;;;;;;;;;;;;GAmBG;AAEH,MAAM,OAAO,eAAe;IAD5B;QAEU,WAAM,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;QACzC,SAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;KA2JnC;IAzJC;;OAEG;IACH,IAAY,OAAO;QACjB,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;IAC9D,CAAC;IAED;;OAEG;IACH,oBAAoB;QAClB,MAAM,EAAE,GAAG,SAAS,CAAC,SAAS,CAAC;QAC/B,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC;YAC/B,EAAE,EAAE,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;YACrB,QAAQ,EAAE,GAAG,MAAM,CAAC,UAAU,IAAI,MAAM,CAAC,WAAW,EAAE;YACtD,QAAQ,EAAE,SAAS,CAAC,QAAQ;YAC5B,SAAS,EAAE,EAAE;YACb,OAAO,EAAE,MAAM,CAAC,QAAQ,CAAC,IAAI;SAC9B,CAAC;IACJ,CAAC;IAED;;;;;;;;;OASG;IACH,MAAM,CACJ,IAAkB,EAClB,KAAa,EACb,WAAmB,EACnB,cAAwB,EAAE,EAC1B,UAAuB;QAEvB,MAAM,OAAO,GAA0B;YACrC,IAAI;YACJ,KAAK;YACL,WAAW;YACX,WAAW;YACX,UAAU;YACV,aAAa,EAAE,IAAI,CAAC,oBAAoB,EAAE;YAC1C,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK;SACzB,CAAC;QAEF,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAyB,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IACvE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CACf,IAAkB,EAClB,KAAa,EACb,WAAmB,EACnB,cAAwB,EAAE,EAC1B,UAAuB;QAEvB,OAAO,cAAc,CACnB,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,KAAK,EAAE,WAAW,EAAE,WAAW,EAAE,UAAU,CAAC,CAC/D,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACH,OAAO,CAAC,UAAkB;QACxB,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAsB,GAAG,IAAI,CAAC,OAAO,IAAI,UAAU,EAAE,CAAC,CAAC;IAC7E,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,YAAY,CAAC,UAAkB;QACnC,OAAO,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC;IAClD,CAAC;IAED;;OAEG;IACH,YAAY,CAAC,IAAU;QACrB,mBAAmB;QACnB,IAAI,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,WAAY,EAAE,CAAC;YACzC,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,WAAY,GAAG,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC;YACvE,OAAO;gBACL,KAAK,EAAE,KAAK;gBACZ,KAAK,EAAE,yCAAyC,SAAS,IAAI;aAC9D,CAAC;QACJ,CAAC;QAED,iBAAiB;QACjB,MAAM,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,gBAAgB,IAAI,EAAE,CAAC;QACxD,MAAM,SAAS,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE;YAC9C,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC3B,MAAM,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;gBAC3C,OAAO,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;YACxC,CAAC;YACD,OAAO,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC;QAC/B,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO;gBACL,KAAK,EAAE,KAAK;gBACZ,KAAK,EAAE,8BAA8B;aACtC,CAAC;QACJ,CAAC;QAED,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;IACzB,CAAC;IAED;;OAEG;IACH,SAAS;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,4EAA4E;IAC5E,gDAAgD;IAChD,4EAA4E;IAEpE,aAAa,CAAC,EAAU;QAC9B,IAAI,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;YAAE,OAAO,MAAM,CAAC;QACvC,IAAI,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC;YAAE,OAAO,QAAQ,CAAC;QAC5C,IAAI,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC;YAAE,OAAO,SAAS,CAAC;QAC9C,IAAI,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAAE,OAAO,QAAQ,CAAC;QACtE,IAAI,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;YAAE,OAAO,OAAO,CAAC;QAChE,OAAO,SAAS,CAAC;IACnB,CAAC;IAEO,QAAQ,CAAC,EAAU;QACzB,IAAI,EAAE,CAAC,QAAQ,CAAC,eAAe,CAAC;YAAE,OAAO,YAAY,CAAC;QACtD,IAAI,EAAE,CAAC,QAAQ,CAAC,eAAe,CAAC;YAAE,OAAO,YAAY,CAAC;QACtD,IAAI,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC;YAAE,OAAO,SAAS,CAAC;QAC7C,IAAI,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;YAC5B,MAAM,KAAK,GAAG,EAAE,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;YAChD,IAAI,KAAK,EAAE,CAAC;gBACV,OAAO,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC;YAC/C,CAAC;YACD,OAAO,OAAO,CAAC;QACjB,CAAC;QACD,IAAI,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC;YAAE,OAAO,SAAS,CAAC;QAC7C,IAAI,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;YAAE,OAAO,KAAK,CAAC;QAC/D,IAAI,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC;YAAE,OAAO,OAAO,CAAC;QACzC,OAAO,SAAS,CAAC;IACnB,CAAC;+GA5JU,eAAe;mHAAf,eAAe,cADF,MAAM;;4FACnB,eAAe;kBAD3B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { Observable, firstValueFrom } from 'rxjs';\nimport { VALTECH_FEEDBACK_CONFIG } from './config';\nimport {\n  CreateFeedbackRequest,\n  CreateFeedbackResponse,\n  GetFeedbackResponse,\n  DeviceContext,\n  FeedbackType,\n  ContentRef,\n} from './types';\n\n/**\n * Servicio para gestionar feedback de usuarios.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class MyComponent {\n *   private feedbackService = inject(FeedbackService);\n *\n *   async submitFeedback() {\n *     const response = await this.feedbackService.createAsync(\n *       'feedback',\n *       'Mi comentario',\n *       'Descripción detallada...'\n *     );\n *     console.log('Feedback enviado:', response.feedbackId);\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FeedbackService {\n  private config = inject(VALTECH_FEEDBACK_CONFIG);\n  private http = inject(HttpClient);\n\n  /**\n   * URL base para endpoints de feedback.\n   */\n  private get baseUrl(): string {\n    return `${this.config.apiUrl}${this.config.feedbackPrefix}`;\n  }\n\n  /**\n   * Captura el contexto del dispositivo automáticamente.\n   */\n  captureDeviceContext(): DeviceContext {\n    const ua = navigator.userAgent;\n    return {\n      browser: this.detectBrowser(ua),\n      os: this.detectOS(ua),\n      viewport: `${window.innerWidth}x${window.innerHeight}`,\n      language: navigator.language,\n      userAgent: ua,\n      pageUrl: window.location.href,\n    };\n  }\n\n  /**\n   * Crea un nuevo feedback.\n   *\n   * @param type - Tipo de feedback\n   * @param title - Título del feedback\n   * @param description - Descripción detallada\n   * @param attachments - URLs de archivos adjuntos (opcional)\n   * @param contentRef - Referencia a contenido específico (opcional)\n   * @returns Observable con la respuesta\n   */\n  create(\n    type: FeedbackType,\n    title: string,\n    description: string,\n    attachments: string[] = [],\n    contentRef?: ContentRef\n  ): Observable<CreateFeedbackResponse> {\n    const request: CreateFeedbackRequest = {\n      type,\n      title,\n      description,\n      attachments,\n      contentRef,\n      deviceContext: this.captureDeviceContext(),\n      appId: this.config.appId,\n    };\n\n    return this.http.post<CreateFeedbackResponse>(this.baseUrl, request);\n  }\n\n  /**\n   * Crea un nuevo feedback (versión async/await).\n   */\n  async createAsync(\n    type: FeedbackType,\n    title: string,\n    description: string,\n    attachments: string[] = [],\n    contentRef?: ContentRef\n  ): Promise<CreateFeedbackResponse> {\n    return firstValueFrom(\n      this.create(type, title, description, attachments, contentRef)\n    );\n  }\n\n  /**\n   * Obtiene un feedback por ID (solo el propietario).\n   *\n   * @param feedbackId - ID del feedback\n   * @returns Observable con la respuesta\n   */\n  getById(feedbackId: string): Observable<GetFeedbackResponse> {\n    return this.http.get<GetFeedbackResponse>(`${this.baseUrl}/${feedbackId}`);\n  }\n\n  /**\n   * Obtiene un feedback por ID (versión async/await).\n   */\n  async getByIdAsync(feedbackId: string): Promise<GetFeedbackResponse> {\n    return firstValueFrom(this.getById(feedbackId));\n  }\n\n  /**\n   * Valida si un archivo cumple con las restricciones.\n   */\n  validateFile(file: File): { valid: boolean; error?: string } {\n    // Verificar tamaño\n    if (file.size > this.config.maxFileSize!) {\n      const maxSizeMB = Math.round(this.config.maxFileSize! / (1024 * 1024));\n      return {\n        valid: false,\n        error: `El archivo excede el tamaño máximo de ${maxSizeMB}MB`,\n      };\n    }\n\n    // Verificar tipo\n    const allowedTypes = this.config.allowedFileTypes || [];\n    const isAllowed = allowedTypes.some((pattern) => {\n      if (pattern.endsWith('/*')) {\n        const baseType = pattern.replace('/*', '');\n        return file.type.startsWith(baseType);\n      }\n      return file.type === pattern;\n    });\n\n    if (!isAllowed) {\n      return {\n        valid: false,\n        error: 'Tipo de archivo no permitido',\n      };\n    }\n\n    return { valid: true };\n  }\n\n  /**\n   * Obtiene la configuración actual del servicio.\n   */\n  getConfig(): Readonly<typeof this.config> {\n    return this.config;\n  }\n\n  // =========================================================================\n  // Helpers privados para detección de browser/OS\n  // =========================================================================\n\n  private detectBrowser(ua: string): string {\n    if (ua.includes('Edg/')) return 'Edge';\n    if (ua.includes('Chrome/')) return 'Chrome';\n    if (ua.includes('Firefox/')) return 'Firefox';\n    if (ua.includes('Safari/') && !ua.includes('Chrome')) return 'Safari';\n    if (ua.includes('Opera') || ua.includes('OPR/')) return 'Opera';\n    return 'Unknown';\n  }\n\n  private detectOS(ua: string): string {\n    if (ua.includes('Windows NT 10')) return 'Windows 10';\n    if (ua.includes('Windows NT 11')) return 'Windows 11';\n    if (ua.includes('Windows')) return 'Windows';\n    if (ua.includes('Mac OS X')) {\n      const match = ua.match(/Mac OS X (\\d+[._]\\d+)/);\n      if (match) {\n        return `macOS ${match[1].replace('_', '.')}`;\n      }\n      return 'macOS';\n    }\n    if (ua.includes('Android')) return 'Android';\n    if (ua.includes('iPhone') || ua.includes('iPad')) return 'iOS';\n    if (ua.includes('Linux')) return 'Linux';\n    return 'Unknown';\n  }\n}\n"]}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Valtech Feedback Service
|
|
3
|
+
*
|
|
4
|
+
* Servicio para gestionar feedback de usuarios a nivel de plataforma.
|
|
5
|
+
*
|
|
6
|
+
* @example
|
|
7
|
+
* ```typescript
|
|
8
|
+
* // main.ts - Configuración
|
|
9
|
+
* import { provideValtechFeedback } from 'valtech-components';
|
|
10
|
+
*
|
|
11
|
+
* bootstrapApplication(AppComponent, {
|
|
12
|
+
* providers: [
|
|
13
|
+
* provideValtechAuth({ apiUrl: environment.apiUrl }),
|
|
14
|
+
* provideValtechFeedback({
|
|
15
|
+
* apiUrl: environment.apiUrl,
|
|
16
|
+
* appId: 'my-app-name',
|
|
17
|
+
* }),
|
|
18
|
+
* ],
|
|
19
|
+
* });
|
|
20
|
+
*
|
|
21
|
+
* // component.ts - Uso
|
|
22
|
+
* import { FeedbackService } from 'valtech-components';
|
|
23
|
+
*
|
|
24
|
+
* @Component({...})
|
|
25
|
+
* export class MyComponent {
|
|
26
|
+
* private feedbackService = inject(FeedbackService);
|
|
27
|
+
*
|
|
28
|
+
* async submitFeedback() {
|
|
29
|
+
* const response = await this.feedbackService.createAsync(
|
|
30
|
+
* 'feedback',
|
|
31
|
+
* 'Título',
|
|
32
|
+
* 'Descripción...'
|
|
33
|
+
* );
|
|
34
|
+
* }
|
|
35
|
+
* }
|
|
36
|
+
* ```
|
|
37
|
+
*/
|
|
38
|
+
// Configuration
|
|
39
|
+
export { VALTECH_FEEDBACK_CONFIG, provideValtechFeedback, DEFAULT_FEEDBACK_CONFIG } from './config';
|
|
40
|
+
// Service
|
|
41
|
+
export { FeedbackService } from './feedback.service';
|
|
42
|
+
// Types
|
|
43
|
+
export { DEFAULT_FEEDBACK_TYPE_OPTIONS, } from './types';
|
|
44
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZlZWRiYWNrL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FvQ0c7QUFFSCxnQkFBZ0I7QUFDaEIsT0FBTyxFQUFFLHVCQUF1QixFQUFFLHNCQUFzQixFQUFFLHVCQUF1QixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBRXBHLFVBQVU7QUFDVixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFckQsUUFBUTtBQUNSLE9BQU8sRUFZTCw2QkFBNkIsR0FDOUIsTUFBTSxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFZhbHRlY2ggRmVlZGJhY2sgU2VydmljZVxuICpcbiAqIFNlcnZpY2lvIHBhcmEgZ2VzdGlvbmFyIGZlZWRiYWNrIGRlIHVzdWFyaW9zIGEgbml2ZWwgZGUgcGxhdGFmb3JtYS5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gbWFpbi50cyAtIENvbmZpZ3VyYWNpw7NuXG4gKiBpbXBvcnQgeyBwcm92aWRlVmFsdGVjaEZlZWRiYWNrIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQsIHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hBdXRoKHsgYXBpVXJsOiBlbnZpcm9ubWVudC5hcGlVcmwgfSksXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hGZWVkYmFjayh7XG4gKiAgICAgICBhcGlVcmw6IGVudmlyb25tZW50LmFwaVVybCxcbiAqICAgICAgIGFwcElkOiAnbXktYXBwLW5hbWUnLFxuICogICAgIH0pLFxuICogICBdLFxuICogfSk7XG4gKlxuICogLy8gY29tcG9uZW50LnRzIC0gVXNvXG4gKiBpbXBvcnQgeyBGZWVkYmFja1NlcnZpY2UgfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICpcbiAqIEBDb21wb25lbnQoey4uLn0pXG4gKiBleHBvcnQgY2xhc3MgTXlDb21wb25lbnQge1xuICogICBwcml2YXRlIGZlZWRiYWNrU2VydmljZSA9IGluamVjdChGZWVkYmFja1NlcnZpY2UpO1xuICpcbiAqICAgYXN5bmMgc3VibWl0RmVlZGJhY2soKSB7XG4gKiAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLmZlZWRiYWNrU2VydmljZS5jcmVhdGVBc3luYyhcbiAqICAgICAgICdmZWVkYmFjaycsXG4gKiAgICAgICAnVMOtdHVsbycsXG4gKiAgICAgICAnRGVzY3JpcGNpw7NuLi4uJ1xuICogICAgICk7XG4gKiAgIH1cbiAqIH1cbiAqIGBgYFxuICovXG5cbi8vIENvbmZpZ3VyYXRpb25cbmV4cG9ydCB7IFZBTFRFQ0hfRkVFREJBQ0tfQ09ORklHLCBwcm92aWRlVmFsdGVjaEZlZWRiYWNrLCBERUZBVUxUX0ZFRURCQUNLX0NPTkZJRyB9IGZyb20gJy4vY29uZmlnJztcblxuLy8gU2VydmljZVxuZXhwb3J0IHsgRmVlZGJhY2tTZXJ2aWNlIH0gZnJvbSAnLi9mZWVkYmFjay5zZXJ2aWNlJztcblxuLy8gVHlwZXNcbmV4cG9ydCB7XG4gIFZhbHRlY2hGZWVkYmFja0NvbmZpZyxcbiAgRmVlZGJhY2tUeXBlLFxuICBGZWVkYmFja1N0YXR1cyxcbiAgQ29udGVudFR5cGUsXG4gIENvbnRlbnRSZWYsXG4gIERldmljZUNvbnRleHQsXG4gIEZlZWRiYWNrLFxuICBDcmVhdGVGZWVkYmFja1JlcXVlc3QsXG4gIENyZWF0ZUZlZWRiYWNrUmVzcG9uc2UsXG4gIEdldEZlZWRiYWNrUmVzcG9uc2UsXG4gIEZlZWRiYWNrVHlwZU9wdGlvbixcbiAgREVGQVVMVF9GRUVEQkFDS19UWVBFX09QVElPTlMsXG59IGZyb20gJy4vdHlwZXMnO1xuIl19
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Configuración por defecto de tipos de feedback.
|
|
3
|
+
*/
|
|
4
|
+
export const DEFAULT_FEEDBACK_TYPE_OPTIONS = [
|
|
5
|
+
{
|
|
6
|
+
value: 'issue',
|
|
7
|
+
label: 'Reportar problema',
|
|
8
|
+
description: 'Algo no funciona correctamente',
|
|
9
|
+
icon: 'bug-outline',
|
|
10
|
+
},
|
|
11
|
+
{
|
|
12
|
+
value: 'poor-content',
|
|
13
|
+
label: 'Contenido incorrecto',
|
|
14
|
+
description: 'Información incorrecta o desactualizada',
|
|
15
|
+
icon: 'document-text-outline',
|
|
16
|
+
},
|
|
17
|
+
{
|
|
18
|
+
value: 'feedback',
|
|
19
|
+
label: 'Comentario general',
|
|
20
|
+
description: 'Tu opinión o experiencia',
|
|
21
|
+
icon: 'chatbubble-outline',
|
|
22
|
+
},
|
|
23
|
+
{
|
|
24
|
+
value: 'suggestion',
|
|
25
|
+
label: 'Sugerencia',
|
|
26
|
+
description: 'Propuesta de mejora o nueva funcionalidad',
|
|
27
|
+
icon: 'bulb-outline',
|
|
28
|
+
},
|
|
29
|
+
];
|
|
30
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZlZWRiYWNrL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQXlIQTs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLDZCQUE2QixHQUF5QjtJQUNqRTtRQUNFLEtBQUssRUFBRSxPQUFPO1FBQ2QsS0FBSyxFQUFFLG1CQUFtQjtRQUMxQixXQUFXLEVBQUUsZ0NBQWdDO1FBQzdDLElBQUksRUFBRSxhQUFhO0tBQ3BCO0lBQ0Q7UUFDRSxLQUFLLEVBQUUsY0FBYztRQUNyQixLQUFLLEVBQUUsc0JBQXNCO1FBQzdCLFdBQVcsRUFBRSx5Q0FBeUM7UUFDdEQsSUFBSSxFQUFFLHVCQUF1QjtLQUM5QjtJQUNEO1FBQ0UsS0FBSyxFQUFFLFVBQVU7UUFDakIsS0FBSyxFQUFFLG9CQUFvQjtRQUMzQixXQUFXLEVBQUUsMEJBQTBCO1FBQ3ZDLElBQUksRUFBRSxvQkFBb0I7S0FDM0I7SUFDRDtRQUNFLEtBQUssRUFBRSxZQUFZO1FBQ25CLEtBQUssRUFBRSxZQUFZO1FBQ25CLFdBQVcsRUFBRSwyQ0FBMkM7UUFDeEQsSUFBSSxFQUFFLGNBQWM7S0FDckI7Q0FDRixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb25maWd1cmFjacOzbiBkZWwgc2VydmljaW8gZGUgRmVlZGJhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgVmFsdGVjaEZlZWRiYWNrQ29uZmlnIHtcbiAgLyoqIFVSTCBiYXNlIGRlIGxhIEFQSSAqL1xuICBhcGlVcmw6IHN0cmluZztcbiAgLyoqIElEIGRlIGxhIGFwbGljYWNpw7NuIChlajogJ215LXZhbHRlY2gtYXBwJykgKi9cbiAgYXBwSWQ6IHN0cmluZztcbiAgLyoqIFByZWZpam8gcGFyYSBlbmRwb2ludHMgKGRlZmF1bHQ6ICcvdjEvZmVlZGJhY2snKSAqL1xuICBmZWVkYmFja1ByZWZpeD86IHN0cmluZztcbiAgLyoqIE7Dum1lcm8gbcOheGltbyBkZSBhZGp1bnRvcyAoZGVmYXVsdDogNSkgKi9cbiAgbWF4QXR0YWNobWVudHM/OiBudW1iZXI7XG4gIC8qKiBUYW1hw7FvIG3DoXhpbW8gcG9yIGFyY2hpdm8gZW4gYnl0ZXMgKGRlZmF1bHQ6IDEwTUIpICovXG4gIG1heEZpbGVTaXplPzogbnVtYmVyO1xuICAvKiogVGlwb3MgZGUgYXJjaGl2byBwZXJtaXRpZG9zIChkZWZhdWx0OiBbJ2ltYWdlLyonLCAndmlkZW8vKicsICdhcHBsaWNhdGlvbi9wZGYnXSkgKi9cbiAgYWxsb3dlZEZpbGVUeXBlcz86IHN0cmluZ1tdO1xuICAvKiogUnV0YSBlbiBGaXJlYmFzZSBTdG9yYWdlIHBhcmEgYWRqdW50b3MgKGRlZmF1bHQ6ICdmZWVkYmFjaycpICovXG4gIHN0b3JhZ2VQYXRoPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIFRpcG9zIGRlIGZlZWRiYWNrIGRpc3BvbmlibGVzLlxuICovXG5leHBvcnQgdHlwZSBGZWVkYmFja1R5cGUgPSAnaXNzdWUnIHwgJ3Bvb3ItY29udGVudCcgfCAnZmVlZGJhY2snIHwgJ3N1Z2dlc3Rpb24nO1xuXG4vKipcbiAqIEVzdGFkbyBkZSB1biBmZWVkYmFjay5cbiAqL1xuZXhwb3J0IHR5cGUgRmVlZGJhY2tTdGF0dXMgPSAnbmV3JyB8ICdyZXZpZXdlZCcgfCAncmVzb2x2ZWQnO1xuXG4vKipcbiAqIFRpcG9zIGRlIGNvbnRlbmlkbyBwYXJhIHJlZmVyZW5jaWEuXG4gKi9cbmV4cG9ydCB0eXBlIENvbnRlbnRUeXBlID1cbiAgfCAnYXJ0aWNsZSdcbiAgfCAnZmFxJ1xuICB8ICduZXdzJ1xuICB8ICdwYWdlJ1xuICB8ICdwcm9kdWN0J1xuICB8ICdldmVudCdcbiAgfCAnb3RoZXInO1xuXG4vKipcbiAqIFJlZmVyZW5jaWEgYSBjb250ZW5pZG8gZXNwZWPDrWZpY28uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29udGVudFJlZiB7XG4gIGNvbnRlbnRJZDogc3RyaW5nO1xuICBjb250ZW50VHlwZTogQ29udGVudFR5cGU7XG59XG5cbi8qKlxuICogQ29udGV4dG8gZGVsIGRpc3Bvc2l0aXZvIGRlbCB1c3VhcmlvLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIERldmljZUNvbnRleHQge1xuICBicm93c2VyOiBzdHJpbmc7XG4gIG9zOiBzdHJpbmc7XG4gIHZpZXdwb3J0OiBzdHJpbmc7XG4gIGxhbmd1YWdlOiBzdHJpbmc7XG4gIHVzZXJBZ2VudDogc3RyaW5nO1xuICBwYWdlVXJsOiBzdHJpbmc7XG59XG5cbi8qKlxuICogRW50cmFkYSBkZSBmZWVkYmFjayBjb21wbGV0YS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBGZWVkYmFjayB7XG4gIGZlZWRiYWNrSWQ6IHN0cmluZztcbiAgYXBwSWQ6IHN0cmluZztcbiAgdXNlcklkOiBzdHJpbmc7XG4gIHR5cGU6IEZlZWRiYWNrVHlwZTtcbiAgdGl0bGU6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgYXR0YWNobWVudHM6IHN0cmluZ1tdO1xuICBjb250ZW50UmVmPzogQ29udGVudFJlZjtcbiAgZGV2aWNlQ29udGV4dDogRGV2aWNlQ29udGV4dDtcbiAgc3RhdHVzOiBGZWVkYmFja1N0YXR1cztcbiAgY3JlYXRlZEF0OiBzdHJpbmc7XG4gIHVwZGF0ZWRBdDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlcXVlc3QgcGFyYSBjcmVhciBmZWVkYmFjay5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDcmVhdGVGZWVkYmFja1JlcXVlc3Qge1xuICB0eXBlOiBGZWVkYmFja1R5cGU7XG4gIHRpdGxlOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIGF0dGFjaG1lbnRzPzogc3RyaW5nW107XG4gIGNvbnRlbnRSZWY/OiBDb250ZW50UmVmO1xuICBkZXZpY2VDb250ZXh0OiBEZXZpY2VDb250ZXh0O1xuICBhcHBJZDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlc3BvbnNlIGFsIGNyZWFyIGZlZWRiYWNrLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIENyZWF0ZUZlZWRiYWNrUmVzcG9uc2Uge1xuICBvcGVyYXRpb25JZDogc3RyaW5nO1xuICBmZWVkYmFja0lkOiBzdHJpbmc7XG4gIHN0YXR1czogRmVlZGJhY2tTdGF0dXM7XG4gIGNyZWF0ZWRBdDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlc3BvbnNlIGFsIG9idGVuZXIgZmVlZGJhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgR2V0RmVlZGJhY2tSZXNwb25zZSB7XG4gIG9wZXJhdGlvbklkOiBzdHJpbmc7XG4gIGZlZWRiYWNrOiBGZWVkYmFjaztcbn1cblxuLyoqXG4gKiBPcGNpb25lcyBkZSB0aXBvIGRlIGZlZWRiYWNrIHBhcmEgVUkuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRmVlZGJhY2tUeXBlT3B0aW9uIHtcbiAgdmFsdWU6IEZlZWRiYWNrVHlwZTtcbiAgbGFiZWw6IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGljb24/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogQ29uZmlndXJhY2nDs24gcG9yIGRlZmVjdG8gZGUgdGlwb3MgZGUgZmVlZGJhY2suXG4gKi9cbmV4cG9ydCBjb25zdCBERUZBVUxUX0ZFRURCQUNLX1RZUEVfT1BUSU9OUzogRmVlZGJhY2tUeXBlT3B0aW9uW10gPSBbXG4gIHtcbiAgICB2YWx1ZTogJ2lzc3VlJyxcbiAgICBsYWJlbDogJ1JlcG9ydGFyIHByb2JsZW1hJyxcbiAgICBkZXNjcmlwdGlvbjogJ0FsZ28gbm8gZnVuY2lvbmEgY29ycmVjdGFtZW50ZScsXG4gICAgaWNvbjogJ2J1Zy1vdXRsaW5lJyxcbiAgfSxcbiAge1xuICAgIHZhbHVlOiAncG9vci1jb250ZW50JyxcbiAgICBsYWJlbDogJ0NvbnRlbmlkbyBpbmNvcnJlY3RvJyxcbiAgICBkZXNjcmlwdGlvbjogJ0luZm9ybWFjacOzbiBpbmNvcnJlY3RhIG8gZGVzYWN0dWFsaXphZGEnLFxuICAgIGljb246ICdkb2N1bWVudC10ZXh0LW91dGxpbmUnLFxuICB9LFxuICB7XG4gICAgdmFsdWU6ICdmZWVkYmFjaycsXG4gICAgbGFiZWw6ICdDb21lbnRhcmlvIGdlbmVyYWwnLFxuICAgIGRlc2NyaXB0aW9uOiAnVHUgb3BpbmnDs24gbyBleHBlcmllbmNpYScsXG4gICAgaWNvbjogJ2NoYXRidWJibGUtb3V0bGluZScsXG4gIH0sXG4gIHtcbiAgICB2YWx1ZTogJ3N1Z2dlc3Rpb24nLFxuICAgIGxhYmVsOiAnU3VnZXJlbmNpYScsXG4gICAgZGVzY3JpcHRpb246ICdQcm9wdWVzdGEgZGUgbWVqb3JhIG8gbnVldmEgZnVuY2lvbmFsaWRhZCcsXG4gICAgaWNvbjogJ2J1bGItb3V0bGluZScsXG4gIH0sXG5dO1xuIl19
|
|
@@ -31,6 +31,8 @@
|
|
|
31
31
|
export * from './types';
|
|
32
32
|
// Configuración
|
|
33
33
|
export { VALTECH_FIREBASE_CONFIG, hasEmulators, provideValtechFirebase } from './config';
|
|
34
|
+
// Configuración compartida del monorepo
|
|
35
|
+
export { APP_IDS, FIREBASE_PROJECTS, SHARED_EMULATOR_CONFIG, collections, createFirebaseConfig, isEmulatorMode, storagePaths, } from './shared-config';
|
|
34
36
|
// Servicios
|
|
35
37
|
export { FirebaseService } from './firebase.service';
|
|
36
38
|
// Firestore
|
|
@@ -43,4 +45,4 @@ export { buildPath, extractPathParams, getCollectionPath, getDocumentId, isColle
|
|
|
43
45
|
export { StorageService } from './storage.service';
|
|
44
46
|
// Messaging (FCM)
|
|
45
47
|
export { MessagingService } from './messaging.service';
|
|
46
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
48
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZpcmViYXNlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBRUgsUUFBUTtBQUNSLGNBQWMsU0FBUyxDQUFDO0FBRXhCLGdCQUFnQjtBQUNoQixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsWUFBWSxFQUFFLHNCQUFzQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBRXpGLHdDQUF3QztBQUN4QyxPQUFPLEVBQ0wsT0FBTyxFQUNQLGlCQUFpQixFQUNqQixzQkFBc0IsRUFDdEIsV0FBVyxFQUNYLG9CQUFvQixFQUNwQixjQUFjLEVBQ2QsWUFBWSxHQUdiLE1BQU0saUJBQWlCLENBQUM7QUFFekIsWUFBWTtBQUNaLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUVyRCxZQUFZO0FBQ1osT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDdkQsT0FBTyxFQUFxQixtQkFBbUIsRUFBb0IsTUFBTSx3QkFBd0IsQ0FBQztBQUVsRyxhQUFhO0FBQ2IsT0FBTyxFQUFFLFlBQVksRUFBRSxLQUFLLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUM1RCxPQUFPLEVBQ0wsU0FBUyxFQUNULGlCQUFpQixFQUNqQixpQkFBaUIsRUFDakIsYUFBYSxFQUNiLGdCQUFnQixFQUNoQixjQUFjLEVBQ2QsV0FBVyxFQUNYLFFBQVEsR0FDVCxNQUFNLHNCQUFzQixDQUFDO0FBRTlCLFVBQVU7QUFDVixPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sbUJBQW1CLENBQUM7QUFFbkQsa0JBQWtCO0FBQ2xCLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBGaXJlYmFzZSBTZXJ2aWNlc1xuICpcbiAqIFNlcnZpY2lvcyByZXV0aWxpemFibGVzIHBhcmEgaW50ZWdyYWNpw7NuIGNvbiBGaXJlYmFzZS5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gRW4gbWFpbi50c1xuICogaW1wb3J0IHsgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZSB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKlxuICogYm9vdHN0cmFwQXBwbGljYXRpb24oQXBwQ29tcG9uZW50LCB7XG4gKiAgIHByb3ZpZGVyczogW1xuICogICAgIHByb3ZpZGVWYWx0ZWNoRmlyZWJhc2Uoe1xuICogICAgICAgZmlyZWJhc2U6IGVudmlyb25tZW50LmZpcmViYXNlLFxuICogICAgICAgcGVyc2lzdGVuY2U6IHRydWUsXG4gKiAgICAgfSksXG4gKiAgIF0sXG4gKiB9KTtcbiAqXG4gKiAvLyBFbiBjb21wb25lbnRlc1xuICogaW1wb3J0IHsgRmlyZWJhc2VTZXJ2aWNlLCBGaXJlc3RvcmVTZXJ2aWNlIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBAQ29tcG9uZW50KHsuLi59KVxuICogZXhwb3J0IGNsYXNzIE15Q29tcG9uZW50IHtcbiAqICAgcHJpdmF0ZSBmaXJlYmFzZSA9IGluamVjdChGaXJlYmFzZVNlcnZpY2UpO1xuICogICBwcml2YXRlIGZpcmVzdG9yZSA9IGluamVjdChGaXJlc3RvcmVTZXJ2aWNlKTtcbiAqIH1cbiAqIGBgYFxuICovXG5cbi8vIFRpcG9zXG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcblxuLy8gQ29uZmlndXJhY2nDs25cbmV4cG9ydCB7IFZBTFRFQ0hfRklSRUJBU0VfQ09ORklHLCBoYXNFbXVsYXRvcnMsIHByb3ZpZGVWYWx0ZWNoRmlyZWJhc2UgfSBmcm9tICcuL2NvbmZpZyc7XG5cbi8vIENvbmZpZ3VyYWNpw7NuIGNvbXBhcnRpZGEgZGVsIG1vbm9yZXBvXG5leHBvcnQge1xuICBBUFBfSURTLFxuICBGSVJFQkFTRV9QUk9KRUNUUyxcbiAgU0hBUkVEX0VNVUxBVE9SX0NPTkZJRyxcbiAgY29sbGVjdGlvbnMsXG4gIGNyZWF0ZUZpcmViYXNlQ29uZmlnLFxuICBpc0VtdWxhdG9yTW9kZSxcbiAgc3RvcmFnZVBhdGhzLFxuICB0eXBlIEFwcElkLFxuICB0eXBlIENyZWF0ZUZpcmViYXNlQ29uZmlnT3B0aW9ucyxcbn0gZnJvbSAnLi9zaGFyZWQtY29uZmlnJztcblxuLy8gU2VydmljaW9zXG5leHBvcnQgeyBGaXJlYmFzZVNlcnZpY2UgfSBmcm9tICcuL2ZpcmViYXNlLnNlcnZpY2UnO1xuXG4vLyBGaXJlc3RvcmVcbmV4cG9ydCB7IEZpcmVzdG9yZVNlcnZpY2UgfSBmcm9tICcuL2ZpcmVzdG9yZS5zZXJ2aWNlJztcbmV4cG9ydCB7IENvbGxlY3Rpb25PcHRpb25zLCBGaXJlc3RvcmVDb2xsZWN0aW9uLCBTdWJDb2xsZWN0aW9uUmVmIH0gZnJvbSAnLi9maXJlc3RvcmUtY29sbGVjdGlvbic7XG5cbi8vIFV0aWxpZGFkZXNcbmV4cG9ydCB7IFF1ZXJ5QnVpbGRlciwgcXVlcnkgfSBmcm9tICcuL3V0aWxzL3F1ZXJ5LWJ1aWxkZXInO1xuZXhwb3J0IHtcbiAgYnVpbGRQYXRoLFxuICBleHRyYWN0UGF0aFBhcmFtcyxcbiAgZ2V0Q29sbGVjdGlvblBhdGgsXG4gIGdldERvY3VtZW50SWQsXG4gIGlzQ29sbGVjdGlvblBhdGgsXG4gIGlzRG9jdW1lbnRQYXRoLFxuICBpc1ZhbGlkUGF0aCxcbiAgam9pblBhdGgsXG59IGZyb20gJy4vdXRpbHMvcGF0aC1idWlsZGVyJztcblxuLy8gU3RvcmFnZVxuZXhwb3J0IHsgU3RvcmFnZVNlcnZpY2UgfSBmcm9tICcuL3N0b3JhZ2Uuc2VydmljZSc7XG5cbi8vIE1lc3NhZ2luZyAoRkNNKVxuZXhwb3J0IHsgTWVzc2FnaW5nU2VydmljZSB9IGZyb20gJy4vbWVzc2FnaW5nLnNlcnZpY2UnO1xuIl19
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Firebase Shared Configuration
|
|
3
|
+
*
|
|
4
|
+
* Configuración base de Firebase compartida entre todas las apps del monorepo.
|
|
5
|
+
* Los secrets (apiKey, appId) se inyectan en build time via environment.
|
|
6
|
+
*/
|
|
7
|
+
export const APP_IDS = {
|
|
8
|
+
SHOWCASE: 'showcase',
|
|
9
|
+
ADMIN_PORTAL: 'admin-portal',
|
|
10
|
+
APP: 'app',
|
|
11
|
+
};
|
|
12
|
+
// ============================================================================
|
|
13
|
+
// FIREBASE PROJECT IDS
|
|
14
|
+
// ============================================================================
|
|
15
|
+
/**
|
|
16
|
+
* IDs de los proyectos Firebase por ambiente.
|
|
17
|
+
* Deben coincidir con los proyectos creados en Firebase Console.
|
|
18
|
+
*/
|
|
19
|
+
export const FIREBASE_PROJECTS = {
|
|
20
|
+
development: 'myvaltech-dev',
|
|
21
|
+
production: 'myvaltech-prod',
|
|
22
|
+
};
|
|
23
|
+
// ============================================================================
|
|
24
|
+
// EMULATOR CONFIGURATION (shared across all apps)
|
|
25
|
+
// ============================================================================
|
|
26
|
+
/**
|
|
27
|
+
* Configuración de emuladores compartida.
|
|
28
|
+
* Todos los puertos deben coincidir con frontend/firebase/firebase.json
|
|
29
|
+
*/
|
|
30
|
+
export const SHARED_EMULATOR_CONFIG = {
|
|
31
|
+
firestore: { host: 'localhost', port: 9080 },
|
|
32
|
+
storage: { host: 'localhost', port: 9199 },
|
|
33
|
+
auth: { host: 'localhost', port: 9099 },
|
|
34
|
+
};
|
|
35
|
+
// ============================================================================
|
|
36
|
+
// STORAGE PATH BUILDERS
|
|
37
|
+
// ============================================================================
|
|
38
|
+
/**
|
|
39
|
+
* Genera paths de Storage con namespace por app.
|
|
40
|
+
*
|
|
41
|
+
* @example
|
|
42
|
+
* // Path específico de la app
|
|
43
|
+
* storagePaths.forApp('showcase', 'uploads', 'image.jpg')
|
|
44
|
+
* // => 'showcase/uploads/image.jpg'
|
|
45
|
+
*
|
|
46
|
+
* // Path compartido
|
|
47
|
+
* storagePaths.shared.profilePhoto('user123', 'avatar.jpg')
|
|
48
|
+
* // => 'profile-photos/user123/avatar.jpg'
|
|
49
|
+
*/
|
|
50
|
+
export const storagePaths = {
|
|
51
|
+
/** Carpeta específica de la app: {appId}/{...paths} */
|
|
52
|
+
forApp: (appId, ...paths) => [appId, ...paths].join('/'),
|
|
53
|
+
/** Carpetas compartidas (sin namespace) */
|
|
54
|
+
shared: {
|
|
55
|
+
/** Foto de perfil de usuario */
|
|
56
|
+
profilePhoto: (userId, fileName) => `profile-photos/${userId}/${fileName}`,
|
|
57
|
+
/** Archivos públicos accesibles sin autenticación */
|
|
58
|
+
public: (...paths) => `public/${paths.join('/')}`,
|
|
59
|
+
},
|
|
60
|
+
/** Carpetas de desarrollo (acceso libre en emuladores) */
|
|
61
|
+
demo: (...paths) => `demo-uploads/${paths.join('/')}`,
|
|
62
|
+
};
|
|
63
|
+
// ============================================================================
|
|
64
|
+
// FIRESTORE COLLECTION BUILDERS
|
|
65
|
+
// ============================================================================
|
|
66
|
+
/**
|
|
67
|
+
* Genera nombres de colecciones con namespace por app.
|
|
68
|
+
*
|
|
69
|
+
* @example
|
|
70
|
+
* // Colección específica de la app
|
|
71
|
+
* collections.forApp('showcase', 'items')
|
|
72
|
+
* // => 'showcase-items'
|
|
73
|
+
*
|
|
74
|
+
* // Colección compartida
|
|
75
|
+
* collections.shared.users
|
|
76
|
+
* // => 'users'
|
|
77
|
+
*/
|
|
78
|
+
export const collections = {
|
|
79
|
+
/** Colección específica de la app: {appId}-{collection} */
|
|
80
|
+
forApp: (appId, collectionName) => `${appId}-${collectionName}`,
|
|
81
|
+
/** Colecciones compartidas (sin namespace) */
|
|
82
|
+
shared: {
|
|
83
|
+
/** Usuarios del sistema */
|
|
84
|
+
users: 'users',
|
|
85
|
+
/** Perfiles públicos */
|
|
86
|
+
profiles: 'profiles',
|
|
87
|
+
/** Notificaciones de usuarios */
|
|
88
|
+
notifications: 'notifications',
|
|
89
|
+
},
|
|
90
|
+
/** Colecciones de desarrollo (acceso libre en emuladores) */
|
|
91
|
+
demo: (collectionName) => `demo-${collectionName}`,
|
|
92
|
+
};
|
|
93
|
+
/**
|
|
94
|
+
* Crea la configuración completa de Firebase desde variables de entorno.
|
|
95
|
+
* Usa esto en el environment.ts de cada app.
|
|
96
|
+
*
|
|
97
|
+
* @example
|
|
98
|
+
* // environment.ts
|
|
99
|
+
* export const environment = {
|
|
100
|
+
* firebase: createFirebaseConfig(
|
|
101
|
+
* {
|
|
102
|
+
* apiKey: 'AIza...',
|
|
103
|
+
* authDomain: 'myvaltech-dev.firebaseapp.com',
|
|
104
|
+
* projectId: 'myvaltech-dev',
|
|
105
|
+
* storageBucket: 'myvaltech-dev.appspot.com',
|
|
106
|
+
* messagingSenderId: '123456789',
|
|
107
|
+
* appId: '1:123456789:web:abc123',
|
|
108
|
+
* },
|
|
109
|
+
* { useEmulators: true, persistence: true }
|
|
110
|
+
* ),
|
|
111
|
+
* };
|
|
112
|
+
*/
|
|
113
|
+
export function createFirebaseConfig(envConfig, options = {}) {
|
|
114
|
+
const { useEmulators = false, persistence = true, enableMessaging = false, messagingVapidKey } = options;
|
|
115
|
+
return {
|
|
116
|
+
firebase: envConfig,
|
|
117
|
+
persistence,
|
|
118
|
+
enableMessaging,
|
|
119
|
+
messagingVapidKey,
|
|
120
|
+
emulator: useEmulators ? SHARED_EMULATOR_CONFIG : undefined,
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
// ============================================================================
|
|
124
|
+
// UTILITY: Check if running in emulator mode
|
|
125
|
+
// ============================================================================
|
|
126
|
+
/**
|
|
127
|
+
* Verifica si la configuración tiene emuladores habilitados
|
|
128
|
+
*/
|
|
129
|
+
export function isEmulatorMode(config) {
|
|
130
|
+
return config.emulator !== undefined;
|
|
131
|
+
}
|
|
132
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shared-config.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/shared-config.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAcH,MAAM,CAAC,MAAM,OAAO,GAAG;IACrB,QAAQ,EAAE,UAAmB;IAC7B,YAAY,EAAE,cAAuB;IACrC,GAAG,EAAE,KAAc;CACX,CAAC;AAEX,+EAA+E;AAC/E,uBAAuB;AACvB,+EAA+E;AAE/E;;;GAGG;AACH,MAAM,CAAC,MAAM,iBAAiB,GAAG;IAC/B,WAAW,EAAE,eAAe;IAC5B,UAAU,EAAE,gBAAgB;CACpB,CAAC;AAEX,+EAA+E;AAC/E,kDAAkD;AAClD,+EAA+E;AAE/E;;;GAGG;AACH,MAAM,CAAC,MAAM,sBAAsB,GAAmB;IACpD,SAAS,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE;IAC5C,OAAO,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE;IAC1C,IAAI,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE;CACxC,CAAC;AAEF,+EAA+E;AAC/E,wBAAwB;AACxB,+EAA+E;AAE/E;;;;;;;;;;;GAWG;AACH,MAAM,CAAC,MAAM,YAAY,GAAG;IAC1B,uDAAuD;IACvD,MAAM,EAAE,CAAC,KAAY,EAAE,GAAG,KAAe,EAAU,EAAE,CAAC,CAAC,KAAK,EAAE,GAAG,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;IAEjF,2CAA2C;IAC3C,MAAM,EAAE;QACN,gCAAgC;QAChC,YAAY,EAAE,CAAC,MAAc,EAAE,QAAgB,EAAU,EAAE,CACzD,kBAAkB,MAAM,IAAI,QAAQ,EAAE;QAExC,qDAAqD;QACrD,MAAM,EAAE,CAAC,GAAG,KAAe,EAAU,EAAE,CAAC,UAAU,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;KACpE;IAED,0DAA0D;IAC1D,IAAI,EAAE,CAAC,GAAG,KAAe,EAAU,EAAE,CAAC,gBAAgB,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;CACxE,CAAC;AAEF,+EAA+E;AAC/E,gCAAgC;AAChC,+EAA+E;AAE/E;;;;;;;;;;;GAWG;AACH,MAAM,CAAC,MAAM,WAAW,GAAG;IACzB,2DAA2D;IAC3D,MAAM,EAAE,CAAC,KAAY,EAAE,cAAsB,EAAU,EAAE,CAAC,GAAG,KAAK,IAAI,cAAc,EAAE;IAEtF,8CAA8C;IAC9C,MAAM,EAAE;QACN,2BAA2B;QAC3B,KAAK,EAAE,OAAO;QACd,wBAAwB;QACxB,QAAQ,EAAE,UAAU;QACpB,iCAAiC;QACjC,aAAa,EAAE,eAAe;KAC/B;IAED,6DAA6D;IAC7D,IAAI,EAAE,CAAC,cAAsB,EAAU,EAAE,CAAC,QAAQ,cAAc,EAAE;CACnE,CAAC;AAoBF;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,UAAU,oBAAoB,CAClC,SAAyB,EACzB,UAAuC,EAAE;IAEzC,MAAM,EAAE,YAAY,GAAG,KAAK,EAAE,WAAW,GAAG,IAAI,EAAE,eAAe,GAAG,KAAK,EAAE,iBAAiB,EAAE,GAC5F,OAAO,CAAC;IAEV,OAAO;QACL,QAAQ,EAAE,SAAS;QACnB,WAAW;QACX,eAAe;QACf,iBAAiB;QACjB,QAAQ,EAAE,YAAY,CAAC,CAAC,CAAC,sBAAsB,CAAC,CAAC,CAAC,SAAS;KAC5D,CAAC;AACJ,CAAC;AAED,+EAA+E;AAC/E,6CAA6C;AAC7C,+EAA+E;AAE/E;;GAEG;AACH,MAAM,UAAU,cAAc,CAAC,MAA6B;IAC1D,OAAO,MAAM,CAAC,QAAQ,KAAK,SAAS,CAAC;AACvC,CAAC","sourcesContent":["/**\n * Firebase Shared Configuration\n *\n * Configuración base de Firebase compartida entre todas las apps del monorepo.\n * Los secrets (apiKey, appId) se inyectan en build time via environment.\n */\n\nimport { EmulatorConfig, FirebaseConfig, ValtechFirebaseConfig } from './types';\n\n// ============================================================================\n// APP IDS - Identificadores únicos por aplicación\n// ============================================================================\n\n/**\n * Identificadores de las apps del monorepo.\n * Usados para namespacing de colecciones Firestore y paths de Storage.\n */\nexport type AppId = 'showcase' | 'admin-portal' | 'app';\n\nexport const APP_IDS = {\n  SHOWCASE: 'showcase' as AppId,\n  ADMIN_PORTAL: 'admin-portal' as AppId,\n  APP: 'app' as AppId,\n} as const;\n\n// ============================================================================\n// FIREBASE PROJECT IDS\n// ============================================================================\n\n/**\n * IDs de los proyectos Firebase por ambiente.\n * Deben coincidir con los proyectos creados en Firebase Console.\n */\nexport const FIREBASE_PROJECTS = {\n  development: 'myvaltech-dev',\n  production: 'myvaltech-prod',\n} as const;\n\n// ============================================================================\n// EMULATOR CONFIGURATION (shared across all apps)\n// ============================================================================\n\n/**\n * Configuración de emuladores compartida.\n * Todos los puertos deben coincidir con frontend/firebase/firebase.json\n */\nexport const SHARED_EMULATOR_CONFIG: EmulatorConfig = {\n  firestore: { host: 'localhost', port: 9080 },\n  storage: { host: 'localhost', port: 9199 },\n  auth: { host: 'localhost', port: 9099 },\n};\n\n// ============================================================================\n// STORAGE PATH BUILDERS\n// ============================================================================\n\n/**\n * Genera paths de Storage con namespace por app.\n *\n * @example\n * // Path específico de la app\n * storagePaths.forApp('showcase', 'uploads', 'image.jpg')\n * // => 'showcase/uploads/image.jpg'\n *\n * // Path compartido\n * storagePaths.shared.profilePhoto('user123', 'avatar.jpg')\n * // => 'profile-photos/user123/avatar.jpg'\n */\nexport const storagePaths = {\n  /** Carpeta específica de la app: {appId}/{...paths} */\n  forApp: (appId: AppId, ...paths: string[]): string => [appId, ...paths].join('/'),\n\n  /** Carpetas compartidas (sin namespace) */\n  shared: {\n    /** Foto de perfil de usuario */\n    profilePhoto: (userId: string, fileName: string): string =>\n      `profile-photos/${userId}/${fileName}`,\n\n    /** Archivos públicos accesibles sin autenticación */\n    public: (...paths: string[]): string => `public/${paths.join('/')}`,\n  },\n\n  /** Carpetas de desarrollo (acceso libre en emuladores) */\n  demo: (...paths: string[]): string => `demo-uploads/${paths.join('/')}`,\n};\n\n// ============================================================================\n// FIRESTORE COLLECTION BUILDERS\n// ============================================================================\n\n/**\n * Genera nombres de colecciones con namespace por app.\n *\n * @example\n * // Colección específica de la app\n * collections.forApp('showcase', 'items')\n * // => 'showcase-items'\n *\n * // Colección compartida\n * collections.shared.users\n * // => 'users'\n */\nexport const collections = {\n  /** Colección específica de la app: {appId}-{collection} */\n  forApp: (appId: AppId, collectionName: string): string => `${appId}-${collectionName}`,\n\n  /** Colecciones compartidas (sin namespace) */\n  shared: {\n    /** Usuarios del sistema */\n    users: 'users',\n    /** Perfiles públicos */\n    profiles: 'profiles',\n    /** Notificaciones de usuarios */\n    notifications: 'notifications',\n  },\n\n  /** Colecciones de desarrollo (acceso libre en emuladores) */\n  demo: (collectionName: string): string => `demo-${collectionName}`,\n};\n\n// ============================================================================\n// HELPER: Create Firebase Config from Environment\n// ============================================================================\n\n/**\n * Opciones para crear la configuración de Firebase\n */\nexport interface CreateFirebaseConfigOptions {\n  /** Usar emuladores locales (para desarrollo) */\n  useEmulators?: boolean;\n  /** Habilitar persistencia offline de Firestore */\n  persistence?: boolean;\n  /** Habilitar Firebase Cloud Messaging */\n  enableMessaging?: boolean;\n  /** VAPID key para FCM (requerido si enableMessaging es true) */\n  messagingVapidKey?: string;\n}\n\n/**\n * Crea la configuración completa de Firebase desde variables de entorno.\n * Usa esto en el environment.ts de cada app.\n *\n * @example\n * // environment.ts\n * export const environment = {\n *   firebase: createFirebaseConfig(\n *     {\n *       apiKey: 'AIza...',\n *       authDomain: 'myvaltech-dev.firebaseapp.com',\n *       projectId: 'myvaltech-dev',\n *       storageBucket: 'myvaltech-dev.appspot.com',\n *       messagingSenderId: '123456789',\n *       appId: '1:123456789:web:abc123',\n *     },\n *     { useEmulators: true, persistence: true }\n *   ),\n * };\n */\nexport function createFirebaseConfig(\n  envConfig: FirebaseConfig,\n  options: CreateFirebaseConfigOptions = {}\n): ValtechFirebaseConfig {\n  const { useEmulators = false, persistence = true, enableMessaging = false, messagingVapidKey } =\n    options;\n\n  return {\n    firebase: envConfig,\n    persistence,\n    enableMessaging,\n    messagingVapidKey,\n    emulator: useEmulators ? SHARED_EMULATOR_CONFIG : undefined,\n  };\n}\n\n// ============================================================================\n// UTILITY: Check if running in emulator mode\n// ============================================================================\n\n/**\n * Verifica si la configuración tiene emuladores habilitados\n */\nexport function isEmulatorMode(config: ValtechFirebaseConfig): boolean {\n  return config.emulator !== undefined;\n}\n"]}
|