detail.vue 7.92 KB
<template>
  <view class="page">
    <scroll-view class="scroll" scroll-y>
      <view class="detail-page">
        <view class="section">
          <text class="row company">{{ form.customerName }}</text>
          <view :class="['status', `status_${form.status}`]" />
          <view class="row"><text class="label">发货日期</text><text class="value">{{ form.shipmentsDate }}</text></view>
          <view class="row"><text class="label">生产厂</text><text class="value">{{ form.workshopName }}</text></view>
          <view class="row"><text class="label">办事处</text><text class="value">{{ form.deptName }}</text></view>
          <view class="row"><text class="label">区域</text><text class="value">{{ form.regionName }}</text></view>
          <view class="row"><text class="label">交货方式</text><text class="value">{{ form.deliveryType }}</text></view>
          <view class="row"><text class="label">目的地</text><text class="value">{{ form.destination }}</text></view>
        </view>

        <!-- 产品 -->
        <view class="section2">
          <Product mode="view" :list="form.detailList" />
        </view>
        <view class="section">
          <view class="row"><text class="label">签收单据</text><text class="value act">{{ form.fileName }}</text></view>
        </view>
      </view>
    </scroll-view>
    <detail-buttons :buttons="displayButtons" @click="handleButtonClick" />
    <uni-popup ref="uploadPopup" type="bottom" :mask-click="false">
			<view class="dialog">
				<view class="dialog_header">
					<text>上传签收单据</text>
					<view class="dialog_close" @click="closeUploadInfo">×</view>
				</view>
				<view class="dialog_body">
					<view class="dialog_row">
						<text class="dialog_label">签收单据</text>
						<FileUpload v-model="uploadFile" />
					</view>
				</view>
				<view class="dialog_footer">
					<button class="btn confirm" type="primary" @click="onUploadSave">保存</button>
				</view>
			</view>
		</uni-popup>
  </view>
</template>

<script>
import { getDetailApi, saveSignInTicket } from '@/api/invoice.js'
import Product from './product.vue'
import DetailButtons from '@/components/detail-buttons/index.vue'
import FileUpload from '@/components/file-upload/index.vue'

export default {
  name: 'InvoiceDetail',
  components: { Product, DetailButtons, FileUpload },
  data() {
    return {
      form: {},
      buttons: [
        { text: '编辑', visible: true, variant: 'outline', event: 'edit' },
        { text: '填写实发数', visible: true, variant: 'outline', event: 'fill' },
        { text: '上传签收单据', visible: true, variant: 'primary', event: 'upload' },
      ],
			uploadFile: { id: '', name: '' },
    }
  },
  computed: {
    statusFlags() {
      const m = this.form || {}
      const e = String(m.status || '')
      return {
        canEdit: e === 'UN_SHIPMENTS',
        canFill: e === 'UN_SHIPMENTS',
        canUpload: e === 'SHIPMENTS',
      }
    },
    displayButtons() {
      const f = this.statusFlags
      return [
        { ...this.buttons[0], visible: f.canEdit && this.$auth.hasPermi('shipping-plan-manage:invoice:modify') },
        { ...this.buttons[1], visible: f.canFill && this.$auth.hasPermi('shipping-plan-manage:invoice:fill') },
        { ...this.buttons[2], visible: f.canUpload && this.$auth.hasPermi('shipping-plan-manage:invoice:upload') },
      ]
    }
  },
  onLoad(query) {
    const id = (query && (query.id || query.code)) || ''
    if (id) this.loadDetail(id)
  },
  methods: {
    async loadDetail(id) {
      try {
        const res = await getDetailApi(id)
        this.form = res.data || {}
      } catch (e) {
        this.form = {}
      }
    },
    handleButtonClick(btn) {
      if (!btn || btn.disabled) return
      const map = {
        edit: () => this.onEdit(),
        fill: () => this.onFill(),
        upload: () => this.onUpload(),
      }
      const fn = map[btn.event]
      if (typeof fn === 'function') fn()
    },
    onEdit() {
      // const id = this.form.id || this.form.code
      // if (id) uni.navigateTo({ url: `/pages/invoice/modify?id=${id}` })
    },
    onFill() {
      const id = this.form.id || this.form.code
      if (id) uni.navigateTo({ url: `/pages/invoice/fill?id=${id}` })
    },
    onUpload() {
			this.$refs.uploadPopup && this.$refs.uploadPopup.open()
    },
    closeUploadInfo() {
      this.uploadFile = { id: '', name: '' };
			this.$refs.uploadPopup && this.$refs.uploadPopup.close()
		},
    async onUploadSave() {
      console.log('onUploadSave__this.uploadFile', this.uploadFile)
      if (!this.uploadFile.id) {
        uni.showToast({
						title: '请上传签收单据',
						icon: 'none'
					})
        return
      }
      const params = {
        fileId: this.uploadFile.id,
        fileName: this.uploadFile.name,
        shipmentsOrderId: this.form.id
      }
      try {
        await saveSignInTicket(params)
        uni.showToast({ title: '保存成功', icon: 'success' })
        setTimeout(() => { uni.redirectTo({ url: '/pages/invoice/index' }) }, 300)
      } catch (e) {
        uni.showToast({ title: (e && e.msg) || '保存失败', icon: 'none' })
      }
    }
  }
}
</script>

