@@ -6,6 +6,7 @@ const expressions = require('posthtml-expressions');
6
6
const scriptDataLocals = require ( 'posthtml-expressions/lib/locals' ) ;
7
7
const { parser : parseToPostHtml } = require ( 'posthtml-parser' ) ;
8
8
const parseAttrs = require ( 'posthtml-attrs-parser' ) ;
9
+ // const matchHelper = require('posthtml-match-helper');
9
10
const { match} = require ( 'posthtml/lib/api' ) ;
10
11
const merge = require ( 'deepmerge' ) ;
11
12
const findPathFromTagName = require ( './find-path' ) ;
@@ -61,19 +62,24 @@ function processNodes(tree, options, messages) {
61
62
62
63
const index = node . content . findIndex ( content => typeof content === 'object' ) ;
63
64
64
- const nodeAttrs = parseAttrs ( node . content [ index ] . attrs ) ;
65
-
66
- Object . keys ( attributes ) . forEach ( attr => {
67
- if ( typeof defaultLocals [ attr ] === 'undefined' ) {
68
- if ( [ 'class' ] . includes ( attr ) ) {
69
- nodeAttrs [ attr ] . push ( attributes [ attr ] ) ;
70
- } else if ( [ 'style' ] . includes ( attr ) ) {
71
- nodeAttrs [ attr ] = attributes [ attr ] ;
65
+ if ( index !== - 1 ) {
66
+ // Map component attributes that it's not defined
67
+ // as locals to first element of node
68
+ // for now only class and style
69
+ const nodeAttrs = parseAttrs ( node . content [ index ] . attrs ) ;
70
+
71
+ Object . keys ( attributes ) . forEach ( attr => {
72
+ if ( typeof defaultLocals [ attr ] === 'undefined' ) {
73
+ if ( [ 'class' ] . includes ( attr ) ) {
74
+ nodeAttrs [ attr ] . push ( attributes [ attr ] ) ;
75
+ } else if ( [ 'style' ] . includes ( attr ) ) {
76
+ nodeAttrs [ attr ] = attributes [ attr ] ;
77
+ }
72
78
}
73
- }
74
- } ) ;
79
+ } ) ;
75
80
76
- node . content [ index ] . attrs = nodeAttrs . compose ( ) ;
81
+ node . content [ index ] . attrs = nodeAttrs . compose ( ) ;
82
+ }
77
83
78
84
messages . push ( {
79
85
type : 'dependency' ,
@@ -89,17 +95,17 @@ function processNodes(tree, options, messages) {
89
95
90
96
/**
91
97
* Parse locals from attributes, globals and via script
92
- * @param {Object } tree - posthtml tree
93
98
* @param {Object } options - plugin options
94
- * @param {Object } html - posthtml tree
99
+ * @param {Object } tree - PostHTML Node
100
+ * @param {Array } html - PostHTML Tree Nodes
95
101
* @return {Object } merged locals and default
96
102
*/
97
103
function parseLocals ( options , { attrs} , html ) {
98
104
let attributes = { ...attrs } ;
99
105
100
106
Object . keys ( attributes ) . forEach ( attribute => {
101
107
try {
102
- // Use merge()
108
+ // Use merge() ?
103
109
attributes = { ...attributes , ...JSON . parse ( attributes [ attribute ] ) } ;
104
110
} catch { }
105
111
} ) ;
@@ -122,14 +128,22 @@ function parseLocals(options, {attrs}, html) {
122
128
/**
123
129
* Merge slots content
124
130
* @param {Object } tree
125
- * @param {Object } component
131
+ * @param {Object } node
126
132
* @param {Boolean } strict
127
133
* @param {String } slotTagName
128
134
* @return {Object } tree
129
135
*/
130
- function mergeSlots ( tree , component , strict , slotTagName ) {
136
+ function mergeSlots ( tree , node , strict , slotTagName ) {
131
137
const slots = getSlots ( slotTagName , tree ) ; // Slot in component.html
132
- const fillSlots = getSlots ( slotTagName , component . content ) ; // Slot in page.html
138
+ const fillSlots = getSlots ( slotTagName , node . content ) ; // Slot in page.html
139
+
140
+ // Retrieve main content, means everything that is not inside slots
141
+ if ( node . content ) {
142
+ const contentOutsideSlots = node . content . filter ( content => content . tag !== 'slot' ) ;
143
+ if ( contentOutsideSlots . length > 0 ) {
144
+ fillSlots [ defaultSlotName ] = [ { tag : 'slot' , attrs : { name : defaultSlotName } , content : [ ...contentOutsideSlots ] } ] ;
145
+ }
146
+ }
133
147
134
148
for ( const slotName of Object . keys ( slots ) ) {
135
149
const fillSlotNodes = fillSlots [ slotName ] ;
@@ -145,6 +159,12 @@ function mergeSlots(tree, component, strict, slotTagName) {
145
159
continue ;
146
160
}
147
161
162
+ if ( ! slotNode . attrs ) {
163
+ slotNode . attrs = {
164
+ type : defaultSlotType
165
+ } ;
166
+ }
167
+
148
168
let slotType = slotTypes [ ( slotNode . attrs . type || defaultSlotType ) . toLowerCase ( ) ] || defaultSlotType ;
149
169
slotType = typeof slotNode . attrs . append === 'undefined' ? ( typeof slotNode . attrs . prepend === 'undefined' ? slotType : slotTypes . prepend ) : slotTypes . append ;
150
170
@@ -211,6 +231,8 @@ function getSlots(tag, content = []) {
211
231
node . attrs = { } ;
212
232
}
213
233
234
+ // When missing name try to retrieve from attribute
235
+ // so slot name can be shorthands like <slot content> => <slot name="content">
214
236
if ( ! node . attrs . name ) {
215
237
node . attrs . name = Object . keys ( { ...node . attrs } ) . find ( name => ! Object . keys ( slotTypes ) . includes ( name ) && name !== 'type' && name !== defaultSlotType ) ;
216
238
}
@@ -257,6 +279,7 @@ module.exports = (options = {}) => {
257
279
fileExtension : 'html' ,
258
280
tagPrefix : 'x-' ,
259
281
tagRegExp : new RegExp ( `^${ options . tagPrefix || 'x-' } ` , 'i' ) ,
282
+ defaultSlotRegex : new RegExp ( '^((?!(slot)).)*$' ) ,
260
283
slotTagName : 'slot' ,
261
284
tagName : 'component' ,
262
285
locals : { } ,
0 commit comments