chore: add SoftSkill, fix panel by adding two InfoPanel views

This commit is contained in:
JesusPerez 2022-03-21 20:24:15 +00:00
parent b8578c3eb6
commit 8d2d0d4310
9 changed files with 691 additions and 45 deletions

View File

@ -1,3 +1,5 @@
en: English
es: Spanish
edit: Edit
onedit: Edit On
changeview: View
@ -85,6 +87,7 @@ cv:
languages: Languages
infrastructures: Infrastructures
skills_tools: Skills & Tools
soft_skills: Soft Skills
skills: Skills
onweb: On the Web
mission: Mission

View File

@ -1,3 +1,5 @@
en: Inglés
es: Español
edit: Editar
onedit: Activar Edición
changeview: Vista
@ -83,6 +85,7 @@ cv:
languages: Idiomas
infrastructures: Infrastructuras
skills_tools: Habilidades y Herramientas
soft_skills: Aptitudes
skills: Habilidades
onweb: En la Web
mission: Misión

View File

@ -282,7 +282,9 @@ const onNavMenu = (item: string) => {
}
}
const onHome = () => {
router.push(useState().APPHOME.value)
goTopPage()
// router.push(router.currentRoute.value.fullPath)
// router.push(useState().APPHOME.value)
}
const goTopPage = () => {
const dom_body = document.getElementsByTagName('body')[0]

View File

@ -132,6 +132,7 @@ export interface DataCoreType {
profile: ProfileType[]
certifications: CertificationType[]
skills: SkillsType[]
soft_skills: SkillsType[]
infra: SkillsType[]
sites: SitesType[]
langs: LangsType[]
@ -241,6 +242,8 @@ export interface ShowInfoType {
sites: boolean
skills: boolean
skills_itms: boolean
soft_skills: boolean
soft_skills_itms: boolean
infra: boolean
certs: boolean
langs: boolean

View File

@ -40,7 +40,14 @@
id="sidebar"
class="rounded-r w-full lg:w-80 sm:max-w-sm p-8 border-l-1 border-indigo-200 bg-gradient-to-b from-indigo-300 via-indigo-200 to-indigo-100 dark:bg-gray-600 dark:from-indigo-500 dark:via-indigo-400 dark:to-indigo-800"
>
<info-panel
<info-panel-main
:data="data_info.core"
:localedata="localedata"
:showinfo="show_info"
:authinfo="auth_info"
@onItem="onItem"
@onEditor="onEditor" />
<info-panel-skills
:data="data_info.core"
:localedata="localedata"
:showinfo="show_info"
@ -49,6 +56,19 @@
@onEditor="onEditor" />
</div>
<div class="content w-full p-5">
<div
v-if="!show_infopanel && data_info.core && data_info.core.title1"
id="main-info"
class="rounded-r w-full lg:w-150 sm:max-w-sm p-8 mb-5 bg-gradient-to-b from-indigo-300 via-indigo-200 to-indigo-100 dark:bg-gray-600 dark:from-indigo-500 dark:via-indigo-400 dark:to-indigo-800"
>
<info-panel-main
:data="data_info.core"
:localedata="localedata"
:showinfo="show_info"
:authinfo="auth_info"
@onItem="onItem"
@onEditor="onEditor" />
</div>
<div class="prose">
<h2
:class="show_info.profile ? 'section-headline' : `noprint ${auth_info.viewchange ? 'pb-11' : ''}`"
@ -160,42 +180,28 @@
<div
v-if="!show_infopanel"
id="cv-skills"
class="prose"
class="prose rounded-r w-full lg:w-150 sm:max-w-sm p-8 border-l-1 border-indigo-200 bg-gradient-to-b from-indigo-300 via-indigo-200 to-indigo-100 dark:bg-gray-600 dark:from-indigo-500 dark:via-indigo-400 dark:to-indigo-800"
>
<h2 class="section-headline">
<span v-if="show_info.skills">{{ t('cv.skills_tools', 'Skill & Tools') }}</span>
<button
v-if="auth_info.viewchange"
class="no-print text-sm float-right icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onSkills()"
<button
class="no-print text-sm float-right icon-btn mt-2 mr-2 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onHome()"
>
<div v-if="show_info.skills" i-carbon-view-off />
<div v-else class="noprint flex-grow-0 flex -mb-5">
<div class="-mt-0.5 mr-2 line-through text-xs">{{ t('cv.skills_tools', 'Skill & Tools') }}</div>
<div i-carbon-view />
</div>
</button>
<button
v-if="show_info.skills"
class="no-print text-sm icon-btn mr-2 float-right !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onHome()"
>
<div i-carbon-home />
</button>
</h2>
<div v-if="show_info.skills" class="list-none w-3/5">
<skills-view
:data="data_info.core.skills"
:localedata="localedata"
:showinfo="show_info.skills"
:authinfo="show_info.auth" />
</div>
<div i-carbon-home />
</button>
<info-panel-skills
:data="data_info.core"
:localedata="localedata"
:showinfo="show_info"
:authinfo="auth_info"
@onItem="onItem"
@onEditor="onEditor" />
</div>
</div>
</main>
<div
v-if="show_content && cvdata.core && cvdata.core.name"
class="mr-auto w-full lg:w-1/2 text-center text-sm py-2 pr-5 text-gray-600 border-gray-300 border-1 border-b-0 rounded-t-lg"
>
<nav-menu
:position="NavPosition.footer"
@ -230,18 +236,16 @@
</template>
<script setup lang="ts">
import { useI18n } from 'vue-i18n'
const i18n = useI18n()
const { t } = useI18n()
const router = useRouter()
import InfoPanel from '~/views/cv/InfoPanel.vue'
// import InfoPanel from '~/views/cv/InfoPanel.vue'
import InfoPanelMain from '~/views/cv/InfoPanelMain.vue'
import InfoPanelSkills from '~/views/cv/InfoPanelSkills.vue'
import Profile from '~/views/cv/Profile.vue'
import WorkExperienceView from '~/views/cv/WorkExperience.vue'
import ProjectsView from '~/views/cv/Projects.vue'
import TeachingView from '~/views/cv/Teaching.vue'
import TalksView from '~/views/cv/Talks.vue'
import EducationView from '~/views/cv/Education.vue'
import SkillsView from '~/views/cv/Skills.vue'
// import SkillsView from '~/views/cv/Skills.vue'
import OthersView from '~/views/cv/Others.vue'
import MessageBoxView from '@/MessageBoxView.vue'
import NavMenu from '@/NavMenu.vue'
@ -255,6 +259,10 @@ import { track_action } from '~/hooks/tracking'
import { MessageType} from '~/typs'
import { MessageBoxType, NavPosition, InputBtnsType, ModelType, ShowInfoType } from '~/typs/cv'
const i18n = useI18n()
const { t } = useI18n()
const router = useRouter()
const prefix = 'cv'
const assetsPath = useState().ASSETS_PATH
const routeKy = router.currentRoute.value.params.ky || router.currentRoute.value.query.k || ''
@ -474,6 +482,11 @@ const onSkills = () => {
useState().showinfo.value.skills_itms = show_info.value.skills
// !useState().showinfo.value.skills_itms
}
const onSoftSkills = () => {
show_info.value.soft_skills = !show_info.value.soft_skills
useState().showinfo.value.soft_skills_itms = show_info.value.soft_skills
// !useState().showinfo.value.soft_skills_itms
}
const onLoadModel = (model: { id: string}) => {
const url = useState().CONFURLS.value.data || ''
if (useState().models.value[model.id] && url.length > 0 ) {

View File

@ -284,6 +284,7 @@
</h2>
<skills-view
v-if="showinfo.skills"
src="soft_skills"
:data="data.skills"
:localedata="localedata"
:showinfo="showinfo"
@ -291,17 +292,53 @@
@onItem="onSkills"
/>
</div>
<div v-if="showinfo.infra" class="panel-group">
<h2 class="h2-title">
<span>{{ t('cv.infrastructures', 'Infrastructures') }}</span>
<div
id="info-soft_skills"
:class="{ 'hidden': !showinfo.soft_skills && !authinfo.viewchange }"
class="panel-group -mt-2"
>
<h2 class="h2-title flex">
<span v-if="showinfo.soft_skills">{{ t('cv.soft_skills', 'Soft Skills') }}</span>
<div v-if="authinfo.viewchange" class="flex-grow-0 flex noprint ml-2">
<span v-if="!showinfo.soft_skills" class="w-4/5 flex-grow" />
<button
class="no-print text-base flex-grow-0 icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('soft_skills')"
>
<div v-if="showinfo.soft_skills" class="flex" i-carbon-view-off />
<div v-else class="noprint flex">
<div class="mr-2 text-sm line-through">{{ t('cv.soft_skills', 'Soft Skills').replaceAll(' ','') }}</div>
<div class="flex-grow-0" i-carbon-view />
</div>
</button>
</div>
</h2>
<skills-view
v-if="showinfo.soft_skills"
src="soft_skills"
:data="data.soft_skills"
:localedata="localedata"
:showinfo="showinfo"
:authinfo="authinfo"
@onItem="onSoftSkills"
/>
</div>
<div :class="{ 'hidden': !showinfo.infra && !authinfo.viewchange }" class="panel-group">
<h2 class="h2-title flex">
<span v-if="showinfo.infra">{{ t('cv.infrastructures', 'Infrastructures') }}</span>
<button
class="no-print text-sm float-right icon-btn mt-2 !outline-none text-gray-500 dark:text-gray-400"
v-if="authinfo.viewchange"
class="no-print text-sm float-right icon-btn ml-2 mt-1 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('infra')"
>
<div i-carbon-view-off />
<div v-if="showinfo.infra" i-carbon-view-off />
<div v-else class="noprint flex flex-grow-0">
<div class="-mt-0.5 mr-2 line-through">{{ t('cv.infrastructures', 'Infrastructures') }}</div>
<div class="flex-grow-0" i-carbon-view />
</div>
</button>
</h2>
<div class="-ml-5 mt-2 leading-8 text-sm grid grid-flow-col grid-cols-2 grid-rows-2 gap-2">
<div v-if="showinfo.infra" class="-ml-5 mt-2 leading-8 text-sm grid grid-flow-col grid-cols-2 grid-rows-2 gap-2">
<div
v-for="infra in data.infra"
:key="infra.id"
@ -397,7 +434,7 @@ import useState from '~/hooks/useState'
import { track_action } from '~/hooks/tracking'
const emit = defineEmits(['onEditor', 'onItem'])
const { t } = useI18n()
const router = useRouter()
// const router = useRouter()
const props = defineProps({
data: {
type: Object as PropType<DataCoreType>,
@ -433,6 +470,8 @@ const onView = (itm: string) => {
showinfo.value[itm] = !showinfo.value[itm]
if (itm === 'skills')
useState().showinfo.value.skills = showinfo.value[itm]
else if (itm === 'soft_skills')
useState().showinfo.value.soft_skills = showinfo.value[itm]
}
}
const onEditor = (info: { src: string, field: string, idx: number, data: string, ev: Event }) => {
@ -470,6 +509,9 @@ const onItem = (src: string, idx: number) => {
const onSkills = (data: { src: string, idx: number }) => {
emit('onItem', { root: 'info', src: data.src, itm: props.data[data.src][data.idx], idx: data.idx })
}
const onSoftSkills = (data: { src: string, idx: number }) => {
emit('onItem', { root: 'info', src: data.src, itm: props.data[data.src][data.idx], idx: data.idx })
}
const onLink = (e: any) => {
track_action(e)
}

View File

@ -0,0 +1,348 @@
<template>
<div class="px-1 mb-8">
<div :class="{ 'flex': showinfo.image }">
<img
v-if="showinfo.image"
:src="`${assets_path}${get_data('imgsrc')}`"
:alt="get_data('imgalt')"
class="flex-grow-0 rounded-full w-48 max-w-xs mx-auto mb-2 border-indigo-200 border-2"
/>
<button
v-if="authinfo.viewchange"
:class="showinfo.image ? 'flex-grow-0 mt-2' : 'float-right -mt-4'"
class="no-print text-sm h-6 icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('image')"
>
<div i-carbon-view-off />
</button>
</div>
<h1
:class="showinfo.fullname ? 'text-lg' : 'text-2xl'"
class="text-center text-indigo-700 dark:text-indigo-200 font-semibold mb-2"
>{{ `${showinfo.fullname ? get_data('fullname') : get_data('name')}` }}</h1>
<h2
v-if="showinfo.title && data.title1 !== ''"
class="text-center text-sm font-light text-indigo-500 dark:text-indigo-100"
>
<tiptap-editor
v-if="authinfo.editable"
:data="data.title1"
:editable="authinfo.editable"
src="info"
field="title1"
:idx="-1"
@onEditorBlur="onEditor"
/>
<span v-else v-html="data.title1" />
</h2>
<h2
v-if="showinfo.title && data.title2 !== ''"
class="text-center text-sm font-light text-indigo-500 dark:text-indigo-100"
>
<tiptap-editor
v-if="authinfo.editable"
:data="data.title2"
:editable="authinfo.editable"
src="info"
field="title2"
:idx="-1"
@onEditorBlur="onEditor"
/>
<span v-else v-html="data.title2" />
</h2>
</div>
<div
v-if="data.mission !== ''"
class="font-light text-lg px-1"
:class="{ 'mb-8': showinfo.mission && authinfo.viewchange }"
>
<h2 class="text-center h2-title dark:text-indigo-900">
<span v-if="showinfo.mission">{{ t('cv.mission', 'Mission') }}</span>
<button
v-if="authinfo.viewchange"
class="no-print text-sm float-right icon-btn mt-2 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('mission')"
>
<div v-if="showinfo.mission" i-carbon-view-off />
<div v-else class="noprint flex -mt-7">
<div class="-mt-0.5 mr-2 line-through">{{ t('cv.mission', 'Mission') }}</div>
<div i-carbon-view />
</div>
</button>
</h2>
<p
v-if="showinfo.mission"
class="-ml-3 mb-2 text-xs font-semibold text-indigo-900 dark:text-indigo-200 tracking-normal"
>
<tiptap-editor
v-if="authinfo.editable"
:data="get_data('mission')"
:editable="authinfo.editable"
src="info"
field="title1"
:idx="-1"
@onEditorBlur="onEditor"
/>
<span v-else v-html="get_data('mission')" />
</p>
<ul
v-if="showinfo.mission && showinfo.mission_how && get_data('mission_how').length > 0"
class="ml-2 list-disc"
>
<li
v-for="(mitem,index) in get_data('mission_how')"
:key="index"
class="text-xs tracking-normal font-light text-indigo-800 dark:text-gray-100"
:class="{ 'noprint list-none': !mitem.auth.show }"
>
<div class="flex">
<button
v-if="authinfo.viewchange"
class="noprint flex-grow-0 icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onItem('mission_how', index)"
>
<div v-if="mitem.auth.show" class="mr-1" i-carbon-view-off />
<div v-else class="noprint flex flex-row">
<div class="mt-0.2 mr-2 line-through text-xs">{{ index }}</div>
<div i-carbon-view />
</div>
</button>
<div v-if=" mitem.auth.show">
<tiptap-editor
v-if="authinfo.editable && mitem.auth.editable"
:data="mitem.desc"
:editable="authinfo.editable"
src="info.mission_how"
field="desc"
:idx="index"
@onEditorBlur="onEditor"
/>
<span v-else>{{ mitem.desc }}</span>
</div>
</div>
</li>
</ul>
<button
v-if="authinfo.viewchange"
class="no-print text-sm float-right icon-btn mt-2 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('mission_how')"
>
<div v-if="showinfo.mission_how" i-carbon-view-off />
<div v-else class="noprint flex -mt-7">
<div class="-mt-0.5 mr-2 line-through">{{ t('cv.mission_how', 'Mission How') }}</div>
<div i-carbon-view />
</div>
</button>
</div>
<div class="font-light text-lg px-1 mt-4 mb-8 panel-group">
<h2 class="h2-title">{{ t('cv.contact', 'Contact') }}</h2>
<div class="flex items-center my-3">
<img :src="`${assets_path}/images/assets/mail-outline.svg`" class="inline w-6 mr-4" alt="Mail icon" />
<a @click="onLink" :href="`mailto:${data.email}`">{{ data.email }}</a>
</div>
<div
:class="{ 'hidden': !showinfo.phone && !authinfo.viewchange }"
class="flex items-center my-3"
>
<img
v-if="showinfo.phone"
:src="`${assets_path}/images/assets/call-outline.svg`"
class="inline w-6 mr-4"
alt="Phone icon"
/>
<a v-if="showinfo.phone" class="text-sm" @click="onLink" :href="`tel:${data.phone}`">
<tiptap-editor
v-if="authinfo.editable"
:data="data.phone"
:editable="authinfo.editable"
src="info"
field="phone"
:idx="-1"
@onEditorBlur="onEditor"
/>
<span v-else>{{ data.phone }}</span>
</a>
<span v-if="!showinfo.phone" class="w-4/5 flex-grow" />
<button
v-if="authinfo.viewchange"
class="no-print text-sm flex icon-btn ml-4 mt-0 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('phone')"
>
<div v-if="showinfo.phone" i-carbon-view-off />
<div v-else class="noprint flex-grow-0 flex -mb-5">
<div class="-mt-0.5 mr-2 line-through">{{ t('phone', 'Phone') }}</div>
<div i-carbon-view />
</div>
</button>
</div>
<div
:class="{ 'hidden': !showinfo.address && !authinfo.viewchange }"
class="flex flex-row my-5"
>
<div>
<img
v-if="showinfo.address"
:src="`${assets_path}/images/assets/home-outline.svg`"
class="flex-grow-0 w-6 pb-1 mr-4"
alt="House icon"
/>
</div>
<div v-if="showinfo.address" class="text-xs">
<p>{{ data.address }}</p>
<p class="text-xs font-semibold">{{ get_data('city') }}</p>
<p>{{ data.postalcode }} {{ get_data('state') }}</p>
<p class="text-sm">{{ get_data('country')}}</p>
</div>
<span v-if="!showinfo.address" class="w-4/5 flex-grow" />
<button
v-if="authinfo.viewchange"
class="no-print text-sm flex icon-btn ml-4 mt-0 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('address')"
>
<div v-if="showinfo.address" i-carbon-view-off />
<div v-else class="noprint flex-grow-0 flex -mb-5">
<div class="-mt-0.5 mr-2 line-through">{{ t('address', 'Adress') }}</div>
<div i-carbon-view />
</div>
</button>
</div>
</div>
<div v-if="showinfo.birthdate || showinfo.status" class="panel-group">
<h2 class="h2-title">{{ t('cv.personal', 'Personal') }}</h2>
<div v-if="showinfo.birthdate" class="flex items-center my-3">
<img :src="`${assets_path}/images/assets/egg-outline.svg`" class="inline w-6 mr-4" alt="Egg Icon" />
<span class="text-sm">{{ data.birthdate }}</span>
</div>
<div v-if="data.status && showinfo.status" class="flex items-center my-3">
<img :src="`${assets_path}/images/assets/people-outline.svg`" class="inline w-6 mr-4" alt="Two Persons Icon" />
<span>{{ t(`cv.${data.status.toLowerCase()}`, data.status) }}</span>
</div>
</div>
<div :class="{ 'hidden': !showinfo.sites && !authinfo.viewchange }" class="panel-group -mt-4">
<h2 class="h2-title flex">
<span v-if="showinfo.sites">{{ t('cv.onweb', 'On the Web') }}</span>
<span v-else class="w-5 flex-grow" />
<button
v-if="authinfo.viewchange"
class="no-print text-sm flex icon-btn ml-2 mt-1.5 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('sites')"
>
<div v-if="showinfo.sites" i-carbon-view-off />
<div v-else class="noprint flex-grow-0 flex -mb-5">
<div class="-mt-0.5 mr-2 line-through text-xs">{{ t('cv.onweb', 'On the Web') }}</div>
<div i-carbon-view />
</div>
</button>
</h2>
<div
v-if="showinfo.sites"
v-for="(site,index) in data.sites"
:key="site.id"
class="flex items-center y-3"
:class="{ 'noprint': !site.auth.show }"
>
<div v-if="authinfo.viewchange" class="flex-grow-0 flex noprint">
<span v-if="!showinfo.sites" class="w-4/5 flex-grow" />
<button
class="noprint text-xs mr-1 flex-grow-0 icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onItem('sites', index)"
>
<div v-if="site.auth.show" i-carbon-view-off />
<div v-else class="noprint flex">
<div class="-mt-0.5 mr-2 line-through">{{ index }}</div>
<div i-carbon-view />
</div>
</button>
</div>
<div v-if="site.auth.show" class="flex-grow flex mb-2 text-sm">
<img :src="`${assets_path}${site.img}`" class="flex-grow-0 inline w-6 mr-4" :alt="site.alt" />
<span class="mt-0.5 mr-2 flex-grow-0 text-gray-500 dark:text-gray-700" style="font-size: 80%">{{ site.title }}</span>
<a @click="onLink" :href="site.link" target="_blank" rel="noopener noreferrer" class="flex-grow">
<span class="flex-grow" style="font-size: 80%">{{ site.sub }}</span>
</a>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { PropType } from 'vue'
import { AuthInfoType, DataCoreType, ShowInfoType,DataLangsType } from '~/typs/cv'
import TiptapEditor from '~/components/TiptapEditor.vue'
import useState from '~/hooks/useState'
import { track_action } from '~/hooks/tracking'
const emit = defineEmits(['onEditor', 'onItem'])
const { t } = useI18n()
// const router = useRouter()
const props = defineProps({
data: {
type: Object as PropType<DataCoreType>,
default: {},
required: true,
},
localedata: {
type: Object as PropType<DataLangsType>,
default: {},
required: true,
},
showinfo: {
type: Object as PropType<ShowInfoType>,
default: {},
required: true,
},
authinfo: {
type: Object as PropType<AuthInfoType>,
default: { editable: false, viewchange: false },
required: true,
},
})
const assets_path = useState().ASSETS_PATH.value
const showinfo = ref(props.showinfo as ShowInfoType)
const authinfo = computed(() => {
return showinfo.value.auth ? showinfo.value.auth : props.authinfo
})
const get_data = (itm: string) => {
return props.localedata.value && props.localedata.value.corelang && props.localedata.value.corelang[itm] ? props.localedata.value.corelang[itm] : props.data[itm]
}
const onView = (itm: string) => {
if (typeof showinfo.value[itm] !== 'undefined') {
showinfo.value[itm] = !showinfo.value[itm]
// if (itm === 'skills')
// useState().showinfo.value.skills = showinfo.value[itm]
}
}
const onEditor = (info: { src: string, field: string, idx: number, data: string, ev: Event }) => {
let has_changed = false
let arr_data: string[] = []
let data = ''
const arr_src = info.field.split('.')
switch (arr_src[0]) {
case 'mission_how':
const field = arr_src.length > 0 ? arr_src[1] : info.field
arr_data = info.data.replace('<ul>', '').replace('</ul>', '').replace(/<li.+?>/g, '').split('</li>')
data = get_data(arr_src[0])[info.idx][field]
break
default:
has_changed = info.data.replace('<p>', '').replace('</p>', '') !== get_data(info.field)
}
if (arr_data.length > 0) {
arr_data = arr_data.filter(it => it !== '')
arr_data.forEach((it, idx) => {
if (!has_changed && it.replace('<p>', '').replace('</p>', '') !== data ) {
has_changed = true
}
})
}
if (has_changed) {
emit('onEditor', { ...info, arr_data })
}
}
// const onEditor = (info: { src: string, idx: number, data: string, ev: Event }) => {
// emit('onEditor', { ...info })
// }
const onItem = (src: string, idx: number) => {
emit('onItem', { root: 'info', src, itm: props.data[src][idx], idx })
}
const onLink = (e: any) => {
track_action(e)
}
</script>

View File

@ -0,0 +1,227 @@
<template>
<div
id="info-skills"
:class="{ 'hidden': !showinfo.skills && !authinfo.viewchange }"
class="panel-group -mt-2"
>
<h2 class="h2-title flex">
<span v-if="showinfo.skills">{{ t('cv.skills_tools', 'Skill & Tools') }}</span>
<div v-if="authinfo.viewchange" class="flex-grow-0 flex noprint ml-2">
<span v-if="!showinfo.skills" class="w-4/5 flex-grow" />
<button
class="no-print text-base flex-grow-0 icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('skills')"
>
<div v-if="showinfo.skills" class="flex" i-carbon-view-off />
<div v-else class="noprint flex">
<div class="mr-2 text-sm line-through">{{ t('cv.skills_tools', 'Skill & Tools').replaceAll(' ','') }}</div>
<div class="flex-grow-0" i-carbon-view />
</div>
</button>
</div>
</h2>
<skills-view
v-if="showinfo.skills"
src="soft_skills"
:data="data.skills"
:localedata="localedata"
:showinfo="showinfo"
:authinfo="authinfo"
@onItem="onSkills"
/>
</div>
<div
id="info-soft_skills"
:class="{ 'hidden': !showinfo.soft_skills && !authinfo.viewchange }"
class="panel-group -mt-2"
>
<h2 class="h2-title flex">
<span v-if="showinfo.soft_skills">{{ t('cv.soft_skills', 'Soft Skills') }}</span>
<div v-if="authinfo.viewchange" class="flex-grow-0 flex noprint ml-2">
<span v-if="!showinfo.soft_skills" class="w-4/5 flex-grow" />
<button
class="no-print text-base flex-grow-0 icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('soft_skills')"
>
<div v-if="showinfo.soft_skills" class="flex" i-carbon-view-off />
<div v-else class="noprint flex">
<div class="mr-2 text-sm line-through">{{ t('cv.soft_skills', 'Soft Skills').replaceAll(' ','') }}</div>
<div class="flex-grow-0" i-carbon-view />
</div>
</button>
</div>
</h2>
<skills-view
v-if="showinfo.soft_skills"
src="soft_skills"
:data="data.soft_skills"
:localedata="localedata"
:showinfo="showinfo"
:authinfo="authinfo"
@onItem="onSoftSkills"
/>
</div>
<div :class="{ 'hidden': !showinfo.infra && !authinfo.viewchange }" class="panel-group">
<h2 class="h2-title flex">
<span v-if="showinfo.infra">{{ t('cv.infrastructures', 'Infrastructures') }}</span>
<button
v-if="authinfo.viewchange"
class="no-print text-sm float-right icon-btn ml-2 mt-1 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('infra')"
>
<div v-if="showinfo.infra" i-carbon-view-off />
<div v-else class="noprint flex flex-grow-0">
<div class="-mt-0.5 mr-2 line-through">{{ t('cv.infrastructures', 'Infrastructures') }}</div>
<div class="flex-grow-0" i-carbon-view />
</div>
</button>
</h2>
<div v-if="showinfo.infra" class="-ml-5 mt-2 leading-8 text-sm grid grid-flow-col grid-cols-2 grid-rows-2 gap-2">
<div
v-for="infra in data.infra"
:key="infra.id"
class="border-1 border-indigo-400 rounded-xl bg-gray-300 w-25 px-3 ml-2"
>{{ infra.title }}</div>
</div>
</div>
<div :class="{ 'hidden': !showinfo.certs && !authinfo.viewchange }" class="panel-group -mt-4">
<h2 class="h2-title flex">
<span v-if="showinfo.certs">{{ t('cv.certifications', 'Certifications') }}</span>
<button
v-if="authinfo.viewchange"
class="no-print text-sm float-right icon-btn ml-2 mt-1 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('certs')"
>
<div v-if="showinfo.certs" i-carbon-view-off />
<div v-else class="noprint flex flex-grow-0">
<div class="-mt-0.5 mr-2 line-through">{{ t('cv.certifications', 'Certifications') }}</div>
<div class="flex-grow-0" i-carbon-view />
</div>
</button>
</h2>
<div
v-if="showinfo.certs"
v-for="(cert,index) in get_data('certifications')"
:key="cert.id"
class="border-b-1 border-gray-400 mb-3 pb-2 flex"
>
<div v-if="cert.auth.show" class="flex-grow">
<h3 class="font-semibold text-sm">{{ cert.title }}</h3>
<div class="font-light text-xs">{{ cert.date }}</div>
<div class="font-light text-xs flex flex-col">
<div v-if="cert.href && cert.href !== ''" class="flex-grow">
<small class="hidden">{{ t('link', 'Link') }}: </small>
<a @click="onLink" :href="cert.href">{{ cert.link }}</a>
</div>
<div v-if="cert.certid && cert.certid !== ''" class="flex-grow-0">
<small>{{ t('id', 'Id') }}: {{ cert.certid }}</small>
</div>
</div>
<p v-if="cert.author && cert.author !== ''" class="font-light text-xs mb-2">
<small class="hidden">{{ t('from', 'From') }}: </small>
<span class="font-semibold">
<i>{{ cert.author }}</i>
</span>
</p>
<p v-else class="mb-2" />
</div>
<div class="flex flex-grow-0">
<button
v-if="authinfo.viewchange"
class="noprint flex-grow-0 icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onItem('certifications', index)"
>
<div v-if="cert.auth.show" class="mr-1" i-carbon-view-off />
<div v-else class="noprint flex flex-row">
<div class="mt-0.2 mr-2 line-through text-xs">{{ index }}</div>
<div i-carbon-view />
</div>
</button>
</div>
</div>
</div>
<div :class="{ 'hidden': !showinfo.langs && !authinfo.viewchange }" class="panel-group -mt-4">
<h2 class="h2-title">
<span v-if="showinfo.langs">{{ t('cv.languages', 'Languages') }}</span>
<span v-else class="w-5 flex-grow" />
<button
v-if="authinfo.viewchange"
class="no-print text-sm float-right icon-btn mt-2 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onView('langs')"
>
<div v-if="showinfo.langs" i-carbon-view-off />
<div v-else class="noprint flex-grow-0 flex -mb-5">
<div class="-mt-0.5 mr-2 line-through text-xs">{{ t('cv.languages', 'Languages') }}</div>
<div i-carbon-view />
</div>
</button>
</h2>
<div v-if='showinfo.langs'
v-for="lang in data.langs" :key="lang.id" class="flex flex-row">
<div class="font-semibold flex-grow">{{ t(`lang.${lang.title.toLowerCase()}`, lang.title) }}</div>
<div class="flex-grow-0 text-sm mt-1">{{ t(`lang.${lang.mode.toLowerCase()}`, lang.mode) }}</div>
</div>
</div>
</template>
<script setup lang="ts">
import { PropType } from 'vue'
import { AuthInfoType, DataCoreType, ShowInfoType,DataLangsType } from '~/typs/cv'
import SkillsView from '~/views/cv/Skills.vue'
import useState from '~/hooks/useState'
import { track_action } from '~/hooks/tracking'
const emit = defineEmits(['onEditor', 'onItem'])
const { t } = useI18n()
// const router = useRouter()
const props = defineProps({
data: {
type: Object as PropType<DataCoreType>,
default: {},
required: true,
},
localedata: {
type: Object as PropType<DataLangsType>,
default: {},
required: true,
},
showinfo: {
type: Object as PropType<ShowInfoType>,
default: {},
required: true,
},
authinfo: {
type: Object as PropType<AuthInfoType>,
default: { editable: false, viewchange: false },
required: true,
},
})
// const assets_path = useState().ASSETS_PATH.value
const showinfo = ref(props.showinfo as ShowInfoType)
const authinfo = computed(() => {
return showinfo.value.auth ? showinfo.value.auth : props.authinfo
})
const get_data = (itm: string) => {
return props.localedata.value && props.localedata.value.corelang && props.localedata.value.corelang[itm] ? props.localedata.value.corelang[itm] : props.data[itm]
}
const onView = (itm: string) => {
if (typeof showinfo.value[itm] !== 'undefined') {
showinfo.value[itm] = !showinfo.value[itm]
if (itm === 'skills')
useState().showinfo.value.skills = showinfo.value[itm]
else if (itm === 'soft_skills')
useState().showinfo.value.soft_skills = showinfo.value[itm]
}
}
const onItem = (src: string, idx: number) => {
emit('onItem', { root: 'info', src, itm: props.data[src][idx], idx })
}
const onSkills = (data: { src: string, idx: number }) => {
emit('onItem', { root: 'info', src: data.src, itm: props.data[data.src][data.idx], idx: data.idx })
}
const onSoftSkills = (data: { src: string, idx: number }) => {
emit('onItem', { root: 'info', src: data.src, itm: props.data[data.src][data.idx], idx: data.idx })
}
const onLink = (e: any) => {
track_action(e)
}
</script>

View File

@ -3,7 +3,7 @@
<li v-for="(skill,index) in get_data('skills')" :key="skill.id" :id="`cv-${skill.id}`" class="flex">
<div class="flex-grow">
<label :for="skill.id" class="flex flex-row">
<span v-if="skill.auth.show" class="flex-grow text-base">{{ skill.title }}</span>
<span v-if="skill.auth.show" class="flex-grow text-sm">{{ skill.title }}</span>
<span v-if="skill.auth.show" class="mt-1 text-sm flex-grow-0">{{ skill.value }}%</span>
</label>
<progress
@ -35,6 +35,11 @@
import { PropType } from 'vue'
import { AuthInfoType, SkillsType, DataLangsType } from '~/typs/cv'
const props = defineProps({
src: {
type: String,
default: 'skill',
required: true,
},
data: {
type: Array as PropType<SkillsType[]>,
default: [],
@ -56,7 +61,7 @@ const get_data = (itm: string) => {
return props.localedata.value && props.localedata.value.corelang && props.localedata.value.corelang[itm] ? props.localedata.value.corelang[itm] : props.data
}
const onItem = (idx: number) => {
emit('onItem', { src: 'skills', itm: props.data[idx], idx })
emit('onItem', { src: props.src, itm: props.data[idx], idx })
}
// const onItem = (itm: string) => {
// const dom_id = document.getElementById(`cv-${itm}`)