Search code examples
vue.jsvuejs2

Vue.js 2: Multiple `template` with `v-for` possible?


For a customer project, I need to use Vue.js version 2.

In a component, I have 2 iterations over some field definition JSONs that use <template> with v-for.

Whichever iteration of these I put first, will get ignored - the content never shows up. If I change their order, it will be the respective other one.

HEAVILY EDITED: now includes a reproducible example, hopefully

src/__tests__/ConcreteFormComponentFillingSlots.spec.js

// @vitest-environment jsdom

import { describe, expect, it } from 'vitest'
import { mount } from '@vue/test-utils'
import ConcreteFormComponentFillingSlots from '@/__tests__/ConcreteFormComponentFillingSlots.vue'

describe('My bug with multiple template v-for iterations', () => {
    it('renders all form fields', () => {
        const wrapper = mount(ConcreteFormComponentFillingSlots)

        const renderedHtml = wrapper.html()

        console.log(`\n\n### + ${renderedHtml}\n\n####`)
        expect(renderedHtml).includes('Search | Key: |searchResultSlot1|')
        expect(renderedHtml).includes('Search | Key: |searchResultSlot2|')
        expect(renderedHtml).includes('Upload | Key: |uploadResultSlot1|')
        expect(renderedHtml).includes('Upload | Key: |uploadResultSlot2|')
    })
})

src/__tests__/GenericFormComponentWithSlots.vue

<script>

export default {
    name: 'GenericFormComponentWithSlots',
    props: {
        id: String,
        formConfig: Object
    },
    data () {
        return {
            pageSelectionIndex: 0,
            sectionSelectionIndex: 0
        }
    },
    methods: {
        getFieldLabel (field) {
            return field.label
        }
    },
    computed: {
        getSelectedSection () {
            return this.formConfig.sections[this.sectionSelectionIndex]
        }
    }
}
</script>

<template>
    <div v-bind:id="`c_smart-form-body_some_id`"
         class="c_smart-form-body">
        <transition-group name="show" tag="div">
            <template v-for="(field) in getSelectedSection.fields">
                <div v-bind:key="`${field.id}_${pageSelectionIndex}_outerForm`"
                     class="c_smart-form-field">
                    <div v-if="field.type === 'Slot'">
                        <p v-html="'Slot goes here: field(' + field.id + ')'"/>
                        <slot v-bind:name="`field(${field.id})`"
                              v-bind:field="field">
                        </slot>
                    </div>
                    <div v-else>
                        <p v-html="'Unknown field type: ' + field.type + ' for field ' + field.id"/>
                    </div>
                </div>
            </template>
        </transition-group>
    </div>
</template>

<style lang="less">
</style>

src/__tests__/ConcreteFormComponentFillingSlots.vue

<script>
import GenericFormComponentWithSlots from '@/__tests__/GenericFormComponentWithSlots.vue'
import theUserForm from '@/__tests__/the-user-form.json'
import _ from 'lodash'

export default {
    name: 'ConcreteFormComponentFillingSlots',
    mixins: [
    ],
    components: {
        GenericFormComponentWithSlots
    },
    props: {
    },
    data () {
        return {
            customForm: _.cloneDeep(theUserForm),
            uploadFieldConfigs: {
                uploadResultSlot1: {
                    isUploading: false,
                    slotName: 'field(uploadResultSlot1)'
                },
                uploadResultSlot2: {
                    isUploading: false,
                    slotName: 'field(uploadResultSlot2)'
                }
            },
            searchFieldConfigs: {
                searchResultSlot1: {
                    slotName: 'field(searchResultSlot1)'
                },
                searchResultSlot2: {
                    slotName: 'field(searchResultSlot2)'
                }
            }
        }
    }
}
</script>

<template>
    <div class="custom-task-content-container">
        <GenericFormComponentWithSlots v-if="customForm" id="customForm" v-bind:form-config=this.customForm
        >
            <!-- Iterates over uploadFieldConfigs; fieldConfig will be the child object, key will be the key under which it is stored as part of uploadFieldConfigs -->
            <!-- '#' is a shorthand for v-slot: (referencing the slot name dynamically is easier with the shorthand) https://vuejs.org/guide/components/slots.html#dynamic-slot-names -->
            <!-- '=data' passes in the field data, and we pass name and field in EssentialSmartForm, cf. https://vuejs.org/api/built-in-directives.html#v-slot -->
            <template v-for="(fieldConfig, key) in uploadFieldConfigs" #[fieldConfig.slotName]="data">
                <p :key="key + '_upload_debug1'" v-html="'Upload | Key: |' + key + '| Data: |' + JSON.stringify(data) + '|'"></p>
            </template>
            <template v-for="(fieldConfig, key) in searchFieldConfigs" #[fieldConfig.slotName]="slotProps">
                <p :key="key + '_upload_debug2'" v-html="'Search | Key: |' + key + '| Data: |' + JSON.stringify(slotProps) + '|'"></p>
            </template>
        </GenericFormComponentWithSlots>
    </div>