<style lang="scss" scoped>
.page {
  display: flex;
  flex-direction: column;
  height: 100vh;
}

.scroll {
  flex: 1;
  background: #f3f3f3;
}

.detail-page {
  padding-bottom: 150rpx;
}

.section {
  padding: 32rpx;
  background: #fff;
  margin-bottom: 20rpx;
  position: relative;

  .status {
    position: absolute;
    top: 16rpx;
    right: 52rpx;
    width: 180rpx;
    height: 146rpx;
    background-repeat: no-repeat;
    background-size: 100% 100%;
    background-position: center;

    // 已签收
    &_DELIVERED {
      background-image: url('~@/static/images/invoice/status_DELIVERED.png');
    }

    // 已发货
    &_SHIPMENTS {
      background-image: url('~@/static/images/invoice/status_SHIPMENTS.png');
    }

    // 未发货
    &_UN_SHIPMENTS {
      background-image: url('~@/static/images/invoice/status_UN_SHIPMENTS.png');
    }


  }
}

.row {
  display: flex;
  margin-bottom: 28rpx;

  &:last-child {
    margin-bottom: 0;
  }

  &.company {
    font-size: 36rpx;
    font-weight: 600;
    color: rgba(0, 0, 0, 0.9);
    padding-top: 10rpx;
    margin-bottom: 32rpx;
    line-height: 50rpx;
  }

  .label {
    width: 240rpx;
    line-height: 32rpx;
    font-size: 28rpx;
    color: rgba(0, 0, 0, 0.6);
  }

  .value {
    flex: 1;
    line-height: 32rpx;
    font-size: 28rpx;
    color: rgba(0, 0, 0, 0.9);
    text-align: right;
    word-break: break-all;
    &.act {
      color: $theme-primary;
    } 

  }
}

.title-header {
  background-color: #fff;
  display: flex;
  align-items: center;
  padding: 32rpx 32rpx 22rpx;
  border-bottom: 1rpx dashed #f0f0f0;

  &_icon {
    width: 32rpx;
    height: 28rpx;
    margin-right: 16rpx;
  }

  span {
    color: rgba(0, 0, 0, 0.9);
    font-size: 32rpx;
    line-height: 44rpx;
    font-weight: 600;
  }
}


.dialog {
	width: 100%;
	max-height: 70vh;
	overflow-y: auto;
	padding: 32rpx 28rpx calc(20rpx + env(safe-area-inset-bottom));
	background: #fff;
	border-radius: 20rpx 20rpx 0 0;

	&_header {
		position: relative;
		text-align: center;
		font-size: 34rpx;
		font-weight: 600;
		margin-bottom: 12rpx;
		color: rgba(0, 0, 0, 0.9);
	}

	&_body {
		padding: 12rpx 4rpx 24rpx;
	}

	&_footer {
		padding-top: 12rpx;
		display: flex;
		justify-content: center;

		.btn {
			width: 100%;
			height: 80rpx;
			line-height: 80rpx;
			border-radius: 12rpx;
			font-size: 32rpx;
			background: $theme-primary;
			color: #fff;

			&::after {
				border: none;
			}
		}
	}
}

.dialog_close {
	position: absolute;
	right: 16rpx;
	top: -14rpx;
	width: 64rpx;
	height: 64rpx;
	line-height: 64rpx;
	text-align: center;
	font-size: 36rpx;
	color: rgba(0, 0, 0, 0.6);
}

.dialog_row {
	margin-bottom: 24rpx;
}

.dialog_label {
	display: block;
	margin-bottom: 12rpx;
	font-size: 28rpx;
	color: rgba(0, 0, 0, 0.9);
}

.upload-show {
	margin-top: 8rpx;
	font-size: 26rpx;
	color: rgba(0, 0, 0, 0.6);
}

</style>