123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- <template>
- <div
- class="courserware"
- :style="[
- {
- backgroundImage: data.background_image_url ? `url(${data.background_image_url})` : '',
- backgroundSize: data.background_image_url
- ? `${data.background_position.width}% ${data.background_position.height}%`
- : '',
- backgroundPosition: data.background_image_url
- ? `${data.background_position.left}% ${data.background_position.top}%`
- : '',
- },
- ]"
- >
- <template v-for="(row, i) in data.row_list">
- <div :key="i" class="row" :style="getMultipleColStyle(i)">
- <!-- 列 -->
- <template v-for="(col, j) in row.col_list">
- <div :key="j" :class="['col', `col-${i}-${j}`]" :style="computedColStyle(col)">
- <!-- 网格 -->
- <template v-for="(grid, k) in col.grid_list">
- <component
- :is="previewComponentList[grid.type]"
- :id="grid.id"
- ref="preview"
- :key="k"
- :courseware-id="data.courseware_id"
- :class="[grid.id]"
- :style="{
- gridArea: grid.grid_area,
- height: grid.height,
- }"
- />
- </template>
- </div>
- </template>
- </div>
- </template>
- </div>
- </template>
- <script>
- import { previewComponentList } from '@/views/book/courseware/data/bookType';
- import { GetBook } from '@/api/book';
- export default {
- name: 'CoursewarePreview',
- provide() {
- return {
- getDragStatus: () => false,
- bookInfo: this.bookInfo,
- };
- },
- props: {
- data: {
- type: Object,
- default: () => ({}),
- },
- },
- data() {
- return {
- previewComponentList,
- bookInfo: {
- theme_color: '',
- },
- };
- },
- created() {
- GetBook({ id: this.$route.query.book_id }).then(({ theme_color }) => {
- this.bookInfo.theme_color = theme_color;
- });
- },
- methods: {
- getMultipleColStyle(i) {
- let row = this.data.row_list[i];
- let col = row.col_list;
- if (col.length <= 1) {
- return {
- gridTemplateColumns: '100fr',
- };
- }
- let gridTemplateColumns = row.width_list.join(' ');
- return {
- gridAutoFlow: 'column',
- gridTemplateColumns,
- gridTemplateRows: 'auto',
- };
- },
- /**
- * 分割整数为多个 1的倍数
- * @param {number} num
- * @param {number} parts
- */
- splitInteger(num, parts) {
- let base = Math.floor(num / parts);
- let arr = Array(parts).fill(base);
- let remainder = num - base * parts;
- for (let i = 0; remainder > 0; i = (i + 1) % parts) {
- arr[i] += 1;
- remainder -= 1;
- }
- return arr;
- },
- computedColStyle(col) {
- const grid = col.grid_list;
- let maxCol = 0; // 最大列数
- let rowList = new Map();
- grid.forEach(({ row }) => {
- rowList.set(row, (rowList.get(row) || 0) + 1);
- });
- let curMaxRow = 0; // 当前数量最大 row 的值
- rowList.forEach((value, key) => {
- if (value > maxCol) {
- maxCol = value;
- curMaxRow = key;
- }
- });
- // 计算 grid_template_areas
- let gridTemplateAreas = '';
- let gridArr = [];
- grid.forEach(({ grid_area, row }) => {
- if (!gridArr[row - 1]) {
- gridArr[row - 1] = [];
- }
- if (curMaxRow === row) {
- gridArr[row - 1].push(`${grid_area}`);
- } else {
- let filter = grid.filter((item) => item.row === row);
- let find = filter.findIndex((item) => item.grid_area === grid_area);
- let needNum = maxCol - filter.length; // 需要的数量
- let str = '';
- if (filter.length === 1) {
- str = ` ${grid_area} `.repeat(needNum + 1);
- } else {
- let arr = this.splitInteger(needNum, filter.length);
- str = arr[find] === 0 ? ` ${grid_area} ` : ` ${grid_area} `.repeat(arr[find] + 1);
- }
- gridArr[row - 1].push(`${str}`);
- }
- });
- gridArr.forEach((item) => {
- gridTemplateAreas += `'${item.join(' ')}' `;
- });
- // 计算 grid_template_columns
- let gridTemplateColumns = '';
- let max = { row: 0, num: 0 };
- grid.forEach(({ row }) => {
- // 计算出 row 的哪个值最多
- let len = grid.filter((item) => item.row === row).length;
- if (max.num < len) {
- max.num = len;
- max.row = row;
- }
- });
- grid.forEach((item) => {
- if (item.row === max.row) {
- gridTemplateColumns += `${item.width} `;
- }
- });
- // 计算 grid_template_rows
- let gridTemplateRows = '';
- // 将 grid 按照 row 分组
- let gridMap = new Map();
- grid.forEach((item) => {
- if (!gridMap.has(item.row)) {
- gridMap.set(item.row, []);
- }
- gridMap.get(item.row).push(item.height);
- });
- gridMap.forEach((value) => {
- if (value.length === 1) {
- gridTemplateRows += `${value[0]} `;
- } else {
- let isAllAuto = value.every((item) => item === 'auto'); // 是否全是 auto
- gridTemplateRows += isAllAuto ? 'auto ' : `max(${value.join(', ')}) `;
- }
- });
- return {
- width: col.width,
- gridTemplateAreas,
- gridTemplateColumns,
- gridTemplateRows,
- };
- },
- },
- };
- </script>
- <style lang="scss" scoped>
- .courserware {
- display: flex;
- flex-direction: column;
- row-gap: 6px;
- width: $courseware-width;
- min-height: 500px;
- padding: 24px;
- background-color: #fff;
- background-repeat: no-repeat;
- border-bottom-right-radius: 12px;
- border-bottom-left-radius: 12px;
- .row {
- display: grid;
- gap: 16px;
- .col {
- display: grid;
- gap: 16px;
- overflow: hidden;
- }
- }
- }
- </style>
|