javascript ToastUI图像编辑器loadImageFromURL不起作用

6l7fqoea  于 2023-01-07  发布在  Java
关注(0)|答案(3)|浏览(193)

bounty将在6天后过期。回答此问题可获得+200声望奖励。Tenders McChiken希望奖励现有答案

请注意,这是一个自我回答的问题。
这个问题是关于ToastUI图像编辑器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

plupiseo

plupiseo1#

靶病变; DR:
Here is a working fiddle: 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));
    };
})();

第三个问题可以这样解决:取loadImageFromURL返回的promise解析的对象,并将新的和旧的width/height属性传递给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上,默认情况下服务器不允许这样做。您必须手动设置服务器以允许您的域或任何域访问它。请参阅here。如果您使用的是其他服务器,例如,你可以像wikipedia对this image所做的那样,将access-control-allow-origin设置为*,然后你(和图像编辑器)可以从任何域的JavaScript访问该图像。

pwuypxnk

pwuypxnk2#

对于那些正在使用Rails的人,当谈到@Forivin所陈述的第四个问题时,这就是我为使它工作所做的。
问题是当吐司调用存储在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项目中执行以下操作:
Gemfile中添加机架代理gem

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代码中,我调用吐司编辑器,首先填充一个空白的白色图像。然后,当组件被挂载时,我加载s3图像,覆盖现有图像并调整画布的大小。我发现在阅读s3图像之前,我需要稍微延迟一点。s3图像是一个预先签名的url,我在props中传递它。

<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>
2j4z5cfb

2j4z5cfb3#

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

crossorigin="anonymous"

As explained here: https://github.com/nhn/tui.image-editor/issues/68#issuecomment-930106372

相关问题