Переглянути джерело

feat: 搜索so单生成标签打印pdf页面.

peter 3 місяців тому
батько
коміт
1c88aee046

+ 1 - 0
index.html

@@ -5,6 +5,7 @@
     <meta http-equiv="X-UA-Compatible" content="IE=edge">
     <meta name="viewport" content="width=device-width,initial-scale=1.0">
     <meta name="keywords" content="zoho crm extend">
+    <script src="/crm.sdk.js"></script>
    <title>crm_extend</title>
   </head>
   <body>

+ 3 - 1
package.json

@@ -13,7 +13,7 @@
     "@element-plus/icons-vue": "^2.1.0",
     "axios": "~0.27.2",
     "dayjs": "^1.11.13",
-    "element-plus": "2.8.8",
+    "element-plus": "2.9.2",
     "html2canvas": "^1.4.1",
     "js-cookie": "^3.0.5",
     "jspdf": "^2.5.2",
@@ -21,6 +21,7 @@
     "lodash.clonedeep": "^4.5.0",
     "lodash.debounce": "^4.0.8",
     "mathjs": "^13.2.3",
+    "qrcode": "^1.5.4",
     "vue": "^3.5.13",
     "vue-draggable-plus": "^0.6.0",
     "vue-router": "^4.4.5",
@@ -31,6 +32,7 @@
     "@types/js-cookie": "^3.0.6",
     "@types/lodash.clonedeep": "^4.5.9",
     "@types/lodash.debounce": "^4.0.9",
+    "@types/qrcode": "^1.5.5",
     "@typescript-eslint/eslint-plugin": "^7.18.0",
     "@typescript-eslint/parser": "^7.18.0",
     "@vitejs/plugin-vue": "^5.2.0",

BIN
public/assets/label/care.png


BIN
public/assets/label/glass.png


BIN
public/assets/label/heavy.png


BIN
public/assets/label/keep_dry.png


BIN
public/assets/label/place_up.png


+ 852 - 0
public/crm.sdk.js