</template>

<style lang="less" scoped>
</style>

src/__tests__/the-user-form.json

{
  "id": "myForm",
  "version": 1,
  "headline": "My form",
  "sections": [
    {
      "id": "BaseInfo",
      "title": "Some Hints",
      "allowMultiple": false,
      "fields": [
        {
          "id": "searchResultSlot1",
          "type": "Slot",
          "label": "Search result slot 1",
          "keyLabel": "labelName",
          "isReadOnly": false
        },
        {
          "id": "searchResultSlot2",
          "type": "Slot",
          "label": "Search result slot 2",
          "keyLabel": "labelName",
          "isReadOnly": false
        },
        {
          "id": "uploadResultSlot1",
          "type": "Slot",
          "label": "Upload Result Slot 1",
          "upload": true,
          "required": true
        },
        {
          "id": "uploadResultSlot2",
          "type": "Slot",
          "label": "Upload Result Slot 2",
          "upload": true,
          "required": true
        }
      ],
      "defaultValues": {
      },
      "pages": [{}]
    }
  ]
}

vite.config.js

import vue from '@vitejs/plugin-vue2'
import { fileURLToPath, URL } from "node:url";

export default {
    plugins: [ vue() ],
    resolve: {
        alias: {
            '@': fileURLToPath(new URL('./src', import.meta.url))
        }
    }
}

package.json

{
    "name": "My-App",
    "version": "1.0.0",
    "private": true,
    "scripts": {
        "serve": "vue-cli-service serve",
        "build": "vue-cli-service build",
        "lint": "vue-cli-service lint",
        "test:unit": "vitest --environment jsdom --root src/",
        "coverage": "vitest --coverage --environment jsdom --root src/"
    },
    "dependencies": {
        "isomorphic-fetch": "^3.0.0",
        "register-service-worker": "^1.7.2",
        "vue": "^2.7.16",
        "vue-dompurify-html": "4.1",
        "vue-router": "^3.5.3"
    },
    "devDependencies": {
        "@vitejs/plugin-vue2": "^2.3.1",
        "@vitest/coverage-c8": "^0.29.2",
        "@vue/cli-plugin-eslint": "^5.0.8",
        "@vue/cli-plugin-pwa": "^5.0.4",
        "@vue/cli-plugin-router": "^5.0.4",
        "@vue/cli-service": "^5.0.8",
        "@vue/eslint-config-standard": "^6.1.0",
        "@vue/test-utils": "^1.3.6",
        "axios": "^1.6.5",
        "eslint": "^7.32.0",
        "eslint-plugin-import": "^2.26.0",
        "eslint-plugin-node": "^11.1.0",
        "eslint-plugin-promise": "^5.2.0",
        "eslint-plugin-vue": "^7.20.0",
        "jsdom": "^20.0.3",
        "keycloak-js": "^23.0.7",
        "less": "^4.1.2",
        "less-loader": "^10.2.0",
        "lodash": "^4.17.21",
        "msw": "^1.1.0",
        "portal-vue": "^2.1.7",
        "terser": "^5.30.4",
        "vitest": "^0.29.8",
        "vue-axios": "^3.4.1",
        "vue-i18n": "^8.27.1",
        "vue-sweetalert2": "^5.0.5",
        "vue-template-compiler": "^2.7.16",
        "vue-virtual-scroller": "^1.0.10",
        "workbox-webpack-plugin": "6.5.3"
    },
    "eslintConfig": {
        "root": true,
        "env": {
            "node": true
        },
        "extends": [
            "plugin:vue/essential",
            "@vue/standard"
        ],
        "parserOptions": {
            "ecmaVersion": 2021
        },
        "rules": {
            "indent": [
                "error",
                4
            ]
        }
    },
    "browserslist": [
        "> 1%",
        "last 2 versions",
        "not dead"
    ]
}

EDIT: The issue remains if I generate the slots in the same component where I fill them. It disappears if I just have two iterations and don't use any slots.


Solution

  • It seems to be such a strange error, and this seems to be caused by the same name of the iterator fieldConfig for both v-fors. It looks to me that the variables are shared for whatever reason. The easiest fix for this is to use a different iterator variable name for each v-for loops, e.g:

                <template v-for="(fieldConfig, key) in uploadFieldConfigs" #[fieldConfig.slotName]="data">
                    <p :key="key + '_upload_debug1'" v-html="'Upload | Key: |' + key + '| Data: |' + JSON.stringify(data) + '|'"></p>
                </template>
                <template v-for="(fieldConfig2, key) in searchFieldConfigs" #[fieldConfig2.slotName]="slotProps">
                    <p :key="key + '_upload_debug2'" v-html="'Search | Key: |' + key + '| Data: |' + JSON.stringify(slotProps) + '|'"></p>
                </template>
    

    Notice how I used fieldConfig2 for the iterator on the searchFieldConfigs. Given this, I suggest to change the key name too with something unique for each loops.