新增:菜单组件增加国际化语言支持;

This commit is contained in:
iczer 2020-06-23 14:22:11 +08:00
parent 68249a0458
commit b16b5893c8
17 changed files with 267 additions and 165 deletions

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.0 KiB

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6.7 KiB

After

Width:  |  Height:  |  Size: 26 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 24 KiB

View File

@ -6,7 +6,7 @@
<h1>{{systemName}}</h1> <h1>{{systemName}}</h1>
</router-link> </router-link>
</div> </div>
<i-menu :theme="theme" :collapsed="collapsed" :menuData="menuData" @select="onSelect" class="menu"/> <i-menu :i18n="menuI18n" :theme="theme" :collapsed="collapsed" :options="menuData" @select="onSelect" class="menu"/>
</a-layout-sider> </a-layout-sider>
</template> </template>
@ -15,6 +15,7 @@ import IMenu from './menu'
export default { export default {
name: 'SiderMenu', name: 'SiderMenu',
components: {IMenu}, components: {IMenu},
inject: ['menuI18n'],
props: { props: {
collapsible: { collapsible: {
type: Boolean, type: Boolean,

View File

@ -1,35 +1,46 @@
/** /**
* 该插件可根据菜单配置自动生成 ANTD menu组件 * 该插件可根据菜单配置自动生成 ANTD menu组件
* menuData示例 * menuOptions示例
* [ * [
* { * {
* title: '菜单标题', * title: '菜单标题',
* icon: '菜单图标',
* path: '菜单路由', * path: '菜单路由',
* invisible: 'boolean, 是否不可见', * meta: {
* icon: '菜单图标',
* invisible: 'boolean, 是否不可见, 默认 false',
* },
* children: [子菜单配置] * children: [子菜单配置]
* }, * },
* { * {
* title: '菜单标题', * title: '菜单标题',
* icon: '菜单图标',
* path: '菜单路由', * path: '菜单路由',
* invisible: 'boolean, 是否不可见', * meta: {
* icon: '菜单图标',
* invisible: 'boolean, 是否不可见, 默认 false',
* },
* children: [子菜单配置] * children: [子菜单配置]
* } * }
* ] * ]
*
* i18n: 国际化配置组件默认会根据 options route配置的 path name 生成英文以及中文的国际化配置如需自定义或增加其他语言配置
* 此项即可
* i18n: {
* CN: {dashboard: {name: '监控中心'}}
* HK: {dashboard: {name: '監控中心'}}
* }
**/ **/
import Menu from 'ant-design-vue/es/menu' import Menu from 'ant-design-vue/es/menu'
import Icon from 'ant-design-vue/es/icon' import Icon from 'ant-design-vue/es/icon'
import '@/utils/Objects'
const {Item, SubMenu} = Menu const {Item, SubMenu} = Menu
// 默认菜单图标数组,如果菜单没配置图标,则会设置从该数组随机取一个图标配置
const iconArr = ['dashboard', 'user', 'form', 'setting', 'message', 'safety', 'bell', 'delete', 'code-o', 'poweroff', 'eye-o', 'hourglass']
export default { export default {
name: 'IMenu', name: 'IMenu',
i18n: {
},
props: { props: {
menuData: { options: {
type: Array, type: Array,
required: true required: true
}, },
@ -47,7 +58,8 @@ export default {
type: Boolean, type: Boolean,
required: false, required: false,
default: false default: false
} },
i18n: Object
}, },
data () { data () {
return { return {
@ -59,14 +71,26 @@ export default {
computed: { computed: {
rootSubmenuKeys: (vm) => { rootSubmenuKeys: (vm) => {
let keys = [] let keys = []
vm.menuData.forEach(item => { vm.options.forEach(item => {
keys.push(item.path) keys.push(item.path)
}) })
return keys return keys
} }
}, },
beforeMount() {
let CN = this.generateI18n(new Object(), this.options, 'name')
let US = this.generateI18n(new Object(), this.options, 'path')
this.$i18n.setLocaleMessage('CN', CN)
this.$i18n.setLocaleMessage('US', US)
if(this.i18n) {
Object.keys(this.i18n).forEach(key => {
this.$i18n.mergeLocaleMessage(key, this.i18n[key])
})
}
},
created () { created () {
this.updateMenu() this.updateMenu()
this.formatOptions(this.options, '')
}, },
watch: { watch: {
collapsed (val) { collapsed (val) {
@ -83,65 +107,72 @@ export default {
}, },
methods: { methods: {
renderIcon: function (h, icon) { renderIcon: function (h, icon) {
return icon === 'none' ? null return !icon || icon == 'none' ? null : h(Icon, {props: {type: icon}})
: h(
Icon,
{
props: {type: icon !== undefined ? icon : iconArr[Math.floor((Math.random() * iconArr.length))]}
})
}, },
renderMenuItem: function (h, menu, pIndex, index) { renderMenuItem: function (h, menu) {
return h( return h(
Item, Item, {key: menu.fullPath},
{
key: menu.path ? menu.path : 'item_' + pIndex + '_' + index
},
[ [
h( h('a', {attrs: {href: '#' + menu.fullPath}},
'RouterLink',
// {attrs: {href: '#' + menu.path}},
{attrs: {to: menu.path}},
[ [
this.renderIcon(h, menu.icon), this.renderIcon(h, menu.meta ? menu.meta.icon : 'none'),
h('span', [menu.name]) h('span', [this.$t(menu.fullPath.substring(1).replace(new RegExp('/', 'g'), '.') + '.name')])
] ]
) )
] ]
) )
}, },
renderSubMenu: function (h, menu, pIndex, index) { renderSubMenu: function (h, menu) {
var this2_ = this let this_ = this
var subItem = [h('span', let subItem = [h('span', {slot: 'title'},
{slot: 'title'},
[ [
this.renderIcon(h, menu.icon), this.renderIcon(h, menu.meta ? menu.meta.icon : 'none'),
h('span', [menu.name]) h('span', [this.$t(menu.fullPath.substring(1).replace(new RegExp('/', 'g'), '.') + '.name')])
] ]
)] )]
var itemArr = [] let itemArr = []
var pIndex_ = pIndex + '_' + index menu.children.forEach(function (item) {
menu.children.forEach(function (item, i) { itemArr.push(this_.renderItem(h, item))
itemArr.push(this2_.renderItem(h, item, pIndex_, i))
}) })
return h( return h(SubMenu, {key: menu.fullPath},
SubMenu,
{key: menu.path ? menu.path : 'submenu_' + pIndex + '_' + index},
subItem.concat(itemArr) subItem.concat(itemArr)
) )
}, },
renderItem: function (h, menu, pIndex, index) { renderItem: function (h, menu) {
if (!menu.invisible) { const meta = menu.meta
return menu.children ? this.renderSubMenu(h, menu, pIndex, index) : this.renderMenuItem(h, menu, pIndex, index) if (!meta || !meta.invisible) {
let renderChildren = false
const children = menu.children
if (children != undefined) {
for (let i = 0; i < children.length; i++) {
const childMeta = children[i].meta
if (!childMeta || !childMeta.invisible) {
renderChildren = true
break
}
}
}
return (menu.children && renderChildren) ? this.renderSubMenu(h, menu) : this.renderMenuItem(h, menu)
} }
}, },
renderMenu: function (h, menuTree) { renderMenu: function (h, menuTree) {
var this2_ = this let this_ = this
var menuArr = [] let menuArr = []
menuTree.forEach(function (menu, i) { menuTree.forEach(function (menu, i) {
menuArr.push(this2_.renderItem(h, menu, '0', i)) menuArr.push(this_.renderItem(h, menu, '0', i))
}) })
return menuArr return menuArr
}, },
formatOptions(options, parentPath) {
let this_ = this
options.forEach(route => {
let isFullPath = route.path.substring(0, 1) == '/'
route.fullPath = isFullPath ? route.path : parentPath + '/' + route.path
if (route.children) {
this_.formatOptions(route.children, route.fullPath)
}
})
},
onOpenChange (openKeys) { onOpenChange (openKeys) {
const latestOpenKey = openKeys.find(key => this.openKeys.indexOf(key) === -1) const latestOpenKey = openKeys.find(key => this.openKeys.indexOf(key) === -1)
if (this.rootSubmenuKeys.indexOf(latestOpenKey) === -1) { if (this.rootSubmenuKeys.indexOf(latestOpenKey) === -1) {
@ -152,12 +183,29 @@ export default {
}, },
updateMenu () { updateMenu () {
let routes = this.$route.matched.concat() let routes = this.$route.matched.concat()
this.selectedKeys = [routes.pop().path] const route = routes.pop()
this.selectedKeys = [this.getSelectedKey(route)]
let openKeys = [] let openKeys = []
routes.forEach((item) => { routes.forEach((item) => {
openKeys.push(item.path) openKeys.push(item.path)
}) })
this.collapsed || this.mode === 'horizontal' ? this.cachedOpenKeys = openKeys : this.openKeys = openKeys this.collapsed || this.mode === 'horizontal' ? this.cachedOpenKeys = openKeys : this.openKeys = openKeys
},
getSelectedKey (route) {
if (route.meta.invisible && route.parent) {
return this.getSelectedKey(route.parent)
}
return route.path
},
generateI18n(lang, options, valueKey) {
options.forEach(menu => {
let keys = menu.fullPath.substring(1).split('/').concat('name')
lang.assignProps(keys, menu[valueKey])
if (menu.children) {
this.generateI18n(lang, menu.children, valueKey)
}
})
return lang
} }
}, },
render (h) { render (h) {
@ -177,7 +225,7 @@ export default {
this.$emit('select', obj) this.$emit('select', obj)
} }
} }
}, this.renderMenu(h, this.menuData) }, this.renderMenu(h, this.options)
) )
} }
} }

View File

@ -1,5 +1,6 @@
// 系统配置 // 系统配置
module.exports = { module.exports = {
lang: 'CN',
themeColor: '#1890ff', themeColor: '#1890ff',
theme: 'dark', theme: 'dark',
layout: 'side', layout: 'side',

View File

@ -8,7 +8,7 @@
<a-divider v-if="isMobile" type="vertical" /> <a-divider v-if="isMobile" type="vertical" />
<a-icon v-if="layout === 'side'" class="trigger" :type="collapsed ? 'menu-unfold' : 'menu-fold'" @click="toggleCollapse"/> <a-icon v-if="layout === 'side'" class="trigger" :type="collapsed ? 'menu-unfold' : 'menu-fold'" @click="toggleCollapse"/>
<div v-if="layout == 'head' && !isMobile" class="global-header-menu"> <div v-if="layout == 'head' && !isMobile" class="global-header-menu">
<i-menu style="height: 64px; line-height: 64px;" :theme="headerTheme" mode="horizontal" :menuData="menuData" @select="onSelect"/> <i-menu style="height: 64px; line-height: 64px;" :i18n="menuI18n" :theme="headerTheme" mode="horizontal" :options="menuData" @select="onSelect"/>
</div> </div>
<div :class="['global-header-right', headerTheme]"> <div :class="['global-header-right', headerTheme]">
<header-search class="header-item" /> <header-search class="header-item" />
@ -19,6 +19,16 @@
</a-tooltip> </a-tooltip>
<header-notice class="header-item"/> <header-notice class="header-item"/>
<header-avatar class="header-item"/> <header-avatar class="header-item"/>
<a-dropdown class="lang header-item">
<div>
<a-icon type="global"/>
</div>
<a-menu @click="changeLang" :selected-keys="[lang]" slot="overlay">
<a-menu-item key="CN"><span >cn</span> 简体中文</a-menu-item>
<a-menu-item key="HK"><span >hk</span> 繁体中文</a-menu-item>
<a-menu-item key="US"><span >us</span> English</a-menu-item>
</a-menu>
</a-dropdown>
</div> </div>
</div> </div>
</a-layout-header> </a-layout-header>
@ -40,8 +50,9 @@ export default {
headerTheme: this.theme headerTheme: this.theme
} }
}, },
inject: ['menuI18n'],
computed: { computed: {
...mapState('setting', ['theme', 'isMobile', 'layout', 'systemName']), ...mapState('setting', ['theme', 'isMobile', 'layout', 'systemName', 'lang']),
headerTheme () { headerTheme () {
return (this.layout == 'side' && !this.isMobile) ? 'light' : this.theme return (this.layout == 'side' && !this.isMobile) ? 'light' : this.theme
} }
@ -52,6 +63,9 @@ export default {
}, },
onSelect (obj) { onSelect (obj) {
this.$emit('menuSelect', obj) this.$emit('menuSelect', obj)
},
changeLang(lang) {
this.$store.commit('setting/setLang', lang.key)
} }
} }
} }
@ -69,7 +83,7 @@ export default {
} }
} }
.global-header{ .global-header{
padding: 0 12px 0 0; padding: 0;
-webkit-box-shadow: 0 1px 4px rgba(0,21,41,.08); -webkit-box-shadow: 0 1px 4px rgba(0,21,41,.08);
box-shadow: 0 1px 4px rgba(0,21,41,.08); box-shadow: 0 1px 4px rgba(0,21,41,.08);
z-index: 1; z-index: 1;
@ -124,11 +138,17 @@ export default {
a, i{ a, i{
color: #fff !important; color: #fff !important;
} }
.header-item:hover{
background-color: @primary-color;
}
} }
.header-item{ .header-item{
padding: 0 12px; padding: 0 12px;
cursor: pointer; cursor: pointer;
align-self: center; align-self: center;
&:hover{
background-color: @gray-3;
}
i{ i{
font-size: 16px; font-size: 16px;
color: rgba(0,0,0,.65); color: rgba(0,0,0,.65);

View File

@ -51,6 +51,7 @@ export default {
provide() { provide() {
return{ return{
layoutMinHeight: minHeight, layoutMinHeight: minHeight,
menuI18n: require('@/router/i18n').default
} }
}, },
computed: { computed: {

View File

@ -1,5 +1,5 @@
<template> <template>
<span class="header-search"> <div class="header-search">
<a-icon type="search" class="search-icon" @click="enterSearchMode"/> <a-icon type="search" class="search-icon" @click="enterSearchMode"/>
<a-auto-complete <a-auto-complete
ref="input" ref="input"
@ -9,7 +9,7 @@
@blur="leaveSearchMode" @blur="leaveSearchMode"
> >
</a-auto-complete> </a-auto-complete>
</span> </div>
</template> </template>
<script> <script>

View File

@ -109,7 +109,7 @@ export default {
if (result.code >= 0) { if (result.code >= 0) {
const user = result.data.user const user = result.data.user
this.$router.push('/dashboard/workplace') this.$router.push('/dashboard/workplace')
this.$store.commit('account/setuser', user) this.$store.commit('account/setUser', user)
this.$message.success(result.message, 3) this.$message.success(result.message, 3)
} else { } else {
this.error = result.message this.error = result.message

48
src/router/i18n.js Normal file
View File

@ -0,0 +1,48 @@
export default {
HK: {
dashboard: {
name: 'Dashboard',
workplace: {name: '工作台'},
analysis: {name: '分析頁'}
},
form: {
name: '表單頁',
basic: {name: '基礎表單'},
step: {name: '分步表單'},
advance: {name: '分步表單'}
},
list: {
name: '列表頁',
query: {name: '查詢表格'},
primary: {name: '標準列表'},
card: {name: '卡片列表'},
search: {
name: '搜索列表',
article: {name: '文章'},
application: {name: '應用'},
project: {name: '項目'}
}
},
details: {
name: '詳情頁',
basic: {name: '基礎詳情頁'},
advance: {name: '高級詳情頁'}
},
result: {
name: '結果頁',
success: {name: '成功'},
error: {name: '失敗'}
},
exception: {
name: '異常頁',
404: {name: '404'},
403: {name: '403'},
500: {name: '500'}
},
components: {
name: '小組件',
taskCard: {name: '任務卡片'},
palette: {name: '顏色複選框'}
}
}
}

View File

@ -20,190 +20,180 @@ export default new Router({
name: '首页', name: '首页',
component: MenuView, component: MenuView,
redirect: '/login', redirect: '/login',
icon: 'none',
invisible: true,
children: [ children: [
{ {
path: '/dashboard', path: 'dashboard',
name: 'Dashboard', name: 'Dashboard',
meta: {
icon: 'dashboard'
},
component: RouteView, component: RouteView,
icon: 'dashboard',
children: [ children: [
{ {
path: '/dashboard/workplace', path: 'workplace',
name: '工作台', name: '工作台',
component: () => import('@/pages/dashboard/WorkPlace'), component: () => import('@/pages/dashboard/WorkPlace'),
icon: 'none'
}, },
{ {
path: '/dashboard/analysis', path: 'analysis',
name: '分析页', name: '分析页',
component: () => import('@/pages/dashboard/Analysis'), component: () => import('@/pages/dashboard/Analysis'),
icon: 'none'
} }
] ]
}, },
{ {
path: '/form', path: 'form',
name: '表单页', name: '表单页',
component: PageView, meta: {
icon: 'form', icon: 'form',
},
component: PageView,
children: [ children: [
{ {
path: '/form/basic', path: 'basic',
name: '基础表单', name: '基础表单',
component: () => import('@/pages/form/BasicForm'), component: () => import('@/pages/form/BasicForm'),
icon: 'none'
}, },
{ {
path: '/form/step', path: 'step',
name: '分步表单', name: '分步表单',
component: () => import('@/pages/form/stepForm/StepForm'), component: () => import('@/pages/form/stepForm/StepForm'),
icon: 'none'
}, },
{ {
path: '/form/advanced', path: 'advance',
name: '高级表单', name: '高级表单',
component: () => import('@/pages/form/advancedForm/AdvancedForm'), component: () => import('@/pages/form/advancedForm/AdvancedForm'),
icon: 'none'
} }
] ]
}, },
{ {
path: '/list', path: 'list',
name: '列表页', name: '列表页',
meta: {
icon: 'table'
},
component: PageView, component: PageView,
icon: 'table',
children: [ children: [
{ {
path: '/list/query', path: 'query',
name: '查询表格', name: '查询表格',
component: () => import('@/pages/list/QueryList'), component: () => import('@/pages/list/QueryList'),
icon: 'none'
}, },
{ {
path: '/list/primary', path: 'primary',
name: '标准列表', name: '标准列表',
component: () => import('@/pages/list/StandardList'), component: () => import('@/pages/list/StandardList'),
icon: 'none'
}, },
{ {
path: '/list/card', path: 'card',
name: '卡片列表', name: '卡片列表',
component: () => import('@/pages/list/CardList'), component: () => import('@/pages/list/CardList'),
icon: 'none'
}, },
{ {
path: '/list/search', path: 'search',
name: '搜索列表', name: '搜索列表',
component: () => import('@/pages/list/search/SearchLayout'), component: () => import('@/pages/list/search/SearchLayout'),
icon: 'none',
children: [ children: [
{ {
path: '/list/search/article', path: 'article',
name: '文章', name: '文章',
component: () => import('@/pages/list/search/ArticleList'), component: () => import('@/pages/list/search/ArticleList'),
icon: 'none'
}, },
{ {
path: '/list/search/application', path: 'application',
name: '应用', name: '应用',
component: () => import('@/pages/list/search/ApplicationList'), component: () => import('@/pages/list/search/ApplicationList'),
icon: 'none'
}, },
{ {
path: '/list/search/project', path: 'project',
name: '项目', name: '项目',
component: () => import('@/pages/list/search/ProjectList'), component: () => import('@/pages/list/search/ProjectList'),
icon: 'none'
} }
] ]
} }
] ]
}, },
{ {
path: '/detail', path: 'details',
name: '详情页', name: '详情页',
icon: 'profile', meta: {
icon: 'profile'
},
component: RouteView, component: RouteView,
children: [ children: [
{ {
path: '/detail/basic', path: 'basic',
name: '基础详情页', name: '基础详情页',
icon: 'none',
component: () => import('@/pages/detail/BasicDetail') component: () => import('@/pages/detail/BasicDetail')
}, },
{ {
path: '/detail/advanced', path: 'advance',
name: '高级详情页', name: '高级详情页',
icon: 'none',
component: () => import('@/pages/detail/AdvancedDetail') component: () => import('@/pages/detail/AdvancedDetail')
} }
] ]
}, },
{ {
path: '/result', path: 'result',
name: '结果页', name: '结果页',
meta: {
icon: 'check-circle-o', icon: 'check-circle-o',
},
component: PageView, component: PageView,
children: [ children: [
{ {
path: '/result/success', path: 'success',
name: '成功', name: '成功',
icon: 'none',
component: () => import('@/pages/result/Success') component: () => import('@/pages/result/Success')
}, },
{ {
path: '/result/error', path: 'error',
name: '失败', name: '失败',
icon: 'none',
component: () => import('@/pages/result/Error') component: () => import('@/pages/result/Error')
} }
] ]
}, },
{ {
path: '/exception', path: 'exception',
name: '异常页', name: '异常页',
meta: {
icon: 'warning', icon: 'warning',
},
component: RouteView, component: RouteView,
children: [ children: [
{ {
path: '/exception/404', path: '404',
name: '404', name: '404',
icon: 'none',
component: () => import('@/pages/exception/404') component: () => import('@/pages/exception/404')
}, },
{ {
path: '/exception/403', path: '403',
name: '403', name: '403',
icon: 'none',
component: () => import('@/pages/exception/403') component: () => import('@/pages/exception/403')
}, },
{ {
path: '/exception/500', path: '500',
name: '500', name: '500',
icon: 'none',
component: () => import('@/pages/exception/500') component: () => import('@/pages/exception/500')
} }
] ]
}, },
{ {
path: '/components', path: 'components',
redirect: '/components/taskcard',
name: '小组件', name: '小组件',
icon: 'appstore-o', meta: {
icon: 'appstore-o'
},
component: PageView, component: PageView,
children: [ children: [
{ {
path: '/components/taskcard', path: 'taskCard',
name: '任务卡片', name: '任务卡片',
icon: 'none',
component: () => import('@/pages/components/TaskCard') component: () => import('@/pages/components/TaskCard')
}, },
{ {
path: '/components/palette', path: 'palette',
name: '颜色复选框', name: '颜色复选框',
icon: 'none',
component: () => import('@/pages/components/Palette') component: () => import('@/pages/components/Palette')
} }
] ]

View File

@ -1,6 +1,6 @@
import PouchDB from 'pouchdb' import PouchDB from 'pouchdb'
var db = new PouchDB('admindb') let db = new PouchDB('adminDb')
export default { export default {
namespaced: true, namespaced: true,
@ -8,7 +8,7 @@ export default {
user: {} user: {}
}, },
mutations: { mutations: {
setuser (state, user) { setUser (state, user) {
state.user = user state.user = user
db.get('currUser').then(doc => { db.get('currUser').then(doc => {
db.put({ db.put({

View File

@ -33,6 +33,9 @@ export default {
}, },
setFixedSider(state, fixedSider) { setFixedSider(state, fixedSider) {
state.fixedSider = fixedSider state.fixedSider = fixedSider
},
setLang(state, lang) {
state.lang = lang
} }
} }
} }

View File

@ -14,3 +14,17 @@
@primary-8: color(~`colorPalette('@{primary}', 8) `); @primary-8: color(~`colorPalette('@{primary}', 8) `);
@primary-9: color(~`colorPalette('@{primary}', 9) `); @primary-9: color(~`colorPalette('@{primary}', 9) `);
@primary-10: color(~`colorPalette('@{primary}', 10) `); @primary-10: color(~`colorPalette('@{primary}', 10) `);
@gray-1: #ffffff;
@gray-2: #fafafa;
@gray-3: #f5f5f5;
@gray-4: #f0f0f0;
@gray-5: #d9d9d9;
@gray-6: #bfbfbf;
@gray-7: #8c8c8c;
@gray-8: #595959;
@gray-9: #434343;
@gray-10: #262626;
@gray-11: #1f1f1f;
@gray-12: #141414;
@gray-13: #000000;

24
src/utils/Objects.js Normal file
View File

@ -0,0 +1,24 @@
/**
* 给对象注入属性
* @param keys 属性key数组 keys = ['config', 'path'] , 则会给对象注入 object.config.path 的属性
* @param value 属性值
* @returns {Object}
*/
Object.defineProperty(Object.prototype, 'assignProps', {
writable: false,
enumerable: false,
configurable: true,
value: function (keys, value) {
let props = this
for (let i = 0; i < keys.length; i++) {
let key = keys[i]
if (i == keys.length - 1) {
props[key] = value
} else {
props[key] = props[key] == undefined ? {} : props[key]
props = props[key]
}
}
return this
}
})

View File

@ -1,48 +0,0 @@
.textOverflow() {
overflow: hidden;
text-overflow: ellipsis;
word-break: break-all;
white-space: nowrap;
}
.textOverflowMulti(@line: 3, @bg: #fff) {
overflow: hidden;
position: relative;
line-height: 1.5em;
max-height: @line * 1.5em;
text-align: justify;
margin-right: -1em;
padding-right: 1em;
&:before {
background: @bg;
content: '...';
padding: 0 1px;
position: absolute;
right: 14px;
bottom: 0;
}
&:after {
background: white;
content: '';
margin-top: 0.2em;
position: absolute;
right: 14px;
width: 1em;
height: 1em;
}
}
.clearfix() {
zoom: 1;
&:before,
&:after {
content: ' ';
display: table;
}
&:after {
clear: both;
visibility: hidden;
font-size: 0;
height: 0;
}
}