@@ -0,0 +1,852 @@
+var ZOHO = (function () {
+  var c,
+    m = {},
+    h = !1,
+    n = void 0
+  return {
+    embeddedApp: {
+      on: function (b, c) {
+        m[b] = c
+      },
+      init: function () {
+        if (!h) {
+          h = !0
+          c = new ZSDK()
+          var b
+          n = new Promise(function (c, n) {
+            b = c
+          })
+          c.OnLoad(function () {
+            c.getContext()
+              .Event.Trigger('ZDK_EVENT', { action: 'get_zdk_url' }, !0)
+              .then(function (k) {
+                self._zdksdk = c
+                var n = document.createElement('script')
+                n.setAttribute('src', k)
+                document.body.appendChild(n)
+                b()
+              })
+              .catch(function (c) {
+                console.error('ZDK import failed', c)
+                b()
+              })
+          })
+          for (var t in m) c.getContext().Event.Listen(t, m[t])
+        }
+        return n
+      },
+    },
+    CRM: (function () {
+      function b(a) {
+        a.sdkVersion = '1'
+        return c.getContext().Event.Trigger('CRM_EVENT', a, !0)
+      }
+      function n(a) {
+        return new File([a], a.name, { type: a.type })
+      }
+      function k(a, d, e, f) {
+        if (d.FileData) {
+          var c = n(d.FileData)
+          d.FileData = c
+        }
+        a = { category: 'CREATE', Entity: a, RelatedID: e, APIData: d }
+        a.type = f || 'RECORD'
+        return b(a)
+      }
+      function m(a) {
+        a.category = 'BLUEPRINT'
+        return b(a)
+      }
+      function h(a) {
+        a.category = 'APPROVALS'
+        return b(a)
+      }
+      function p(a, d, e) {
+        if (d.FILE) {
+          var f = n(d.FILE.file)
+          d.FILE.file = f
+        }
+        f = void 0
+        if (e) f = d
+        else {
+          var f = d.url,
+            c = d.params,
+            g = d.headers,
+            k = d.body,
+            h = d.PARTS,
+            m = d.PART_BOUNDARY,
+            w = d.CONTENT_TYPE,
+            D = d.RESPONSE_TYPE
+          d = d.FILE
+          if (!f) throw { Message: 'Url missing' }
+          if (c) {
+            var x,
+              l = []
+            for (x in c)
+              l.push(encodeURIComponent(x) + '\x3d' + encodeURIComponent(c[x]))
+            x = l.join('\x26')
+            f += (-1 < f.indexOf('?') ? '\x26' : '?') + x
+          }
+          f = {
+            url: f,
+            Header: g,
+            Body: k,
+            CONTENT_TYPE: w,
+            RESPONSE_TYPE: D,
+            PARTS: h,
+            PARTS_BOUNDARY: m,
+            FILE: d,
+          }
+        }
+        return b({ category: 'CONNECTOR', nameSpace: a, data: f, type: e })
+      }
+      function g(a) {
+        a.category = 'UI'
+        return b(a)
+      }
+      function r(a, d, e) {
+        return b({ category: 'CONFIG', type: a, nameSpace: d, APIData: e })
+      }
+      function u(a) {
+        var d = { category: 'USER' }
+        a.ID
+          ? (d.ID = a.ID)
+          : a.Type &&
+            ((d.Type = a.Type),
+            a.page && (d.page = a.page),
+            a.per_page && (d.per_page = a.per_page))
+        return b(d)
+      }
+      function l(a) {
+        return b({ category: 'META', type: a.type, Entity: a.Entity, Id: a.Id })
+      }
+      return {
+        ACTION: {
+          setConfig: function (a) {
+            return b({
+              category: 'ACTION',
+              type: 'CUSTOM_ACTION_SAVE_CONFIG',
+              object: a,
+            })
+          },
+          enableAccountAccess: function (a) {
+            return b({
+              category: 'ACTION',
+              type: 'ENABLE_ACCOUNT_ACCESS',
+              object: a,
+            })
+          },
+        },
+        FUNCTIONS: {
+          execute: function (a, d) {
+            var e = {}
+            d.auth_type = 'oauth'
+            e.data = d
+            return b({
+              category: 'FUNCTIONS_EXECUTE',
+              customFunctionName: a,
+              data: e,
+            })
+          },
+        },
+        CONFIG: {
+          getOrgInfo: function (a) {
+            return r('ORG')
+          },
+          getCurrentUser: function () {
+            return r('CURRENT_USER')
+          },
+          GetCurrentEnvironment: function () {
+            return r('ORG_LEVEL_INFO')
+          },
+        },
+        META: {
+          getFields: function (a) {
+            a.type = 'FIELD_LIST'
+            return l(a)
+          },
+          getModules: function () {
+            return l({ type: 'MODULE_LIST' })
+          },
+          getAssignmentRules: function (a) {
+            a.type = 'ASSIGNMENT_RULES'
+            return l(a)
+          },
+          getLayouts: function (a) {
+            a.id = a.id ? a.id : a.LayoutId
+            a.type = a.Id ? 'LAYOUT' : 'LAYOUTS'
+            return l(a)
+          },
+          getRelatedList: function (a) {
+            a.type = 'RELATED_LIST'
+            return l(a)
+          },
+          getCustomViews: function (a) {
+            a.type = a.Id ? 'CUSTOM_VIEW' : 'CUSTOM_VIEWS'
+            return l(a)
+          },
+        },
+        API: {
+          addNotes: function (a) {
+            return k(
+              a.Entity,
+              { data: [{ Note_Title: a.Title, Note_Content: a.Content }] },
+              a.RecordID,
+              'NOTES',
+            )
+          },
+          addNotesAttachment: function (a) {
+            return b({
+              category: 'UPDATE',
+              type: 'NOTES',
+              Entity: a.Entity,
+              RecordID: a.RecordID,
+              RelatedRecordID: a.RelatedRecordID,
+              APIData: {
+                Files: { FileName: File.Name, FileData: File.Content },
+              },
+            })
+          },
+          coql: function (a) {
+            return b({ category: 'QUERY', APIData: a })
+          },
+          insertRecord: function (a) {
+            var b = a.Entity,
+              e = a.APIData
+            e.trigger = a.Trigger
+            return k(b, e)
+          },
+          upsertRecord: function (a) {
+            var b = a.Entity,
+              e = a.APIData
+            e.trigger = a.Trigger
+            e.action = 'UPSERT'
+            a.duplicate_check_fields &&
+              a.duplicate_check_fields instanceof Array &&
+              (e.duplicate_check_fields = a.duplicate_check_fields.join(','))
+            return k(b, e)
+          },
+          getRecord: function (a) {
+            return b({ category: 'READ', APIData: a })
+          },
+          getBluePrint: function (a) {
+            return m({
+              Entity: a.Entity,
+              RecordID: a.RecordID,
+              action: 'GET_BLUEPRINT_STATUS',
+            })
+          },
+          updateBluePrint: function (a) {
+            return m({
+              Entity: a.Entity,
+              RecordID: a.RecordID,
+              BlueprintData: a.BlueprintData,
+              action: 'UPDATE_BLUEPRINT_STATUS',
+            })
+          },
+          uploadFile: function (a) {
+            if (a.FILE) {
+              var d = n(a.FILE.file)
+              a.FILE.file = d
+            }
+            return b({ FileData: a, category: 'FILES', type: 'UPLOAD_FILE' })
+          },
+          getFile: function (a) {
+            a.category = 'FILES'
+            a.type = 'DOWNLOAD_FILE'
+            return b(a)
+          },
+          getAllRecords: function (a) {
+            return b({ category: 'READ', APIData: a })
+          },
+          updateRecord: function (a) {
+            var d = a.Entity,
+              e = a.APIData
+            e.trigger = a.Trigger
+            return b({
+              category: 'UPDATE',
+              type: 'RECORD',
+              Entity: d,
+              APIData: e,
+            })
+          },
+          deleteRecord: function (a) {
+            return b({
+              category: 'DELETE',
+              type: 'RECORD',
+              Entity: a.Entity,
+              RecordID: a.RecordID,
+            })
+          },
+          searchRecord: function (a) {
+            return b({
+              category: 'SEARCH',
+              Entity: a.Entity,
+              Type: a.Type,
+              Query: a.Query,
+              page: a.page,
+              per_page: a.per_page,
+              delay: a.delay,
+            })
+          },
+          getAllActions: function (a) {
+            a.action = 'GET_ALL_ACTIONS'
+            return h(a)
+          },
+          getApprovalRecords: function (a) {
+            var b = {}
+            a
+              ? (a.action = 'GET_APPROVAL_RECORDS')
+              : ((b.action = 'GET_APPROVAL_RECORDS'), (a = b))
+            return h(a)
+          },
+          getApprovalById: function (a) {
+            a.action = 'GET_APPROVALBYID'
+            return h(a)
+          },
+          getApprovalsHistory: function () {
+            return h({ action: 'GET_APPROVALS_HISTORY' })
+          },
+          approveRecord: function (a) {
+            a.action = 'UPDATE_APPROVAL'
+            return h(a)
+          },
+          getAllUsers: function (a) {
+            return u({ Type: a.Type, page: a.page, per_page: a.per_page })
+          },
+          getUser: function (a) {
+            return u({ ID: a.ID })
+          },
+          getRelatedRecords: function (a) {
+            return b({ category: 'READ', APIData: a })
+          },
+          updateRelatedRecords: function (a) {
+            return b({
+              category: 'UPDATE',
+              type: 'RELATED_RECORD',
+              Entity: a.Entity,
+              RecordID: a.RecordID,
+              RelatedList: a.RelatedList,
+              RelatedRecordID: a.RelatedRecordID,
+              APIData: a.APIData,
+            })
+          },
+          delinkRelatedRecord: function (a) {
+            return b({
+              category: 'DELETE',
+              type: 'RELATED_RECORD',
+              Entity: a.Entity,
+              RecordID: a.RecordID,
+              RelatedList: a.RelatedList,
+              RelatedRecordID: a.RelatedRecordID,
+            })
+          },
+          attachFile: function (a) {
+            var b = a.Entity,
+              e = a.RecordID
+            a = a.File
+            a = { FileName: a.Name, FileData: a.Content }
+            return k(b, a, e, 'ATTACHMENT')
+          },
+          getAllProfiles: function (a) {
+            return b({ category: 'PROFILES', type: 'GET_ALL_PROFILES' })
+          },
+          getProfile: function (a) {
+            return b({ category: 'PROFILES', type: 'GET_PROFILE', ID: a.ID })
+          },
+          updateProfile: function (a) {
+            return b({
+              category: 'UPDATE',
+              type: 'PROFILE',
+              ID: a.ID,
+              APIData: a.APIData,
+            })
+          },
+          getOrgVariable: function (a) {
+            return r('VARIABLE', a)
+          },
+        },
+        UI: {
+          Resize: function (a) {
+            a = { action: 'RESIZE', data: { width: a.width, height: a.height } }
+            return g(a)
+          },
+          Dialer: {
+            maximize: function () {
+              return g({ action: { telephony: 'MAXIMIZE' } })
+            },
+            minimize: function () {
+              return g({ action: { telephony: 'MINIMIZE' } })
+            },
+            notify: function () {
+              return g({ action: { telephony: 'NOTIFY' } })
+            },
+          },
+          Record: {
+            open: function (a) {
+              a = {
+                action: { record: 'OPEN' },
+                data: {
+                  Entity: a.Entity,
+                  RecordID: a.RecordID,
+                  target: a.Target,
+                },
+              }
+              return g(a)
+            },
+            edit: function (a) {
+              a = {
+                action: { record: 'EDIT' },
+                data: {
+                  Entity: a.Entity,
+                  RecordID: a.RecordID,
+                  target: a.Target,
+                },
+              }
+              return g(a)
+            },
+            create: function (a) {
+              a = {
+                action: { record: 'CREATE' },
+                data: {
+                  Entity: a.Entity,
+                  RecordID: a.RecordID,
+                  target: a.Target,
+                },
+              }
+              return g(a)
+            },
+            populate: function (a) {
+              return g({ action: { record: 'POPULATE' }, data: a })
+            },
+          },
+          Popup: {
+            close: function () {
+              return g({ action: { popup: 'CLOSE' } })
+            },
+            closeReload: function () {
+              return g({ action: { popup: 'CLOSE_RELOAD' } })
+            },
+          },
+          Widget: {
+            open: function (a) {
+              a = { action: { webTab: 'OPEN' }, data: a }
+              return g(a)
+            },
+          },
+        },
+        HTTP: {
+          get: function (a) {
+            return p('wget.get', a)
+          },
+          post: function (a) {
+            return p('wget.post', a)
+          },
+          put: function (a) {
+            return p('wget.put', a)
+          },
+          patch: function (a) {
+            return p('wget.patch', a)
+          },
+          delete: function (a) {
+            return p('wget.delete', a)
+          },
+        },
+        CONNECTOR: {
+          invokeAPI: function (a, b) {
+            return p(a, b, 'CONNECTOR_API')
+          },
+          authorize: function (a) {
+            return p(a, {}, 'CONNECTOR_AUTHORIZE')
+          },
+        },
+        CONNECTION: {
+          invoke: function (a, d) {
+            var e = {},
+              f = {}
+            f.url = d.url
+            f.method = d.method
+            f.param_type = d.param_type
+            f.parameters = JSON.stringify(d.parameters)
+            f.headers = JSON.stringify(d.headers)
+            e.data = f
+            return b({ category: 'CRM_CONNECTION', connectionName: a, data: e })
+          },
+        },
+        WIZARD: {
+          post: function (a) {
+            a = { category: 'CRM_WIZARD', data: JSON.stringify(a) }
+            return b(a)
+          },
+        },
+        BLUEPRINT: {
+          proceed: function () {
+            return b({ category: 'CRM_BLUEPRINT' })
+          },
+        },
+      }
+    })(),
+  }
+})()
+var ZSDKUtil = (function (c) {
+    function m(b) {}
+    function h(b) {
+      var c = {}
+      b = b || window.location.href
+      b.substr(b.indexOf('?') + 1)
+        .split('\x26')
+        .forEach(function (b, n) {
+          var h = b.split('\x3d')
+          c[h[0]] = h[1]
+        })
+      c.hasOwnProperty('serviceOrigin') &&
+        (c.serviceOrigin = decodeURIComponent(c.serviceOrigin))
+      return c
+    }
+    var n = h(),
+      b
+    m.prototype.Info = function () {
+      (c.isDevMode() || c.isLogEnabled()) &&
+        window.console.info.apply(null, arguments)
+    }
+    m.prototype.Error = function () {
+      (c.isDevMode() || c.isLogEnabled()) &&
+        window.console.error.apply(null, arguments)
+    }
+    c.GetQueryParams = h
+    c.isDevMode = function () {
+      return n && n.isDevMode
+    }
+    c.isLogEnabled = function () {
+      return n && n.isLogEnabled
+    }
+    c.getLogger = function () {
+      (b && b instanceof m) || (b = new m())
+      return b
+    }
+    c.Sleep = function (b) {
+      for (var c = new Date().getTime(); c + b > new Date().getTime(); );
+    }
+    return c
+  })(window.ZSDKUtil || {}),
+  ZSDKMessageManager = (function (c) {
+    function m(e) {
+      try {
+        var f = 'string' === typeof e.data ? JSON.parse(e.data) : e.data
+      } catch (c) {
+        f = e.data
+      }
+      var l = f.type,
+        m = f.eventName
+      try {
+        var k
+        if (!(k = 'SET_CONTEXT' === m)) {
+          var p = e.source,
+            q = e.origin
+          k =
+            g.isAppRegistered() && a === p && d === q
+              ? !0
+              : Error('Un-Authorized Message.')
+        }
+        if (k)
+          switch (l) {
+            case 'FRAMEWORK.EVENT':
+              var w = {
+                SET_CONTEXT: h,
+                UPDATE_CONTEXT: n,
+                EVENT_RESPONSE: b,
+                EVENT_RESPONSE_FAILURE: t,
+              }[f.eventName]
+              w && 'function' === typeof w
+                ? w(e, f)
+                : ZSDKEventManager.NotifyEventListeners(
+                    g.AppContext,
+                    f.eventName,
+                    f.data,
+                  )
+              break
+            default:
+              g.MessageInterceptor(e, f)
+          }
+      } catch (c) {
+        r.Error('[SDK.MessageHandler] \x3d\x3e ', c.stack)
+      }
+    }
+    function h(b, c) {
+      a = window.parent
+      d = g.QueryParams.serviceOrigin
+      g.SetContext(c.data)
+      g.ExecuteLoadHandler()
+    }
+    function n(a, b) {}
+    function b(a, b) {
+      k(b.promiseid, !0, b.data)
+    }
+    function t(a, b) {
+      k(b.promiseid, !1, b.data)
+    }
+    function k(a, b, c) {
+      l.hasOwnProperty(a) &&
+        (b ? l[a].resolve(c) : l[a].reject(c), (l[a] = void 0), delete l[a])
+    }
+    function y(a) {
+      return new Promise(function (b, c) {
+        l[a] = { resolve: b, reject: c, time: new Date().getTime() }
+      })
+    }
+    function v(b) {
+      'object' === typeof b && (b.appOrigin = encodeURIComponent(p()))
+      if (!a) throw Error('Parentwindow reference not found.')
+      a.postMessage(b, g.QueryParams.serviceOrigin)
+    }
+    function p() {
+      return (
+        window.location.protocol +
+        '//' +
+        window.location.host +
+        window.location.pathname
+      )
+    }
+    var g,
+      r = ZSDKUtil.getLogger(),
+      u = 100,
+      l = {},
+      a,
+      d
+    c.Init = function (a, b) {
+      if (!a || 'object' !== typeof a)
+        throw Error('Invalid Context object passed')
+      if (b && 'object' !== typeof b)
+        throw Error('Invalid Configuration Passed to MessageManager')
+      g = a
+      return m.bind(c)
+    }
+    c.RegisterApp = function () {
+      var a = {
+        type: 'SDK.EVENT',
+        eventName: 'REGISTER',
+        appOrigin: encodeURIComponent(p()),
+      }
+      window.parent.postMessage(a, g.QueryParams.serviceOrigin)
+    }
+    c.DERegisterApp = function () {
+      var a = {
+        type: 'SDK.EVENT',
+        eventName: 'DEREGISTER',
+        uniqueID: g.getUniqueID(),
+      }
+      v(a)
+    }
+    c.SendRequest = function (a) {
+      if (!a || 'object' !== typeof a) throw Error('Invalid Options passed')
+      var b
+      b = 'Promise' + u++
+      a = {
+        type: 'SDK.EVENT',
+        eventName: 'HTTP_REQUEST',
+        uniqueID: g.getUniqueID(),
+        time: new Date().getTime(),
+        promiseid: b,
+        data: a,
+      }
+      v(a)
+      b = y(b)
+      return b
+    }
+    c.TriggerEvent = function (a, b, c) {
+      if (!a) throw Error('Invalid Eventname : ', a)
+      var d = c ? 'Promise' + u++ : void 0
+      a = {
+        type: 'SDK.EVENT',
+        eventName: a,
+        uniqueID: g.getUniqueID(),
+        time: new Date().getTime(),
+        promiseid: d,
+        data: b,
+      }
+      v(a)
+      if (c) return y(d)
+    }
+    return c
+  })(window.ZSDKMessageManager || {}),
+  ZSDKEventManager = (function (c) {
+    var m = ZSDKUtil.getLogger(),
+      h = {}
+    c.AttachEventListener = function (c, b) {
+      'function' === typeof b &&
+        (Array.isArray(h[c]) || (h[c] = []), h[c].push(b))
+    }
+    c.NotifyEventListeners = function (c, b, t) {
+      var k = b.match(/^\__[A-Za-z_]+\__$/gi)
+      Array.isArray(k)
+      if ((k = h[b]) && Array.isArray(k))
+        for (b = 0; b < k.length; b++) k[b].call(c, t)
+      else m.Info('Cannot find EventListeners for Event : ', b)
+    }
+    c.NotifyInternalEventHandler = function (c, b) {
+      var h = b.eventName
+      '__APP_INIT__' === h
+        ? (c.SetContext(b.data), c.ExecuteLoadHandler())
+        : '__APP_CONTEXT_UPDATE__' === h &&
+          (c.UpdateContext(b.data), c.ExecuteContextUpdateHandler())
+    }
+    return c
+  })(window.ZSDKEventManager || {})
+function ZSDK() {
+  function c() {
+    'function' !== typeof l
+      ? z.Error('No OnLoad Handler provided to execute.')
+      : C
+        ? z.Error('OnLoad event already triggered.')
+        : (l.call(q, q), (C = !0))
+  }
+  function m() {
+    a.call(q, q)
+  }
+  function h() {
+    return B
+  }
+  function n(a, b, c) {
+    return ZSDKMessageManager.TriggerEvent(a, b, c)
+  }
+  function b(a) {
+    z.Info('Setting AppContext data')
+    var b = (a && a.model) || {}
+    isDevMode &&
+      a.locale &&
+      a.localeResource &&
+      0 === Object.keys(a.localeResource).length &&
+      a.localeResource.constructor === Object &&
+      a.locale &&
+      v(a.locale)
+    if ('undefined' !== typeof ZSDKModelManager) {
+      for (var c in b) ZSDKModelManager.AddModel(c, b[c])
+      q.Model = ZSDKModelManager.GetModelStore()
+    }
+    f = a.uniqueID
+    d = a.connectors
+    z.Info('App Connectors ', d)
+    B = !0
+  }
+  function t() {
+    return f
+  }
+  function k(a) {}
+  function y() {
+    return d
+  }
+  function v(a) {
+    p('/app-translations/' + a + '.json', function (a) {
+      A = JSON.parse(a)
+      u()
+    })
+  }
+  function p(a, b) {
+    var c = new XMLHttpRequest()
+    c.open('GET', a, !1)
+    c.onreadystatechange = function () {
+      4 == c.readyState && '200' == c.status && b(c.responseText)
+    }
+    c.send(null)
+  }
+  function g(a, b, c) {
+    for (var d = ''; d != a; ) (d = a), (a = a.replace(b, c))
+    return a
+  }
+  function r(a, b) {
+    b = b.replace(/\[(\w+)\]/g, '.$1')
+    b = b.replace(/^\./, '')
+    for (var c = b.split('.'), d = 0, e = c.length; d < e; ++d) {
+      var f = c[d]
+      if (f in a) a = a[f]
+      else return
+    }
+    return a
+  }
+  function u() {
+    var a = document.querySelectorAll('[data-i18n]'),
+      b
+    for (b in a)
+      if (a.hasOwnProperty(b)) {
+        var c = r(A, a[b].getAttribute('data-i18n'))
+        if (!c) return !1
+        if (a[b].hasAttribute('data-options')) {
+          var d = JSON.parse(
+              JSON.stringify(
+                eval('(' + a[b].getAttribute('data-options') + ')'),
+              ),
+            ),
+            e = Object.keys(d),
+            f
+          for (f in e) c = g(c, '${' + e[f] + '}', d[e[f]])
+        }
+        a[b].innerHTML = c
+      }
+  }
+  var l,
+    a,
+    d,
+    e,
+    f,
+    A = {},
+    z = ZSDKUtil.getLogger(),
+    B = !1,
+    C = !1
+  this.isContextReady = !1
+  this.HelperContext = {}
+  this.isDevMode = !1
+  this.getContext = function () {
+    return q
+  }
+  var q = { Model: {}, Event: {} }
+  q.Event.Listen = function (a, b) {
+    ZSDKEventManager.AttachEventListener(a, b)
+  }
+  q.Event.Trigger = n
+  q.GetRequest = function (a) {
+    return ZSDKMessageManager.SendRequest(a)
+  }
+  q.QueryParams = e
+  q.Translate = function (a, b) {
+    var c = ''
+    a && (c = r(A, a))
+    if (!c) return !1
+    if (b) {
+      var d = JSON.parse(JSON.stringify(eval(b))),
+        e = Object.keys(d)
+      for (a in e) c = g(c, '${' + e[a] + '}', d[e[a]])
+    }
+    return c
+  }
+  this.OnLoad = function (a) {
+    if ('function' !== typeof a) throw Error('Invalid Function value is passed')
+    l = a
+    B && c()
+  }
+  this.OnUnLoad = function (a) {}
+  this.OnContextUpdate = function (b) {
+    a = b
+  }
+  ;(function () {
+    e = ZSDKUtil.GetQueryParams()
+    isDevMode = !!e.isDevMode
+    var a = {}
+    a.isDevMode = isDevMode
+    a.ExecuteLoadHandler = c
+    a.SetContext = b
+    a.UpdateContext = k
+    a.QueryParams = e
+    a.GetConnectors = y
+    a.TriggerEvent = n
+    a.ExecuteContextUpdateHandler = m
+    a.getUniqueID = t
+    a.isAppRegistered = h
+    var d = ZSDKMessageManager.Init(a)
+    window.addEventListener('message', d)
+    window.addEventListener('unload', function () {
+      ZSDKMessageManager.DERegisterApp()
+    })
+    'undefined' !== typeof ZSDKModelManager && ZSDKModelManager.Init(a)
+    ZSDKMessageManager.RegisterApp()
+  })()
+}

