3

请注意,这是一个自我回答的问题。

这个问题是关于 ToastUI Image Editor v3.3.0,但它也可能适用于较新的版本。

当您使用此官方示例加载图像时:

// Create image editor
var imageEditor = new tui.component.ImageEditor('#my-image-editor canvas', {
    cssMaxWidth: 1000, // Component default value: 1000
    cssMaxHeight: 800  // Component default value: 800
});

// Load image
imageEditor.loadImageFromURL('img/sampleImage.jpg', 'My sample image')

编辑器不会加载图像。该函数既不抛出也不返回任何指示失败的内容,并且您不会收到任何错误消息。它返回一个按照文档中指定的方式解析的承诺。

它仅通过在初始配置中指定来加载图像,之后您无法更改它:

// Create image editor
var imageEditor = new tui.component.ImageEditor('#my-image-editor canvas', {
     includeUI: {
         loadImage: {
             path: 'img/sampleImage.jpg',
             name: 'My sample image'
         },
     },
    cssMaxWidth: 1000, // Component default value: 1000
    cssMaxHeight: 800  // Component default value: 800
});

看来loadImageFromURL函数已损坏,并且根据其他用户的说法loadImageFromFile也有同样的问题。

关于这方面的问题已经在 GitHub 上提出过,但基本上都被忽略了。现在已经一个月了,不幸的是它仍然没有修复。

所以问题是在这个问题存在的情况下如何欺骗图像编辑器工作。

这是一个表明它不起作用的小提琴:https ://fiddle.sencha.com/#view/editor&fiddle/2org

4

3 回答 3

16

TL; DR:
这是一个工作小提琴:https ://fiddle.sencha.com/#view/editor&fiddle/2p0o


长版:

有四个问题:

  • 您需要加载初始图像,否则您无法使用编辑控件。
  • 您需要等到图像编辑器对象准备好后再调用loadImageFromURL,否则您可能会收到错误或静默失败。
  • 加载图像时,您需要告诉图像编辑器新的尺寸,否则图像将被隐藏或尺寸不正确。
  • 如果加载外部图像,外部服务器必须设置Access-Control-Allow-Origin标头并明确允许您的域访问它,否则图像编辑器无法访问它。

第一个问题可以通过加载这样的空白图像来解决:

var imageEditor = new tui.ImageEditor('#tui-image-editor-container', {
    includeUI: {
        loadImage: {
            path: '',
            name: 'Blank'
        },
        theme: whiteTheme,
        menuBarPosition: 'bottom'
    },
    cssMaxWidth: 700,
    cssMaxHeight: 700
});

第二个问题可以通过等待图像编辑器使用未记录的功能退出其锁定状态来解决。您可以loadImageFromURL像这样在运行时修补您的:

imageEditor.loadImageFromURL = (function() {
    var cached_function = imageEditor.loadImageFromURL;
    function waitUntilImageEditorIsUnlocked(imageEditor) {
        return new Promise((resolve,reject)=>{
            const interval = setInterval(()=>{
                if (!imageEditor._invoker._isLocked) {
                    clearInterval(interval);
                    resolve();
                }
            }, 100);
        })
    }
    return function() {
        return waitUntilImageEditorIsUnlocked(imageEditor).then(()=>cached_function.apply(this, arguments));
    };
})();

第三个问题可以通过获取 promise 返回的对象来loadImageFromURL解决,并将新的和旧的宽度/高度属性传递给ui.resizeEditor函数,如下所示:

imageEditor.loadImageFromURL("https://upload.wikimedia.org/wikipedia/en/thumb/8/80/Wikipedia-logo-v2.svg/526px-Wikipedia-logo-v2.svg.png", "SampleImage").then(result=>{
    imageEditor.ui.resizeEditor({
        imageSize: {oldWidth: result.oldWidth, oldHeight: result.oldHeight, newWidth: result.newWidth, newHeight: result.newHeight},
    });
}).catch(err=>{
    console.error("Something went wrong:", err);
})

第四个问题可能有点令人困惑。让我解释。在网站上,您可以使用标签包含几乎任何您想要的外部图像<img>,但如果您想使用 JavaScript 访问外部图像,提供图像的服务器必须明确允许您使用access-control-allow-origin标头来执行此操作。例如,在 Amazon S3 上,服务器默认不允许这样做。您必须手动设置服务器以允许您或任何域访问它。见这里。如果您使用的是不同的服务器,例如,您可以将 设置access-control-allow-origin*wikipedia has done on this image。然后您(和图像编辑器)可以从任何域的 JavaScript 访问该图像。

