├── src ├── assets │ └── .gitkeep ├── app │ ├── app.component.scss │ ├── form-builder │ │ ├── form-builder.component.scss │ │ ├── form-builder.component.html │ │ ├── form-builder.component.spec.ts │ │ ├── form-builder.component.ts │ │ └── options.ts │ ├── form-renderer │ │ ├── form-renderer.component.scss │ │ ├── form-renderer.component.html │ │ ├── form-renderer.component.ts │ │ └── form-renderer.component.spec.ts │ ├── app-routing.module.ts │ ├── app.module.ts │ ├── app.component.spec.ts │ ├── app.component.ts │ └── app.component.html ├── favicon.ico ├── styles.scss ├── main.ts └── index.html ├── screenshots ├── formBuilder.png ├── formBuilderBlank.png ├── formRendererBlank.png ├── formRendererFilled.png └── formRendererReadOnly.png ├── .vscode ├── extensions.json ├── launch.json └── tasks.json ├── tsconfig.app.json ├── tsconfig.spec.json ├── .editorconfig ├── .gitignore ├── tsconfig.json ├── package.json ├── angular.json └── README.md /src/assets/.gitkeep: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /src/app/app.component.scss: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /src/app/form-builder/form-builder.component.scss: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /src/app/form-renderer/form-renderer.component.scss: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /src/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AniruddhaSadhukhan/dynamic-forms/HEAD/src/favicon.ico -------------------------------------------------------------------------------- /src/styles.scss: -------------------------------------------------------------------------------- 1 | /* You can add global styles to this file, and also import other style files */ 2 | -------------------------------------------------------------------------------- /screenshots/formBuilder.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AniruddhaSadhukhan/dynamic-forms/HEAD/screenshots/formBuilder.png -------------------------------------------------------------------------------- /screenshots/formBuilderBlank.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AniruddhaSadhukhan/dynamic-forms/HEAD/screenshots/formBuilderBlank.png -------------------------------------------------------------------------------- /screenshots/formRendererBlank.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AniruddhaSadhukhan/dynamic-forms/HEAD/screenshots/formRendererBlank.png -------------------------------------------------------------------------------- /screenshots/formRendererFilled.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AniruddhaSadhukhan/dynamic-forms/HEAD/screenshots/formRendererFilled.png -------------------------------------------------------------------------------- /screenshots/formRendererReadOnly.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AniruddhaSadhukhan/dynamic-forms/HEAD/screenshots/formRendererReadOnly.png -------------------------------------------------------------------------------- /.vscode/extensions.json: -------------------------------------------------------------------------------- 1 | { 2 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=827846 3 | "recommendations": ["angular.ng-template"] 4 | } 5 | -------------------------------------------------------------------------------- /src/app/form-renderer/form-renderer.component.html: -------------------------------------------------------------------------------- 1 |
2 | 8 |
9 | -------------------------------------------------------------------------------- /src/main.ts: -------------------------------------------------------------------------------- 1 | import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; 2 | 3 | import { AppModule } from './app/app.module'; 4 | 5 | 6 | platformBrowserDynamic().bootstrapModule(AppModule) 7 | .catch(err => console.error(err)); 8 | -------------------------------------------------------------------------------- /src/app/app-routing.module.ts: -------------------------------------------------------------------------------- 1 | import { NgModule } from '@angular/core'; 2 | import { RouterModule, Routes } from '@angular/router'; 3 | 4 | const routes: Routes = []; 5 | 6 | @NgModule({ 7 | imports: [RouterModule.forRoot(routes)], 8 | exports: [RouterModule] 9 | }) 10 | export class AppRoutingModule { } 11 | -------------------------------------------------------------------------------- /tsconfig.app.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "extends": "./tsconfig.json", 4 | "compilerOptions": { 5 | "outDir": "./out-tsc/app", 6 | "types": [] 7 | }, 8 | "files": [ 9 | "src/main.ts" 10 | ], 11 | "include": [ 12 | "src/**/*.d.ts" 13 | ] 14 | } 15 | -------------------------------------------------------------------------------- /tsconfig.spec.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "extends": "./tsconfig.json", 4 | "compilerOptions": { 5 | "outDir": "./out-tsc/spec", 6 | "types": [ 7 | "jasmine" 8 | ] 9 | }, 10 | "include": [ 11 | "src/**/*.spec.ts", 12 | "src/**/*.d.ts" 13 | ] 14 | } 15 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | # Editor configuration, see https://editorconfig.org 2 | root = true 3 | 4 | [*] 5 | charset = utf-8 6 | indent_style = space 7 | indent_size = 2 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | 11 | [*.ts] 12 | quote_type = single 13 | 14 | [*.md] 15 | max_line_length = off 16 | trim_trailing_whitespace = false 17 | -------------------------------------------------------------------------------- /src/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | DynamicForms 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/app/form-builder/form-builder.component.html: -------------------------------------------------------------------------------- 1 |
2 | 7 |
8 | 11 |
12 |
13 | -------------------------------------------------------------------------------- /src/app/form-renderer/form-renderer.component.ts: -------------------------------------------------------------------------------- 1 | import { Component, EventEmitter, Input, Output } from '@angular/core'; 2 | 3 | @Component({ 4 | selector: 'dynamic-form-renderer', 5 | templateUrl: './form-renderer.component.html', 6 | styleUrls: ['./form-renderer.component.scss'], 7 | }) 8 | export class FormRendererComponent { 9 | @Input() form; 10 | @Input() submission; 11 | @Input() readOnly; 12 | @Output() submit: EventEmitter = new EventEmitter(); 13 | 14 | onSubmit(event) { 15 | this.submit.emit(event); 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 3 | "version": "0.2.0", 4 | "configurations": [ 5 | { 6 | "name": "ng serve", 7 | "type": "chrome", 8 | "request": "launch", 9 | "preLaunchTask": "npm: start", 10 | "url": "http://localhost:4200/" 11 | }, 12 | { 13 | "name": "ng test", 14 | "type": "chrome", 15 | "request": "launch", 16 | "preLaunchTask": "npm: test", 17 | "url": "http://localhost:9876/debug.html" 18 | } 19 | ] 20 | } 21 | -------------------------------------------------------------------------------- /src/app/form-builder/form-builder.component.spec.ts: -------------------------------------------------------------------------------- 1 | import { ComponentFixture, TestBed } from '@angular/core/testing'; 2 | 3 | import { FormBuilderComponent } from './form-builder.component'; 4 | 5 | describe('FormBuilderComponent', () => { 6 | let component: FormBuilderComponent; 7 | let fixture: ComponentFixture; 8 | 9 | beforeEach(() => { 10 | TestBed.configureTestingModule({ 11 | declarations: [FormBuilderComponent] 12 | }); 13 | fixture = TestBed.createComponent(FormBuilderComponent); 14 | component = fixture.componentInstance; 15 | fixture.detectChanges(); 16 | }); 17 | 18 | it('should create', () => { 19 | expect(component).toBeTruthy(); 20 | }); 21 | }); 22 | -------------------------------------------------------------------------------- /src/app/form-renderer/form-renderer.component.spec.ts: -------------------------------------------------------------------------------- 1 | import { ComponentFixture, TestBed } from '@angular/core/testing'; 2 | 3 | import { FormRendererComponent } from './form-renderer.component'; 4 | 5 | describe('FormRendererComponent', () => { 6 | let component: FormRendererComponent; 7 | let fixture: ComponentFixture; 8 | 9 | beforeEach(() => { 10 | TestBed.configureTestingModule({ 11 | declarations: [FormRendererComponent] 12 | }); 13 | fixture = TestBed.createComponent(FormRendererComponent); 14 | component = fixture.componentInstance; 15 | fixture.detectChanges(); 16 | }); 17 | 18 | it('should create', () => { 19 | expect(component).toBeTruthy(); 20 | }); 21 | }); 22 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See http://help.github.com/ignore-files/ for more about ignoring files. 2 | 3 | # Compiled output 4 | /dist 5 | /tmp 6 | /out-tsc 7 | /bazel-out 8 | 9 | # Node 10 | /node_modules 11 | npm-debug.log 12 | yarn-error.log 13 | 14 | # IDEs and editors 15 | .idea/ 16 | .project 17 | .classpath 18 | .c9/ 19 | *.launch 20 | .settings/ 21 | *.sublime-workspace 22 | 23 | # Visual Studio Code 24 | .vscode/* 25 | !.vscode/settings.json 26 | !.vscode/tasks.json 27 | !.vscode/launch.json 28 | !.vscode/extensions.json 29 | .history/* 30 | 31 | # Miscellaneous 32 | /.angular/cache 33 | .sass-cache/ 34 | /connect.lock 35 | /coverage 36 | /libpeerconnection.log 37 | testem.log 38 | /typings 39 | 40 | # System files 41 | .DS_Store 42 | Thumbs.db 43 | -------------------------------------------------------------------------------- /src/app/app.module.ts: -------------------------------------------------------------------------------- 1 | import { NgModule } from '@angular/core'; 2 | import { BrowserModule } from '@angular/platform-browser'; 3 | 4 | import { AppRoutingModule } from './app-routing.module'; 5 | import { AppComponent } from './app.component'; 6 | import { FormBuilderComponent } from './form-builder/form-builder.component'; 7 | import { FormRendererComponent } from './form-renderer/form-renderer.component'; 8 | import { FormioModule } from '@formio/angular'; 9 | 10 | @NgModule({ 11 | declarations: [ 12 | AppComponent, 13 | FormBuilderComponent, 14 | FormRendererComponent 15 | ], 16 | imports: [ 17 | BrowserModule, 18 | AppRoutingModule, 19 | FormioModule 20 | ], 21 | providers: [], 22 | bootstrap: [AppComponent] 23 | }) 24 | export class AppModule { } 25 | -------------------------------------------------------------------------------- /src/app/form-builder/form-builder.component.ts: -------------------------------------------------------------------------------- 1 | import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; 2 | import { options } from './options'; 3 | 4 | @Component({ 5 | selector: 'dynamic-form-builder', 6 | templateUrl: './form-builder.component.html', 7 | styleUrls: ['./form-builder.component.scss'], 8 | }) 9 | export class FormBuilderComponent implements OnInit { 10 | @Input() initialForm: any; 11 | @Output() save: EventEmitter = new EventEmitter(); 12 | options: any; 13 | 14 | form: Object = { components: [] }; 15 | 16 | generatedForm: Object; 17 | 18 | constructor() { 19 | this.options = options; 20 | } 21 | 22 | ngOnInit(): void { 23 | if (this.initialForm?.components?.length) { 24 | this.form = this.initialForm; 25 | } 26 | this.generatedForm = this.initialForm; 27 | } 28 | 29 | onChange(event) { 30 | this.generatedForm = event.form; 31 | } 32 | 33 | saveForm() { 34 | this.save.emit(this.generatedForm); 35 | } 36 | } 37 | -------------------------------------------------------------------------------- /.vscode/tasks.json: -------------------------------------------------------------------------------- 1 | { 2 | // For more information, visit: https://go.microsoft.com/fwlink/?LinkId=733558 3 | "version": "2.0.0", 4 | "tasks": [ 5 | { 6 | "type": "npm", 7 | "script": "start", 8 | "isBackground": true, 9 | "problemMatcher": { 10 | "owner": "typescript", 11 | "pattern": "$tsc", 12 | "background": { 13 | "activeOnStart": true, 14 | "beginsPattern": { 15 | "regexp": "(.*?)" 16 | }, 17 | "endsPattern": { 18 | "regexp": "bundle generation complete" 19 | } 20 | } 21 | } 22 | }, 23 | { 24 | "type": "npm", 25 | "script": "test", 26 | "isBackground": true, 27 | "problemMatcher": { 28 | "owner": "typescript", 29 | "pattern": "$tsc", 30 | "background": { 31 | "activeOnStart": true, 32 | "beginsPattern": { 33 | "regexp": "(.*?)" 34 | }, 35 | "endsPattern": { 36 | "regexp": "bundle generation complete" 37 | } 38 | } 39 | } 40 | } 41 | ] 42 | } 43 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "compileOnSave": false, 4 | "compilerOptions": { 5 | "baseUrl": "./", 6 | "outDir": "./dist/out-tsc", 7 | "forceConsistentCasingInFileNames": true, 8 | "strict": true, 9 | "noImplicitOverride": true, 10 | "noPropertyAccessFromIndexSignature": true, 11 | "noImplicitReturns": true, 12 | "noFallthroughCasesInSwitch": true, 13 | "sourceMap": true, 14 | "declaration": false, 15 | "downlevelIteration": true, 16 | "experimentalDecorators": true, 17 | "moduleResolution": "node", 18 | "importHelpers": true, 19 | "useDefineForClassFields": false, 20 | "noImplicitAny": false, 21 | "strictPropertyInitialization": false, 22 | "strictNullChecks": false, 23 | "target": "ESNext", 24 | "module": "ESNext", 25 | "lib": ["ESNext", "dom"] 26 | }, 27 | "angularCompilerOptions": { 28 | "enableI18nLegacyMessageIdFormat": false, 29 | "strictInjectionParameters": true, 30 | "strictInputAccessModifiers": true, 31 | "strictTemplates": true 32 | } 33 | } 34 | -------------------------------------------------------------------------------- /src/app/app.component.spec.ts: -------------------------------------------------------------------------------- 1 | import { TestBed } from '@angular/core/testing'; 2 | import { RouterTestingModule } from '@angular/router/testing'; 3 | import { AppComponent } from './app.component'; 4 | 5 | describe('AppComponent', () => { 6 | beforeEach(() => TestBed.configureTestingModule({ 7 | imports: [RouterTestingModule], 8 | declarations: [AppComponent] 9 | })); 10 | 11 | it('should create the app', () => { 12 | const fixture = TestBed.createComponent(AppComponent); 13 | const app = fixture.componentInstance; 14 | expect(app).toBeTruthy(); 15 | }); 16 | 17 | it(`should have as title 'dynamic-forms'`, () => { 18 | const fixture = TestBed.createComponent(AppComponent); 19 | const app = fixture.componentInstance; 20 | expect(app.title).toEqual('dynamic-forms'); 21 | }); 22 | 23 | it('should render title', () => { 24 | const fixture = TestBed.createComponent(AppComponent); 25 | fixture.detectChanges(); 26 | const compiled = fixture.nativeElement as HTMLElement; 27 | expect(compiled.querySelector('.content span')?.textContent).toContain('dynamic-forms app is running!'); 28 | }); 29 | }); 30 | -------------------------------------------------------------------------------- /src/app/app.component.ts: -------------------------------------------------------------------------------- 1 | import { Component, OnInit } from '@angular/core'; 2 | 3 | @Component({ 4 | selector: 'app-root', 5 | templateUrl: './app.component.html', 6 | styleUrls: ['./app.component.scss'], 7 | }) 8 | export class AppComponent implements OnInit { 9 | pages = ['Form Builder', 'Form Renderer', 'Form Reader']; 10 | currentPage = 'Form Builder'; 11 | form: Object; 12 | submission: Object; 13 | initialForm: Object; 14 | currentSubmission = {}; 15 | 16 | changePage(page: string) { 17 | this.currentPage = page; 18 | } 19 | 20 | ngOnInit(): void { 21 | let jsonForm = localStorage.getItem('jsonForm'); 22 | let submissionStr = localStorage.getItem('jsonFormSubmission'); 23 | if (jsonForm) { 24 | this.form = JSON.parse(jsonForm); 25 | } 26 | if (submissionStr) { 27 | this.submission = JSON.parse(submissionStr); 28 | } 29 | this.initialForm = JSON.parse(JSON.stringify(this.form)); 30 | } 31 | 32 | onFormSave(event) { 33 | this.form = event; 34 | localStorage.setItem('jsonForm', JSON.stringify(event)); 35 | } 36 | 37 | onSubmission(event) { 38 | this.submission = event; 39 | this.currentSubmission = event; 40 | localStorage.setItem('jsonFormSubmission', JSON.stringify(event)); 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "dynamic-forms", 3 | "version": "0.0.0", 4 | "scripts": { 5 | "ng": "ng", 6 | "update": "npm update --legacy-peer-deps", 7 | "serve": "ng serve", 8 | "build": "ng build", 9 | "deploy": "surge --project ./dist/dynamic-forms --domain ani-dynamic-forms-demo.surge.sh", 10 | "test": "ng test" 11 | }, 12 | "private": true, 13 | "dependencies": { 14 | "@angular/animations": "^16.1.0", 15 | "@angular/common": "^16.1.0", 16 | "@angular/compiler": "^16.1.0", 17 | "@angular/core": "^16.1.0", 18 | "@angular/elements": "^16.2.2", 19 | "@angular/forms": "^16.1.0", 20 | "@angular/platform-browser": "^16.1.0", 21 | "@angular/platform-browser-dynamic": "^16.1.0", 22 | "@angular/router": "^16.1.0", 23 | "@formio/angular": "^6.0.0-rc.5", 24 | "@formio/js": "^5.0.0-rc.27", 25 | "font-awesome": "^4.7.0", 26 | "rxjs": "~7.8.0", 27 | "tslib": "^2.3.0", 28 | "zone.js": "~0.13.0" 29 | }, 30 | "devDependencies": { 31 | "@angular-devkit/build-angular": "^16.1.8", 32 | "@angular/cli": "~16.2.0", 33 | "@angular/compiler-cli": "^16.1.0", 34 | "@types/jasmine": "~4.3.0", 35 | "jasmine-core": "~4.6.0", 36 | "karma": "~6.4.0", 37 | "karma-chrome-launcher": "~3.2.0", 38 | "karma-coverage": "~2.2.0", 39 | "karma-jasmine": "~5.1.0", 40 | "karma-jasmine-html-reporter": "~2.1.0", 41 | "typescript": "~5.1.3" 42 | } 43 | } 44 | -------------------------------------------------------------------------------- /angular.json: -------------------------------------------------------------------------------- 1 | { 2 | "$schema": "./node_modules/@angular/cli/lib/config/schema.json", 3 | "version": 1, 4 | "newProjectRoot": "projects", 5 | "projects": { 6 | "dynamic-forms": { 7 | "projectType": "application", 8 | "schematics": { 9 | "@schematics/angular:component": { 10 | "style": "scss" 11 | } 12 | }, 13 | "root": "", 14 | "sourceRoot": "src", 15 | "prefix": "app", 16 | "architect": { 17 | "build": { 18 | "builder": "@angular-devkit/build-angular:browser", 19 | "options": { 20 | "outputPath": "dist/dynamic-forms", 21 | "index": "src/index.html", 22 | "main": "src/main.ts", 23 | "polyfills": ["zone.js"], 24 | "tsConfig": "tsconfig.app.json", 25 | "inlineStyleLanguage": "scss", 26 | "assets": ["src/favicon.ico", "src/assets"], 27 | "styles": [ 28 | "src/styles.scss", 29 | "font-awesome/css/font-awesome.min.css", 30 | "bootstrap/dist/css/bootstrap.min.css" 31 | ], 32 | "stylePreprocessorOptions": { 33 | "includePaths": ["./node_modules"] 34 | }, 35 | "scripts": ["node_modules/bootstrap/dist/js/bootstrap.min.js"] 36 | }, 37 | "configurations": { 38 | "production": { 39 | "budgets": [ 40 | { 41 | "type": "initial", 42 | "maximumWarning": "3mb", 43 | "maximumError": "5mb" 44 | }, 45 | { 46 | "type": "anyComponentStyle", 47 | "maximumWarning": "2kb", 48 | "maximumError": "4kb" 49 | } 50 | ], 51 | "outputHashing": "all" 52 | }, 53 | "development": { 54 | "buildOptimizer": false, 55 | "optimization": false, 56 | "vendorChunk": true, 57 | "extractLicenses": false, 58 | "sourceMap": true, 59 | "namedChunks": true 60 | } 61 | }, 62 | "defaultConfiguration": "production" 63 | }, 64 | "serve": { 65 | "builder": "@angular-devkit/build-angular:dev-server", 66 | "configurations": { 67 | "production": { 68 | "browserTarget": "dynamic-forms:build:production" 69 | }, 70 | "development": { 71 | "browserTarget": "dynamic-forms:build:development" 72 | } 73 | }, 74 | "defaultConfiguration": "development" 75 | }, 76 | "extract-i18n": { 77 | "builder": "@angular-devkit/build-angular:extract-i18n", 78 | "options": { 79 | "browserTarget": "dynamic-forms:build" 80 | } 81 | }, 82 | "test": { 83 | "builder": "@angular-devkit/build-angular:karma", 84 | "options": { 85 | "polyfills": ["zone.js", "zone.js/testing"], 86 | "tsConfig": "tsconfig.spec.json", 87 | "inlineStyleLanguage": "scss", 88 | "assets": ["src/favicon.ico", "src/assets"], 89 | "styles": ["src/styles.scss"], 90 | "scripts": [] 91 | } 92 | } 93 | } 94 | } 95 | } 96 | } 97 | -------------------------------------------------------------------------------- /src/app/app.component.html: -------------------------------------------------------------------------------- 1 | 39 | 40 | 41 |
42 | Create a form here and save the form 46 |
47 | 51 | 52 |
53 | The output of the above Form Builder on save is a JSON Schema that 56 | represents the dynamic form. 58 |
59 | 60 |
61 |

As JSON Schema

62 |
63 |

 67 |         {{form | json }}
 68 |       
69 |
70 |
71 |
72 | 73 | 74 |
75 | You can fill up the form created in the Form Builder. Make sure to add a 78 | Submit Button in the form to submit the result. 80 |
81 | 82 | 83 |
84 | The output of the above Form on submit is a JSON Schema that represents 87 | the answers given. 89 |
90 | 91 |
92 |

Submission JSON

93 |
94 |

 98 |         {{currentSubmission | json }}
 99 |       
100 |
101 |
102 |
103 | 104 | 105 |
106 | You can read previously submitted form 110 |
111 | 117 |
118 | -------------------------------------------------------------------------------- /src/app/form-builder/options.ts: -------------------------------------------------------------------------------- 1 | const commonEditFormOptions = [ 2 | { 3 | key: 'layout', 4 | components: [ 5 | { 6 | key: 'overlay', 7 | ignore: true, 8 | }, 9 | ], 10 | }, 11 | { 12 | key: 'api', 13 | label: 'Submission', 14 | components: [ 15 | { 16 | key: 'tags', 17 | ignore: true, 18 | }, 19 | { 20 | key: 'properties', 21 | ignore: true, 22 | }, 23 | ], 24 | }, 25 | ]; 26 | 27 | const commonDisplayEditFormOptions = [ 28 | { 29 | key: 'tableView', 30 | ignore: true, 31 | }, 32 | ]; 33 | 34 | const commonDataEditFormOptions = [ 35 | { 36 | key: 'persistent', 37 | ignore: true, 38 | }, 39 | { 40 | key: 'protected', 41 | ignore: true, 42 | }, 43 | { 44 | key: 'dbIndex', 45 | ignore: true, 46 | }, 47 | { 48 | key: 'encrypted', 49 | ignore: true, 50 | }, 51 | { 52 | key: 'calculateServer', 53 | ignore: true, 54 | }, 55 | ]; 56 | 57 | const commonValidationEditFormOptions = [ 58 | { 59 | key: 'unique', 60 | ignore: true, 61 | }, 62 | ]; 63 | 64 | const defaultEditFormOptions = [ 65 | ...commonEditFormOptions, 66 | { 67 | key: 'display', 68 | components: commonDisplayEditFormOptions, 69 | }, 70 | { 71 | key: 'data', 72 | components: commonDataEditFormOptions, 73 | }, 74 | { 75 | key: 'validation', 76 | components: commonValidationEditFormOptions, 77 | }, 78 | ]; 79 | 80 | export const options = { 81 | noDefaultSubmitButton: true, 82 | builder: { 83 | basic: { 84 | components: { 85 | password: false, 86 | submitButton: { 87 | title: 'Submit Button', 88 | key: 'submitButton', 89 | icon: 'floppy-o', 90 | weight: 100, 91 | schema: { 92 | type: 'button', 93 | key: 'submit', 94 | leftIcon: 'fa fa-floppy-o', 95 | disableOnInvalid: true, 96 | }, 97 | }, 98 | resetButton: { 99 | title: 'Reset Button', 100 | key: 'resetButton', 101 | icon: 'eraser', 102 | weight: 101, 103 | schema: { 104 | label: 'Reset', 105 | type: 'button', 106 | key: 'reset', 107 | leftIcon: 'fa fa-eraser', 108 | action: 'reset', 109 | theme: 'danger', 110 | }, 111 | }, 112 | }, 113 | }, 114 | advanced: { 115 | components: { 116 | address: false, 117 | }, 118 | }, 119 | premium: false, 120 | resource: false, 121 | }, 122 | editForm: { 123 | textfield: defaultEditFormOptions, 124 | textarea: [ 125 | ...commonEditFormOptions, 126 | { 127 | key: 'display', 128 | components: [ 129 | ...commonDisplayEditFormOptions, 130 | { 131 | key: 'isUploadEnabled', 132 | ignore: true, 133 | }, 134 | ], 135 | }, 136 | { 137 | key: 'data', 138 | components: commonDataEditFormOptions, 139 | }, 140 | { 141 | key: 'validation', 142 | components: commonValidationEditFormOptions, 143 | }, 144 | ], 145 | number: defaultEditFormOptions, 146 | checkbox: [ 147 | ...commonEditFormOptions, 148 | { 149 | key: 'display', 150 | components: [ 151 | ...commonDisplayEditFormOptions, 152 | { 153 | key: 'inputType', 154 | ignore: true, 155 | }, 156 | ], 157 | }, 158 | { 159 | key: 'data', 160 | components: commonDataEditFormOptions, 161 | }, 162 | { 163 | key: 'validation', 164 | components: commonValidationEditFormOptions, 165 | }, 166 | ], 167 | selectboxes: defaultEditFormOptions, 168 | select: defaultEditFormOptions, 169 | radio: defaultEditFormOptions, 170 | button: defaultEditFormOptions, 171 | email: [ 172 | ...commonEditFormOptions, 173 | { 174 | key: 'display', 175 | components: commonDisplayEditFormOptions, 176 | }, 177 | { 178 | key: 'data', 179 | components: commonDataEditFormOptions, 180 | }, 181 | { 182 | key: 'validation', 183 | components: [ 184 | ...commonValidationEditFormOptions, 185 | { 186 | key: 'kickbox', 187 | ignore: true, 188 | }, 189 | ], 190 | }, 191 | ], 192 | url: defaultEditFormOptions, 193 | phoneNumber: defaultEditFormOptions, 194 | tags: defaultEditFormOptions, 195 | datetime: defaultEditFormOptions, 196 | day: defaultEditFormOptions, 197 | time: defaultEditFormOptions, 198 | currency: defaultEditFormOptions, 199 | signature: defaultEditFormOptions, 200 | htmlelement: defaultEditFormOptions, 201 | content: defaultEditFormOptions, 202 | columns: defaultEditFormOptions, 203 | fieldset: defaultEditFormOptions, 204 | panel: defaultEditFormOptions, 205 | well: defaultEditFormOptions, 206 | hidden: defaultEditFormOptions, 207 | container: defaultEditFormOptions, 208 | datamap: defaultEditFormOptions, 209 | datagrid: defaultEditFormOptions, 210 | editgrid: defaultEditFormOptions, 211 | }, 212 | }; 213 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Dynamic Forms 2 | 3 | This is an Angular project consisting of two primary components: Form Builder and Form Renderer. These components enable users to create dynamic forms through a drag-and-drop interface, render the forms, submit them, and view the submissions. And all the forms and submission are stored in a JSON format. This project is built using the FORMIO library, focusing on the client-side aspects, and excludes server-side functionality. 4 | 5 | ### Demo : https://ani-dynamic-forms-demo.surge.sh/ 6 | 7 | ## Form Builder 8 | 9 | The Angular Dynamic Form Builder is a powerful tool that allows users to create forms in the UI using a drag-and-drop interface. This project provides a user-friendly way to design forms with various components and layout options and then export the form's configuration or structure in JSON format. 10 | 11 | #### Inputs & Outputs 12 | 13 | | Input/Output | Name | Description | 14 | | ------------ | ------ | --------------------------------------------------------------------------------------------------------------------- | 15 | | initialForm | Input | This is an optional input. If a JSON of a form is provided, that form will be loaded and then the user can edit that. | 16 | | save | Output | On clicking the Save Form button, this will emit the JSON of the form structure. | 17 | 18 | #### Screenshot 19 | 20 | ![](./screenshots/formBuilderBlank.png) 21 | 22 | ## Form Renderer 23 | 24 | The Angular Dynamic Form Renderer can then render the form in json format created previously by the form 25 | builder for the user to fill up. After submitting the form, a JSON will be created with the values of the submission by the Form Renderer. Then submission JSON can also be used to prefill the Form Renderer or show the submission in read only mode in the Form Renderer. 26 | 27 | #### Inputs & Outputs 28 | 29 | | Input/Output | Name | Description | 30 | | ------------ | ------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------ | 31 | | form | Input | This is the JSON of a form structure that will be rendered. The JSON will be created by the Form Builder previously. | 32 | | submission | Input | This is an optional input. If submission JSON is provided, that form will be prefilled by that and then the user can edit that if read-only mode is not set. | 33 | | readOnly | Input | This is an optional input. If this readOnly mode is true, that form will not allow the user to edit it. | 34 | | submit | Output | On clicking the Submit button, this will emit the JSON of the form submission. | 35 | 36 | #### Screenshot 37 | 38 | ![](./screenshots/formRendererBlank.png) 39 | 40 | ## Example 41 | 42 | 1. Creating Form using Form Builder 43 | ![](./screenshots/formBuilder.png) 44 | 2. The form will be saved in JSON format 45 | 46 | ```js 47 | { 48 | "components": [ 49 | { 50 | "label": "HTML", 51 | "tag": "h2", 52 | "attrs": [ 53 | { 54 | "attr": "", 55 | "value": "" 56 | } 57 | ], 58 | "content": "A Random Form", 59 | "refreshOnChange": false, 60 | "key": "html", 61 | "type": "htmlelement", 62 | "input": false, 63 | "tableView": false 64 | }, 65 | { 66 | "label": "Name", 67 | "columns": [ 68 | { 69 | "components": [ 70 | { 71 | "label": "First Name", 72 | "applyMaskOn": "change", 73 | "validate": { 74 | "required": true 75 | }, 76 | "key": "firstName", 77 | "type": "textfield", 78 | "input": true, 79 | "tableView": true 80 | } 81 | ], 82 | "width": 4, 83 | "offset": 0, 84 | "push": 0, 85 | "pull": 0, 86 | "size": "md", 87 | "currentWidth": 4 88 | }, 89 | { 90 | "components": [ 91 | { 92 | "label": "Middle Name", 93 | "applyMaskOn": "change", 94 | "key": "middleName", 95 | "type": "textfield", 96 | "input": true, 97 | "tableView": true 98 | } 99 | ], 100 | "width": 4, 101 | "offset": 0, 102 | "push": 0, 103 | "pull": 0, 104 | "size": "md", 105 | "currentWidth": 4 106 | }, 107 | { 108 | "components": [ 109 | { 110 | "label": "Last Name", 111 | "applyMaskOn": "change", 112 | "validate": { 113 | "required": true 114 | }, 115 | "key": "firstName1", 116 | "type": "textfield", 117 | "input": true, 118 | "tableView": true 119 | } 120 | ], 121 | "size": "md", 122 | "width": 4, 123 | "offset": 0, 124 | "push": 0, 125 | "pull": 0, 126 | "currentWidth": 4 127 | } 128 | ], 129 | "key": "name", 130 | "type": "columns", 131 | "input": false, 132 | "tableView": false 133 | }, 134 | { 135 | "label": "Tabs", 136 | "components": [ 137 | { 138 | "label": "Tab 1", 139 | "key": "tab1", 140 | "components": [ 141 | { 142 | "label": "Columns", 143 | "columns": [ 144 | { 145 | "components": [ 146 | { 147 | "label": "Date of Birth", 148 | "datePicker": { 149 | "disableWeekends": false, 150 | "disableWeekdays": false 151 | }, 152 | "enableMinDateInput": false, 153 | "enableMaxDateInput": false, 154 | "key": "dateOfBirth", 155 | "type": "datetime", 156 | "input": true, 157 | "tableView": false, 158 | "widget": { 159 | "type": "calendar", 160 | "displayInTimezone": "viewer", 161 | "locale": "en", 162 | "useLocaleSettings": false, 163 | "allowInput": true, 164 | "mode": "single", 165 | "enableTime": true, 166 | "noCalendar": false, 167 | "format": "yyyy-MM-dd hh:mm a", 168 | "hourIncrement": 1, 169 | "minuteIncrement": 1, 170 | "time_24hr": false, 171 | "minDate": null, 172 | "disableWeekends": false, 173 | "disableWeekdays": false, 174 | "maxDate": null 175 | } 176 | } 177 | ], 178 | "width": 6, 179 | "offset": 0, 180 | "push": 0, 181 | "pull": 0, 182 | "size": "md", 183 | "currentWidth": 6 184 | }, 185 | { 186 | "components": [ 187 | { 188 | "label": "Phone Number", 189 | "applyMaskOn": "change", 190 | "key": "phoneNumber", 191 | "type": "phoneNumber", 192 | "input": true, 193 | "tableView": true 194 | } 195 | ], 196 | "width": 6, 197 | "offset": 0, 198 | "push": 0, 199 | "pull": 0, 200 | "size": "md", 201 | "currentWidth": 6 202 | } 203 | ], 204 | "key": "columns", 205 | "type": "columns", 206 | "input": false, 207 | "tableView": false 208 | } 209 | ] 210 | }, 211 | { 212 | "label": "Tab 2", 213 | "key": "tab2", 214 | "components": [] 215 | } 216 | ], 217 | "key": "tabs", 218 | "type": "tabs", 219 | "input": false, 220 | "tableView": false 221 | }, 222 | { 223 | "label": "Survey", 224 | "tableView": false, 225 | "questions": [ 226 | { 227 | "label": "How was the product?", 228 | "value": "howWasTheProduct", 229 | "tooltip": "" 230 | }, 231 | { 232 | "label": "How was the delivery?", 233 | "value": "howWasTheDelivery", 234 | "tooltip": "" 235 | } 236 | ], 237 | "values": [ 238 | { 239 | "label": "Very Good", 240 | "value": "veryGood", 241 | "tooltip": "" 242 | }, 243 | { 244 | "label": "Good", 245 | "value": "good", 246 | "tooltip": "" 247 | }, 248 | { 249 | "label": "Medium", 250 | "value": "medium", 251 | "tooltip": "" 252 | }, 253 | { 254 | "label": "Bad", 255 | "value": "bad", 256 | "tooltip": "" 257 | }, 258 | { 259 | "label": "Very Bad", 260 | "value": "veryBad", 261 | "tooltip": "" 262 | } 263 | ], 264 | "key": "survey", 265 | "type": "survey", 266 | "input": true 267 | }, 268 | { 269 | "label": "Signature", 270 | "key": "signature", 271 | "type": "signature", 272 | "input": true, 273 | "tableView": false 274 | }, 275 | { 276 | "label": "Columns", 277 | "columns": [ 278 | { 279 | "components": [ 280 | { 281 | "label": "Reset", 282 | "action": "reset", 283 | "showValidations": false, 284 | "theme": "danger", 285 | "block": true, 286 | "leftIcon": "fa fa-eraser", 287 | "key": "reset", 288 | "type": "button", 289 | "input": true, 290 | "tableView": false 291 | } 292 | ], 293 | "width": 6, 294 | "offset": 0, 295 | "push": 0, 296 | "pull": 0, 297 | "size": "md", 298 | "currentWidth": 6 299 | }, 300 | { 301 | "components": [ 302 | { 303 | "label": "Submit", 304 | "showValidations": false, 305 | "block": true, 306 | "leftIcon": "fa fa-floppy-o", 307 | "disableOnInvalid": true, 308 | "key": "submit", 309 | "type": "button", 310 | "input": true, 311 | "tableView": false, 312 | "saveOnEnter": false 313 | } 314 | ], 315 | "width": 6, 316 | "offset": 0, 317 | "push": 0, 318 | "pull": 0, 319 | "size": "md", 320 | "currentWidth": 6 321 | } 322 | ], 323 | "autoAdjust": true, 324 | "key": "columns1", 325 | "type": "columns", 326 | "input": false, 327 | "tableView": false 328 | } 329 | ] 330 | } 331 | ``` 332 | 333 | 3. Viewing the form and filling it up 334 | ![](./screenshots/formRendererFilled.png) 335 | 4. After submit, it will give a submission JSON 336 | 337 | ```js 338 | { 339 | "data": { 340 | "firstName": "Abc", 341 | "middleName": "", 342 | "firstName1": "Xyz", 343 | "dateOfBirth": "2023-09-05T12:00:00+05:30", 344 | "phoneNumber": "", 345 | "survey": { 346 | "howWasTheProduct": "good", 347 | "howWasTheDelivery": "medium" 348 | }, 349 | "signature": "", 350 | "reset": false, 351 | "submit": true 352 | }, 353 | "metadata": { 354 | "timezone": "Asia/Kolkata", 355 | "offset": 330, 356 | "origin": "https://ani-dynamic-forms-demo.surge.sh", 357 | "referrer": "", 358 | "browserName": "Netscape", 359 | "userAgent": "Mozilla/5.0 (X11; Linux x86_64; rv:109.0) Gecko/20100101 Firefox/116.0", 360 | "pathName": "/", 361 | "onLine": true 362 | }, 363 | "state": "submitted", 364 | "_vnote": "" 365 | } 366 | ``` 367 | 368 | 5. You can use the submission JSON to pre-fill the form or see it in read only mode. 369 | ![](./screenshots/formRendererReadOnly.png) 370 | 371 | ## Development server 372 | 373 | Run `npm run serve` for a dev server. Navigate to `http://localhost:4200/`. The application will automatically reload if you change any of the source files. 374 | 375 | ## Build 376 | 377 | Run `npm run build` to build the project. The build artifacts will be stored in the `dist/` directory. 378 | --------------------------------------------------------------------------------