+ 525 - 0
src/pages/so-search/index.vue

@@ -0,0 +1,525 @@
+<template>
+  <div class="w-[100vw] bg-white">
+    <div
+      v-loading="loading"
+      class="pt-2 w-[1500px] mx-auto flex"
+    >
+      <div class="flex-auto flex flex-col items-stretch pr-8">
+        <div class="flex">
+          <el-form
+            style="width: 100%"
+            inline
+            :loading="loading"
+            @submit.prevent="search"
+          >
+            <el-form-item label-width="0">
+              <div class="flex">
+                <el-input
+                  v-model="keyword"
+                  style="width: 250px"
+                  :disabled="loading"
+                  clearable
+                  placeholder="Job Name / Reference"
+                ></el-input>
+                <el-button
+                  class="custom-button ml-2 fb"
+                  @click="search"
+                >
+                  Search Sales Order
+                </el-button>
+              </div>
+            </el-form-item>
+          </el-form>
+        </div>
+        <br />
+        <el-checkbox-group v-model="checkedContent">
+          <div class="flex flex-col max-h-[75vh] overflow-y-scroll">
+            <div
+              v-for="(item, index) in searchResult"
+              :key="index"
+              class=""
+            >
+              <el-checkbox :value="item.id">
+                {{ item.Reference }}-{{ item.Contract_Title }}-{{
+                  item.Owner.name || ''
+                }}
+              </el-checkbox>
+            </div>
+          </div>
+        </el-checkbox-group>
+        <br />
+        <div class="flex">
+          <el-tooltip
+            content="最好不要一次过提交太多, 携带太多信息会让生成的二维码会非常难识别"
+          >
+            <el-button
+              v-show="searchResult.length"
+              v-loading="loading"
+              class="custom-button"
+              @click="submit"
+            >
+              Submit To Right
+            </el-button>
+          </el-tooltip>
+        </div>
+
+        <!-- <div class="border border-solid border-grep-1">
+          <img
+            :src="tempImage"
+            style="width: 20cm"
+          />
+        </div> -->
+      </div>
+      <div class="flex flex-col items-stretch w-[24cm] max-w-[24cm]">
+        <div class="flex justify-between">
+          <div>
+            <el-tooltip
+              content="其实二维码在提交SO后就会自动刷新, 除非你点了右边的Clean QR Code"
+            >
+              <el-button
+                class="custom-button"
+                @click="reGenerateQrCode"
+              >
+                Generate QR Code
+              </el-button>
+            </el-tooltip>
+            <el-tooltip content="除非你不想打印二维码">
+              <el-button @click="qrcodeVisible = false">
+                Clean QR Code
+              </el-button>
+            </el-tooltip>
+          </div>
+          <div>
+            <el-button
+              type="danger"
+              @click="clean"
+            >
+              Clean Input Job
+            </el-button>
+            <el-tooltip content="确认下面的预览内容没问题再点打印">
+              <el-button
+                class="custom-button"
+                :loading="printing"
+                @click="onBtnPrint"
+              >
+                Print Page
+              </el-button>
+            </el-tooltip>
+          </div>
+        </div>
+        <br />
+        <div class="">
+          <el-form inline>
+            <!-- style="width: 100%" -->
+            <el-form-item
+              label="QR Code Content"
+              prop=""
+              class="text-left w-[100%]"
+            >
+              <div
+                class="el-input el-input__wrapper el-input__inner cursor-not-allowed hover:cursor-not-allowed"
+              >
+                {{ selectedContent.map((i) => i.Reference).join(';') }}
+              </div>
+            </el-form-item>
+
+            <el-form-item
+              label="Total Page"
+              style="width: 20%"
+              prop="total"
+            >
+              <el-input
+                v-model.number="total"
+                :type="'number'"
+              ></el-input>
+            </el-form-item>
+            <el-form-item
+              label="Text direction"
+              style="width: 15%"
+              prop="rotated"
+            >
+              <el-switch v-model="rotated"></el-switch>
+            </el-form-item>
+            <el-form-item
+              label="Page width(CM)"
+              style="width: 20%"
+              prop="pageWidth"
+            >
+              <el-input
+                v-model.number="pageWidth"
+                :type="'number'"
+              ></el-input>
+            </el-form-item>
+            <el-form-item
+              label="Page Height(CM)"
+              style="width: 20%"
+              prop="pageHeight"
+            >
+              <el-input
+                v-model.number="pageHeight"
+                :type="'number'"
+              ></el-input>
+            </el-form-item>
+
+            <el-form-item
+              label="Font Size"
+              style="width: 50%"
+            >
+              <el-slider
+                v-model="pdfFontSize"
+                :step="1"
+              ></el-slider>
+            </el-form-item>
+            <el-form-item
+              style="width: 100%"
+              label="箱唛"
+            >
+              <el-input
+                v-model="mai"
+                type="textarea"
+              ></el-input>
+            </el-form-item>
+            <div class="flex gap-1">
+              <div
+                v-for="(item, index) in logoList"
+                :key="item"
+                class="w-[60px] h-[60px] bg-contain bg-center bg-no-repeat cursor-pointer"
+                :style="{ backgroundImage: `url(${item})` }"
+                @click="selectlogo(index)"
+              ></div>
+            </div>
+          </el-form>
+        </div>
+        <br />
+        <div class="flex flex-wrap">
+          <div
+            v-for="i in total"
+            :key="i"
+            :style="{
+              width: `${pageWidth}cm`,
+              height: `${pageHeight}cm`,
+            }"
+            class="pdf-wrap relative border border-solid border-gray-e mb-2 mr-2"
+          >
+            <div
+              :id="`pdfItem-${i}`"
+              :ref="(el) => setElement(el, i)"
+              class="pdf-area w-[100%] h-[100%] relative text-center p-4 flex flex-col items-stretch justify-center"
+              :class="{}"
+              :style="{
+                width: rotated ? `${pageHeight}cm` : `${pageWidth}cm`,
+                height: rotated ? `${pageWidth}cm` : `${pageHeight}cm`,
+                transformOrigin: 'top left',
+                transform: rotated ? 'rotate(90deg)' : '',
+                right: rotated ? `-${pageWidth}cm` : '',
+              }"
+            >
+              <!-- bottom: rotated ? `-${(i - 1) * (pageWidth - pageHeight)}cm` : '', -->
+              <div
+                class="mai font-medium break-all mb-4"
+                :style="{
+                  fontSize: `${pdfFontSize}pt`,
+                }"
+                @dblclick="ElMessage.info('hello')"
+              >
+                {{ mai }}
+              </div>
+
+              <div class="flex justify-center">
+                <div class="font-medium text-5xl">{{ i }} / {{ total }}</div>
+              </div>
+              <div class="flex gap-1 absolute bottom-4 left-4">
+                <div
+                  v-for="item in selectedLogo"
+                  :key="item"
+                  class="w-[90px] h-[90px] bg-contain bg-center bg-no-repeat"
+                  :style="{ backgroundImage: `url(${item})` }"
+                ></div>
+              </div>
+              <div
+                v-show="qrcodeVisible"
+                class="qrcode absolute bottom-4 right-4 w-[110px] h-[110px] border border-solid border-gray-7"
+              >
+                <img
+                  v-show="qrcodeURL.length"
+                  style="width: 100%; height: auto"
+                  :src="qrcodeURL"
+                  alt="qr-code"
+                />
+              </div>
+            </div>
+          </div>
+        </div>
+      </div>
+    </div>
+  </div>
+</template>
+<script lang="ts">
+export default defineComponent({
+  name: 'PageSoSearch',
+})
+</script>
+<script lang="ts" setup>
+import { defineComponent, nextTick, ref, watch, computed } from 'vue'
+import {
+  ElMessage,
+  ElCheckbox,
+  ElInput,
+  ElButton,
+  ElForm,
+  ElFormItem,
+  ElCheckboxGroup,
+  ElSlider,
+  ElTooltip,
+  ElSwitch,
+} from 'element-plus'
+import jspdf from 'jspdf'
+import qrcode from 'qrcode'
+import html2canvas from 'html2canvas'
+import debounce from 'lodash.debounce'
+
+const searchResult = ref([] as any[])
+// @ts-ignore
+const zoho = window.ZOHO
+zoho.embeddedApp.on('PageLoad', function () {
+  zoho.CRM.CONFIG.getCurrentUser().then(function (data: any) {
+    console.log(data, 'current user')
+  })
+})
+zoho.embeddedApp.init()
+const loading = ref(false)
+const keyword = ref('')
+const search = () => {
+  if (!keyword.value.trim().length) {
+    ElMessage.warning('Please input Job Name or Reference')
+    return
+  }
+  searchResult.value = []
+  loading.value = true
+  var conn_name = 'promocollection_connection'
+  var req_data = {
+    parameters: {
+      criteria: `(Reference:in:${keyword.value})or(Reference:starts_with:${keyword.value})or(Contract_Title:in:${keyword.value})or(Contract_Title:starts_with:${keyword.value})`,
+      converted: 'both',
+    },
+    method: 'GET',
+    url: 'https://www.zohoapis.com/crm/v7/Sales_Orders/search',
+    param_type: 1,
+  }
+  zoho.CRM.CONNECTION.invoke(conn_name, req_data)
+    .then(function (data: any) {
+      // console.log(data, 'data')
+      if (data.status !== 'success' || !data.details.status) {
+        return ElMessage.error('crm sdk connection invoke error')
+      }
+      const result = data.details.statusMessage || {}
+      if (Array.isArray(result.data) && result.data.length) {
+        searchResult.value = result.data
+        console.log(result.data, 'result')
+      } else {
+        ElMessage.warning('No Data Found')
+      }
+    })
+    .finally(() => {
+      loading.value = false
+    })
+}
+// 左侧选中的
+let checkedContent = ref([] as any[])
+// 提交到右侧的
+let selectedContent = ref([] as any[])
+let submit = () => {
+  if (!checkedContent.value.length) {
+    ElMessage.warning('Select sales order before click submit')
+  }
+  checkedContent.value.forEach((i: number) => {
+    const temp = searchResult.value.filter((a: any) => a.id === i)
+    if (
+      temp.length &&
+      !selectedContent.value.map((i: any) => i.id).includes(temp[0].id)
+    ) {
+      selectedContent.value.push(temp[0])
+    }
+  })
+  nextTick(() => {
+    checkedContent.value = []
+  })
+}
+const mai = ref('')
+watch(
+  () => selectedContent,
+  () => {
+    mai.value = selectedContent.value
+      .map((c: any) => `${c.Reference}_${c.Contract_Title}`)
+      .join('; ')
+    if (mai.value.length) {
+      reGenerateQrCode()
+    }
+  },
+  { deep: true, immediate: true },
+)
+
+const clean = () => {
+  mai.value = ''
+  selectedContent.value = []
+}
+let qrcodeVisible = ref(false)
+let qrcodeURL = ref('')
+async function reGenerateQrCode() {
+  await qrcode.toDataURL(
+    mai.value,
+    { errorCorrectionLevel: 'H' },
+    function (err: any, url: any) {
+      qrcodeURL.value = url
+    },
+  )
+  qrcodeVisible.value = true
+}
+
+const pdfList = ref({} as any)
+const setElement = (el: any, index: number) => {
+  pdfList.value[`${index}`] = el
+}
+const getLogoPath = (path: string) => {
+  return new URL(path, import.meta.url).href
+}
+
+const logoList = ref([] as any[])
+
+async function getLogoList() {
+  (
+    [
+      getLogoPath('/assets/label/heavy.png'),
+      getLogoPath('/assets/label/keep_dry.png'),
+      getLogoPath('/assets/label/place_up.png'),
+      getLogoPath('/assets/label/care.png'),
+      getLogoPath('/assets/label/glass.png'),
+    ] as string[]
+  ).forEach((i: string, index: number) => {
+    fetch(i)
+      .then((res) => res.blob())
+      .then((blob) => (logoList.value[index] = URL.createObjectURL(blob)))
+  })
+}
+getLogoList()
+
+const selectlogo = (index: number) => {
+  const i = selectedLogoIndex.value.indexOf(index)
+  if (i > -1) {
+    selectedLogoIndex.value.splice(i, 1)
+  } else {
+    selectedLogoIndex.value.push(index)
+  }
+}
+let selectedLogoIndex = ref([] as number[])
+let selectedLogo = computed(() =>
+  logoList.value.filter((i: any, index: number) =>
+    selectedLogoIndex.value.includes(index),
+  ),
+)
+
+let total = ref(1 as number)
+let pdfFontSize = ref(32)
+let pageWidth = ref(20) // 目标pdf的页宽
+let pageHeight = ref(10) // 目标pdf的页高
+/**
+ * 宽大于等于高
+ */
+let orientation = computed(() => pageWidth.value >= pageHeight.value)
+let rotated = ref(false)
+let pdf: any = null
+let printing = ref(false)
+const reGeneratePDF = debounce(function () {
+  printing.value = true
+  pdf = null
+  pdf = new jspdf({
+    orientation: orientation.value ? 'l' : 'p',
+    unit: 'cm',
+    format: [pageWidth.value, pageHeight.value],
+  })
+  const pool = []
+  for (const key in pdfList.value) {
+    if (Object.prototype.hasOwnProperty.call(pdfList.value, key)) {
+      pool.push(generatePDF(pdfList.value[key], key))
+    }
+  }
+  Promise.all(pool).then(() => {
+    printing.value = false
+  })
+}, 300)
+
+watch(
+  [total, mai, pdfFontSize, pageWidth, pageHeight, qrcodeVisible, rotated, selectedLogo],
+  () => {
+    nextTick(() => reGeneratePDF())
+  },
+)
+async function onBtnPrint() {
+  window.open(
+    // @ts-ignore 照着jspdf文档搬的...ts报错
+    pdf.output('bloburl', {
+      fileName: selectedContent.value.map((i: any) => i.Reference).join('_'),
+    }),
+  )
+}
+function generatePDF(pdfItem: HTMLElement, key: string) {
+  return html2canvas(pdfItem, {
+    allowTaint: true,
+    useCORS: true,
+    backgroundColor: '#fff', // 一定要设背景颜色,否则有的浏览器就会变花,比如Edge
+    scale: 3, // 缩放倍率调整清晰度
+  }).then((canvas) => {
+    const width = canvas.width
+    const height = canvas.height
+    const image = new Image()
+    const target = document.createElement('canvas')
+    target.height = height
+    target.width = width
+    image.onload = function () {
+      const ctx = target.getContext('2d')
+      ctx?.drawImage(image, 0, 0)
+      if (Number(key) > 1) pdf.addPage()
+      pdf.addImage(
+        target.toDataURL('image/jpeg', 1.0),
+        'JPEG',
+        0,
+        0,
+        pageWidth.value,
+        pageHeight.value,
+      )
+    }
+    image.src = canvas.toDataURL('image/jpeg', 1.0)
+  })
+}
+</script>
+
+<style lang="scss" scoped>
+.el-button.custom-button {
+  height: 32px;
+  line-height: 20px;
+  background-image: linear-gradient(
+    171deg,
+    rgb(28, 74, 136) 49%,
+    rgb(0, 130, 193) 100%
+  );
+  // padding: 6px 20px 6px 7px;
+  padding: 6px 12px;
+  background-color: rgb(97, 165, 245);
+  color: #fff;
+  font-size: 13px;
+  font-family: Zoho_Puvi_Bold sans-serif;
+  border-radius: 6px;
+  cursor: pointer;
+  &:hover,
+  &:active {
+    background-image: linear-gradient(
+      171deg,
+      rgb(28, 74, 136) 49%,
+      rgb(22, 208, 239) 100%
+    );
+  }
+  &.fb {
+    font-weight: 900;
+  }
+}
+</style>

+ 5 - 0
src/route.ts

@@ -97,6 +97,11 @@ const router = createRouter({
         },
       ],
     },
+    {
+      name: 'soSearch',
+      path: '/so-search',
+      component: () => import('@/pages/so-search/index.vue'),
+    },
     {
       path: '/:pathMatch(.*)*',
       name: 'pageNotFound',