于 2019-01-11T23:38:31.483 回答
0

在打开之前将此属性添加到要编辑的图像标签中:

crossorigin="anonymous"

如此处所述: https ://github.com/nhn/tui.image-editor/issues/68#issuecomment-930106372

于 2021-10-12T08:10:57.513 回答
0

对于那些使用 Rails 的人,当谈到 @Forivin 提出的第四个问题时,我就是这样做的。

问题是当 Toast 调用存储在 S3 上的图像时,我会在 Chrome 上收到 CORS 错误,但 firefox 很好。有很多关于这方面的文章,基本上我发现最好的方法是在我的代码中使用代理。我仍然可以让我的 CORS 源指向我的主机,并且由于呼叫是通过代理从我的主机发出的,所以 S3 和 Chrome 很高兴。我的 S3 CORS 配置如下所示(允许子域):

<?xml version="1.0" encoding="UTF-8"?>
<CORSConfiguration xmlns="http://s3.amazonaws.com/doc/2006-03-01/">
<CORSRule>
    <AllowedOrigin>http://*.mycompany.com</AllowedOrigin>
    <AllowedMethod>POST</AllowedMethod>
    <AllowedMethod>GET</AllowedMethod>
    <AllowedMethod>PUT</AllowedMethod>
    <AllowedMethod>DELETE</AllowedMethod>
    <AllowedMethod>HEAD</AllowedMethod>
    <AllowedHeader>*</AllowedHeader>
</CORSRule>
</CORSConfiguration>

在您的 Rails 项目中执行以下操作:

将 rack-proxy gem 添加到你的Gemfile

gem 'rack-proxy'

创建代理文件。s3 路径经过 URI 编码并附加到路由的末尾。该路由仅用于代理,因此它可以是任何东西,因为它将被重新路由到 s3。

app/proxy/s3_proxy.rb

class S3Proxy < Rack::Proxy

  def perform_request(env)
    if env['REQUEST_PATH'] =~ %r{^/my/dummy/path}
      s3_path = CGI.unescape(env['REQUEST_PATH'][15..-1])

      uri = URI.parse(s3_path)
      env['HTTP_HOST'] = uri.host
      env['SERVER_PORT'] = uri.port
      env['REQUEST_PATH'] = s3_path
      env['REQUEST_URI'] = s3_path
      env['PATH_INFO'] = s3_path
      env['rack.url_scheme'] = 'https'

      super(env)
    else
      @app.call(env)
    end
  end

end

添加到application.rb文件:

require "./app/proxy/s3_proxy"

class Application < Rails::Application
  ...

  config.middleware.use S3Proxy
end

routes.rb

get "/my/dummy/path/:s3_url", to: "my_controller#dummy_path"

中的控制器方法my_controller.rb。在这里渲染什么都没关系,因为它将被代理重定向。我们可能没有办法逃脱,因为代理无论如何都会改变。

  def dummy_path
    render plain: ""
  end

最后在我的 Vue 代码中,我调用 Toast 编辑器,首先填充一个空白图像。然后在安装组件时,我加载 s3 图像并覆盖现有图像并调整画布大小。我发现在读取 s3 图像之前安装它需要稍微延迟。s3 图像是我在 props 中传递的预签名 url。

<template lang="pug">
.v-image-editor-tool
  tui-image-editor(:include-ui='useDefaultUI' :options="editorOptions" ref="tuiImageEditor")
</template>

<script lang="coffee">
import { ImageEditor } from '@toast-ui/vue-image-editor'
import 'tui-image-editor/dist/tui-image-editor.min.css'

export default app =
  props: ['imageUrl']
  data: ->
    useDefaultUI: true
    editorOptions:
      cssMaxWidth: 700
      cssMaxHeight: 700
      usageStatistics: false
      includeUI:
        loadImage:
          path: ''
          name: 'Blank'
        menuBarPosition: 'bottom'

  mounted: ->
    fn = => this.$refs.tuiImageEditor.invoke('loadImageFromURL', @imageUrl, 'Image').then (result) =>
      this.$refs.tuiImageEditor.invoke('ui.resizeEditor', { imageSize: { newWidth: result.newWidth, newHeight: result.newHeight }})
    setTimeout(fn, 600)

  components:
    'tui-image-editor': ImageEditor
</script>

于 2019-12-12T23:58:20.580 回答