JavaScript'te panoya nasıl kopyalayabilirim?


3321

Metni panoya kopyalamanın en iyi yolu nedir? (Çoklu tarayıcı)

Denedim:

function copyToClipboard(text) {
    if (window.clipboardData) { // Internet Explorer
        window.clipboardData.setData("Text", text);
    } else {  
        unsafeWindow.netscape.security.PrivilegeManager.enablePrivilege("UniversalXPConnect");  
        const clipboardHelper = Components.classes["@mozilla.org/widget/clipboardhelper;1"].getService(Components.interfaces.nsIClipboardHelper);  
        clipboardHelper.copyString(text);
    }
}

ancak Internet Explorer'da bir sözdizimi hatası verir. Firefox'ta diyor unsafeWindow is not defined.

Flaşsız hoş bir numara: Trello kullanıcının panosuna nasıl erişir?


Sadece merak ediyorum, kullanıcının kendi yapamadığı panoya kopyalamak istediğiniz nedir?
scunliffe

233
Özel birşey yok. Bunu kendileri yapabilirler, ancak metnin doğru bölümünü seçmekten endişe etmeden bir düğmeyi tıklatma olasılığını da sunmak istiyorum.
Santiago Corredoira

4
Bu uzun blog yazısı, bunu yapmanın birçok yolunu içeriyor: Sistem Panosuna JavaScript ile Erişme - Kutsal Kase?
Aaron Digulla

IE'de olduğu gibi FF'de de tarayıcıya tanımlanmamış istisna veriyor
Jagadeesh

1
Kullanıcının panosuna metin koyabilirsek, panosunu mahvedebiliriz.
Frank Fang

Yanıtlar:


2247

genel bakış

Panoya kopyalamak için üç birincil tarayıcı API'si vardır:

  1. Async Clipboard API'sı [navigator.clipboard.writeText]
    • Metin odaklı bölüm Chrome 66'da kullanılabilir (Mart 2018)
    • Erişim zaman uyumsuzdur ve JavaScript Vaatleri kullanır , güvenlik kullanıcı istemleri (görüntüleniyorsa) sayfadaki JavaScript'i kesintiye için yazılabilir.
    • Metin bir değişkene doğrudan panoya kopyalanabilir.
    • Yalnızca HTTPS üzerinden sunulan sayfalarda desteklenir.
    • Chrome'da etkin sekmelerdeki 66 sayfa, izin istemi olmadan panoya yazabilir.
  2. document.execCommand('copy')
    • Çoğu tarayıcı bunu ~ Nisan 2015'ten itibaren desteklemektedir (aşağıdaki Tarayıcı Desteği'ne bakın).
    • Erişim eşzamanlıdır; başka bir deyişle, herhangi bir güvenlik istemiyle görüntüleme ve kullanıcı etkileşimi dahil olmak üzere sayfadaki JavaScript'i tamamlar.
    • Metin DOM'den okunur ve panoya yerleştirilir.
    • Test sırasında ~ Nisan 2015 panoya yazılırken yalnızca Internet Explorer'ın izin istemlerini görüntülediği belirtildi.
  3. Kopyalama olayını geçersiz kılma
    • Kopyalama olayını geçersiz kılma hakkında Pano API'sı belgelerine bakın .
    • Herhangi bir copy olayından panoda görünenleri değiştirmenize olanak tanır, düz metin dışındaki diğer veri formatlarını içerebilir.
    • Burada ele alınmadığı için doğrudan soruyu cevaplamıyor.

Genel geliştirme notları

Konsolda kodu test ederken pano ile ilgili komutların çalışmasını beklemeyin. Genel olarak sayfanın etkin (Async Clipboard API) olması gerekir ( document.execCommand('copy')panoya erişmesine izin vermek için) kullanıcı etkileşimi (örn. Kullanıcı tıklaması) gerekir . Daha fazla ayrıntı için aşağıya bakın.

ÖNEMLİ (burada not edilmiştir 2020/02/20)

Bu gönderi başlangıçta çapraz kaynaklı IFRAME'lerde ve diğer IFRAME "korumalı alanında" izinlerin kullanımdan kaldırıldığından , gömülü demoların "Kod snippet'ini çalıştır" düğmeleri ve "codepen.io örneğinin" bazı tarayıcılarda (Chrome ve Microsoft Edge dahil) çalışmasını engellediğini unutmayın. ).

Kendi web sayfanızı oluşturmak için, bu sayfayı HTTPS bağlantısı üzerinden test edin ve geliştirin.

Kodun çalıştığını gösteren bir test / demo sayfası: https://deanmarktaylor.github.io/clipboard-test/

Async + Yedek

Yeni Async Clipboard API'sinin tarayıcı desteği düzeyi nedeniyle, document.execCommand('copy')iyi bir tarayıcı kapsamı elde etme yöntemine geri dönmek isteyeceksiniz .

İşte basit bir örnek (bu siteye gömülü olmayabilir, yukarıdaki "önemli" notu okuyun):

function fallbackCopyTextToClipboard(text) {
  var textArea = document.createElement("textarea");
  textArea.value = text;
  
  // Avoid scrolling to bottom
  textArea.style.top = "0";
  textArea.style.left = "0";
  textArea.style.position = "fixed";

  document.body.appendChild(textArea);
  textArea.focus();
  textArea.select();

  try {
    var successful = document.execCommand('copy');
    var msg = successful ? 'successful' : 'unsuccessful';
    console.log('Fallback: Copying text command was ' + msg);
  } catch (err) {
    console.error('Fallback: Oops, unable to copy', err);
  }

  document.body.removeChild(textArea);
}
function copyTextToClipboard(text) {
  if (!navigator.clipboard) {
    fallbackCopyTextToClipboard(text);
    return;
  }
  navigator.clipboard.writeText(text).then(function() {
    console.log('Async: Copying to clipboard was successful!');
  }, function(err) {
    console.error('Async: Could not copy text: ', err);
  });
}

var copyBobBtn = document.querySelector('.js-copy-bob-btn'),
  copyJaneBtn = document.querySelector('.js-copy-jane-btn');

copyBobBtn.addEventListener('click', function(event) {
  copyTextToClipboard('Bob');
});


copyJaneBtn.addEventListener('click', function(event) {
  copyTextToClipboard('Jane');
});
<div style="display:inline-block; vertical-align:top;">
  <button class="js-copy-bob-btn">Set clipboard to BOB</button><br /><br />
  <button class="js-copy-jane-btn">Set clipboard to JANE</button>
</div>
<div style="display:inline-block;">
  <textarea class="js-test-textarea" cols="35" rows="4">Try pasting into here to see what you have on your clipboard:

  </textarea>
</div>

(codepen.io örneği çalışmayabilir, yukarıdaki "önemli" notu okuyun) Bu snippet'in Stack Overflow'un gömülü önizlemesinde iyi çalışmadığını buradan deneyebilirsiniz: https://codepen.io/DeanMarkTaylor/pen/RMRaJX?editors = 1011

Async Clipboard API'sı

Chrome 66'daki izinler API'sı aracılığıyla "izin isteme" ve panoya erişimi test etme yeteneğinin bulunduğunu unutmayın.

var text = "Example text to appear on clipboard";
navigator.clipboard.writeText(text).then(function() {
  console.log('Async: Copying to clipboard was successful!');
}, function(err) {
  console.error('Async: Could not copy text: ', err);
});

document.execCommand ( 'kopya')

Bu yazının geri kalanı, document.execCommand('copy')API'nın nüanslarına ve ayrıntılarına gider .

Tarayıcı Desteği

JavaScript document.execCommand('copy')desteği büyüdü, tarayıcı güncellemeleri için aşağıdaki bağlantılara bakın:

Basit Örnek

(bu siteye gömülü çalışmayabilir, yukarıdaki "önemli" notu okuyun)

var copyTextareaBtn = document.querySelector('.js-textareacopybtn');

copyTextareaBtn.addEventListener('click', function(event) {
  var copyTextarea = document.querySelector('.js-copytextarea');
  copyTextarea.focus();
  copyTextarea.select();

  try {
    var successful = document.execCommand('copy');
    var msg = successful ? 'successful' : 'unsuccessful';
    console.log('Copying text command was ' + msg);
  } catch (err) {
    console.log('Oops, unable to copy');
  }
});
<p>
  <button class="js-textareacopybtn" style="vertical-align:top;">Copy Textarea</button>
  <textarea class="js-copytextarea">Hello I'm some text</textarea>
</p>

Karmaşık Örnek: Girdi göstermeden panoya kopyala

Yukarıdaki basit örnek, bir textareaveyainput görünen öğe .

Bazı durumlarda, bir input/ textareaelement göstermeden panoya metin kopyalamak isteyebilirsiniz . Bu, bu soruna geçici bir çözüm bulmak için bir örnektir (temel olarak öğe ekle, panoya kopyala, öğeyi kaldır):

Google Chrome 44, Firefox 42.0a1 ve Internet Explorer 11.0.8600.17814 ile test edilmiştir.

(bu siteye gömülü çalışmayabilir, yukarıdaki "önemli" notu okuyun)

function copyTextToClipboard(text) {
  var textArea = document.createElement("textarea");

  //
  // *** This styling is an extra step which is likely not required. ***
  //
  // Why is it here? To ensure:
  // 1. the element is able to have focus and selection.
  // 2. if element was to flash render it has minimal visual impact.
  // 3. less flakyness with selection and copying which **might** occur if
  //    the textarea element is not visible.
  //
  // The likelihood is the element won't even render, not even a
  // flash, so some of these are just precautions. However in
  // Internet Explorer the element is visible whilst the popup
  // box asking the user for permission for the web page to
  // copy to the clipboard.
  //

  // Place in top-left corner of screen regardless of scroll position.
  textArea.style.position = 'fixed';
  textArea.style.top = 0;
  textArea.style.left = 0;

  // Ensure it has a small width and height. Setting to 1px / 1em
  // doesn't work as this gives a negative w/h on some browsers.
  textArea.style.width = '2em';
  textArea.style.height = '2em';

  // We don't need padding, reducing the size if it does flash render.
  textArea.style.padding = 0;

  // Clean up any borders.
  textArea.style.border = 'none';
  textArea.style.outline = 'none';
  textArea.style.boxShadow = 'none';

  // Avoid flash of white box if rendered for any reason.
  textArea.style.background = 'transparent';


  textArea.value = text;

  document.body.appendChild(textArea);
  textArea.focus();
  textArea.select();

  try {
    var successful = document.execCommand('copy');
    var msg = successful ? 'successful' : 'unsuccessful';
    console.log('Copying text command was ' + msg);
  } catch (err) {
    console.log('Oops, unable to copy');
  }

  document.body.removeChild(textArea);
}


var copyBobBtn = document.querySelector('.js-copy-bob-btn'),
  copyJaneBtn = document.querySelector('.js-copy-jane-btn');

copyBobBtn.addEventListener('click', function(event) {
  copyTextToClipboard('Bob');
});


copyJaneBtn.addEventListener('click', function(event) {
  copyTextToClipboard('Jane');
});
<div style="display:inline-block; vertical-align:top;">
  <button class="js-copy-bob-btn">Set clipboard to BOB</button><br /><br />
  <button class="js-copy-jane-btn">Set clipboard to JANE</button>
</div>
<div style="display:inline-block;">
  <textarea class="js-test-textarea" cols="35" rows="4">Try pasting into here to see what you have on your clipboard:

  </textarea>
</div>

Ek Notlar

Yalnızca kullanıcı bir işlem yaparsa çalışır

Tüm document.execCommand('copy')aramalar, bir kullanıcı işleminin doğrudan bir sonucu olarak gerçekleştirilmelidir, ör. Click event handler. Bu, kullanıcının panoyu beklemediğinde karışmasını önlemek için bir önlemdir.

Bkz burada Google Developers yazı fazla bilgi için .

Pano API'sı

Clipboard API taslak spesifikasyonunun tamamını burada bulabilirsiniz: https://w3c.github.io/clipboard-apis/

Destekleniyor mu?

  • document.queryCommandSupported('copy')true"tarayıcı tarafından destekleniyorsa" komutu döndürülmelidir .
  • ve document.queryCommandEnabled('copy')iade trueeğer document.execCommand('copy')şimdi denilen eğer başarılı olur. Komutun kullanıcı tarafından başlatılan bir iş parçacığından çağrıldığından ve diğer gereksinimlerin karşılandığından emin olunması.

Ancak sadece döndü tarayıcı uyumluluk sorunları, ~ Nisan'dan ~ Ekim 2015 Google Chrome örneği olarak truegelen document.queryCommandSupported('copy')komutu bir kullanıcı tarafından başlatılan iplikten çağrıldı eğer.

Aşağıdaki uyumluluk ayrıntısına dikkat edin.

Tarayıcı Uyumluluğu Ayrıntısı

Basit bir arama olsa da document.execCommand('copy'), bir sarılmış try/catchBir kullanıcı tıklaması sonucunda çağrılan bloğuna size en uygunluğu sağlarken, aşağıdakilerin bazı koşulları vardır:

Herhangi bir çağrı document.execCommand, document.queryCommandSupportedya da document.queryCommandEnabledbir sarılmış olmalıdır try/ catchblok.

Farklı tarayıcı uygulamaları ve tarayıcı sürümleri, geri döndürmek yerine çağrıldığında farklı istisnalar oluşturur false.

Farklı tarayıcı uygulamaları hala akışta ve Pano API'sı hala taslak halinde, bu yüzden testinizi yapmayı unutmayın.


41
nasıl bir değişken veri .ie doğrudan kopyalamak için: var str = "word";?
jscripter

10
@BubuDaba <textarea>JS ile gizlenmiş bir kukla oluşturun , ona ekleyin document.body, değerini değişkene ayarlayın ve değerini kullanın ve copyTextareaardından içerik kopyalandıktan hemen sonra kaldırın.
SeinopSys

3
Safari için herhangi bir şey veya Safari'de uygulanacak herhangi bir gösterge var mı?
www139

3
Tüm tarayıcılarda çalışan bulduğum tek sürüm. Bunu Boostrap Modal'da kullanırken metin alanına modal eklemek zorunda kaldım. Eğer çözüm için yapabilseydim +1000 verirdim !!! TEŞEKKÜRLER!
Patrick

3
@AyaSalama, kilit nokta, kullanıcının eylemi gerçekleştirdiği tarayıcıya görünmediği sürece "kopya" işleminin yapılamayacağıdır. Öğe, "display: none" ile stilize edilmişse, eylemi göremeyeceği veya etkileşimde bulunamayacağı durumlarda kullanıcı işlemi gerçekleştiremez.
Dean Taylor

1256

Panoya otomatik kopyalama tehlikeli olabilir, bu nedenle çoğu tarayıcı (IE hariç) bunu çok zorlaştırır. Şahsen, aşağıdaki basit hileyi kullanıyorum:

function copyToClipboard(text) {
  window.prompt("Copy to clipboard: Ctrl+C, Enter", text);
}

Kullanıcıya, kopyalanacak metnin zaten seçildiği bilgi istemi kutusu görüntülenir. Şimdi Ctrl+ Cve Enter(kutuyu kapatmak için) - ve voila!

Pano kopyalama işlemi GÜVENLİDİR, çünkü kullanıcı elle yapar (ancak oldukça basit bir şekilde). Tabii ki, tüm tarayıcılarda çalışır.

<button id="demo" onclick="copyToClipboard(document.getElementById('demo').innerHTML)">This is what I want to copy</button>

<script>
  function copyToClipboard(text) {
    window.prompt("Copy to clipboard: Ctrl+C, Enter", text);
  }
</script>


91
Zeki, ama bu sadece tek bir satırı destekler.
Aram Kocharyan

61
"İstem" işlevini özel bir modda değiştirmek önemsizdir, hile etinin düzenlenebilir bir içerik alanı kullanması ve metni önceden seçmesi ve kullanıcının kullanıcı tarafından eylem kendilerini. A ++
Jon z

110
panoya kopyalamak için hala javascript kullanmıyor ^ _ ^
RozzA

23
Metniniz 2000 karakterden
fazlaysa

445
Garip bir soru cevap vermiyor iken 457 upvotes alırsınız: Javascript panoya kopyalayın !
stevenvh

298

Aşağıdaki yaklaşım Chrome, Firefox, Internet Explorer ve Edge ve Safari'nin son sürümlerinde çalışır (kopya desteği Ekim 2016'da yayınlanan sürüm 10'da eklenmiştir).

  • Bir metin alanı oluşturun ve içeriğini panoya kopyalanmasını istediğiniz metne ayarlayın.
  • Metin alanına DOM ekleyin.
  • Metin alanındaki metni seçin.
  • Document.execCommand ("kopyala") öğesini arayın
  • Metin alanını domdan kaldırın.

Not: Javascript kodunun aynı eşzamanlı çağrısında eklendiği ve kaldırıldığı için textarea'yı görmezsiniz.

Bunu kendiniz uyguluyorsanız dikkat etmeniz gereken bazı şeyler:

  • Güvenlik nedeniyle, bu yalnızca tıklama gibi bir olay işleyicisinden çağrılabilir (Pencereleri açarken olduğu gibi).
  • Internet Explorer, pano ilk kez güncelleştirildiğinde bir izin iletişim kutusu görüntüler.
  • Internet Explorer ve Edge, metin alanına odaklandığında kayar.
  • execCommand () bazı durumlarda atılabilir.
  • Metin alanı kullanmadığınız sürece, yeni satırlar ve sekmeler yutulabilir. (Çoğu makale bir div kullanmanızı önerir)
  • İnternet Explorer iletişim kutusu gösterilirken metin alanı görünür olacak, gizlemeniz veya Internet Explorer'a özgü clipboardData API'sını kullanmanız gerekecek.
  • Internet Explorer'da sistem yöneticileri pano API'sını devre dışı bırakabilir.

Aşağıdaki işlev, aşağıdaki sorunların tümünü olabildiğince temiz bir şekilde ele almalıdır. Herhangi bir sorun bulursanız veya iyileştirmek için herhangi bir öneriniz varsa lütfen bir yorum bırakın.

// Copies a string to the clipboard. Must be called from within an
// event handler such as click. May return false if it failed, but
// this is not always possible. Browser support for Chrome 43+,
// Firefox 42+, Safari 10+, Edge and Internet Explorer 10+.
// Internet Explorer: The clipboard feature may be disabled by
// an administrator. By default a prompt is shown the first
// time the clipboard is used (per session).
function copyToClipboard(text) {
    if (window.clipboardData && window.clipboardData.setData) {
        // Internet Explorer-specific code path to prevent textarea being shown while dialog is visible.
        return clipboardData.setData("Text", text);

    }
    else if (document.queryCommandSupported && document.queryCommandSupported("copy")) {
        var textarea = document.createElement("textarea");
        textarea.textContent = text;
        textarea.style.position = "fixed";  // Prevent scrolling to bottom of page in Microsoft Edge.
        document.body.appendChild(textarea);
        textarea.select();
        try {
            return document.execCommand("copy");  // Security exception may be thrown by some browsers.
        }
        catch (ex) {
            console.warn("Copy to clipboard failed.", ex);
            return false;
        }
        finally {
            document.body.removeChild(textarea);
        }
    }
}

https://jsfiddle.net/fx6a6n6x/


9
Güzel cevap: çapraz tarayıcı desteği, hata işleme + temizleme. QueryCommandSupported'ın bugünkü yeni desteği itibariyle, panoya kopyalamak artık Javascript'te mümkün ve bu garip 'window.prompt ("Panoya kopyala: Ctrl + C, Enter", metin)' geçici çözümü yerine kabul edilen cevap olmalıdır. window.clipboardData IE9'da desteklenir, bu yüzden tarayıcı destek listesine IE9 eklemeniz gerekir ve belki IE8 ve önceki sürümlerini de düşünürüm, ancak doğrulamanız gerekir.
user627283

Evet. IE 8/9 İyi olmalı. Uygulamamız onları desteklemiyor. Bu yüzden test etmedim. IE Ocak ayında desteği durdurur, bu yüzden çok huzursuz değilim. Umarım Safari desteği yakında inecektir. Eminim onların radarında.
Greg Lowe

4
@SantiagoCorredoira: 2016'da bu kabul edilen cevap olmayı hak ediyor. Lütfen BGT'yi (büyük yeşil onay işareti) yeniden atamayı düşünün.
Lawrence Dol

3
@Noitidart I Test edildi ve odak html belgesinde olmasa bile firefox 54, chrome 60 ve edge tarayıcısı için mükemmel çalışıyor, yaşadığınız hata muhtemelen FF 55
Tosin John

2
@Noitidart Burada hala mükemmel çalışıyor, dev araçlarına odaklanmak durmadı. Ve bu arada, normal bir web uygulaması kullanıcısı geliştirici araçlarında ne yapacak
Tosin John

97

İşte bunu benim almam ...

function copy(text) {
    var input = document.createElement('input');
    input.setAttribute('value', text);
    document.body.appendChild(input);
    input.select();
    var result = document.execCommand('copy');
    document.body.removeChild(input);
    return result;
 }

@korayem: Html inputalanı kullanmanın satır sonlarına saygı göstermeyeceğini \nve herhangi bir metni tek bir satıra düzleştireceğini unutmayın.

@Nikksan tarafından yorumlarda belirtildiği gibi, kullanmak textareasorunu şu şekilde çözecektir:

function copy(text) {
    var input = document.createElement('textarea');
    input.innerHTML = text;
    document.body.appendChild(input);
    input.select();
    var result = document.execCommand('copy');
    document.body.removeChild(input);
    return result;
}

@nikksan dize nasıl kopyalanır \n?
sof-03

2
@ sof-03 girdi yerine textarea kullanın ve \r\nsatır sonu için ekleyin
nikksan

1
Win10x64
Honsa

3
Cevabınızı kopyaladım. Kromda çalışıyor ve tek ihtiyacım olan bu.
user875234 15:18

Bu, Firefox v68.0.2 (64 bit) ile çalışan en basit çözümdür.
Arya

88

Gerçekten basit bir çözüm istiyorsanız (entegre edilmesi 5 dakikadan az sürer) ve kutudan çıkar çıkmaz iyi görünüyorsa, Clippy bazı daha karmaşık çözümlere hoş bir alternatiftir.

GitHub'ın kurucu ortağı tarafından yazılmıştır. Aşağıdaki Flash yerleştirme kodu örneği:

<object
   classid="clsid:d27cdb6e-ae6d-11cf-96b8-444553540000"
   width="110"
   height="14"
   id="clippy">
  <param name="movie" value="/flash/clippy.swf"/>
  <param name="allowScriptAccess" value="always"/>
  <param name="quality" value="high"/>
  <param name="scale" value="noscale"/>
  <param NAME="FlashVars" value="text=#{text}"/>
  <param name="bgcolor" value="#{bgcolor}"/>
  <embed
     src="/flash/clippy.swf"
     width="110"
     height="14"
     name="clippy"
     quality="high"
     allowScriptAccess="always"
     type="application/x-shockwave-flash"
     pluginspage="http://www.macromedia.com/go/getflashplayer"
     FlashVars="text=#{text}"
     bgcolor="#{bgcolor}"/>
</object>

Kopyalamanız gereken #{text}metinle ve #{bgcolor}bir renkle değiştirmeyi unutmayın .


12
İlgilenen herkes için, repo URL'sini kopyalarken GitHub'da kullanılan Clippy'yi kontrol edin.
Radek

66
Bilginize, GitHub'da Clippy kullanımı ZeroClipboard ile değiştirildi.
James M. Greene

219
OP, JavaScript'te bir çözüm istedi. Flaş değil.
MT.

21
@MT, "javascript" ile bazı insanlar "tarayıcı istemcisinde" anlamına gelir, bu nedenle yalnızca JS bir gereksinim olsa da, bu yanıtı alma şansı olan çoğu kişi gerçekten JS veya diğer yaygın olarak desteklenen istemci teknoloji. Flash tüm platformlara çarpmaz, ancak pano desteği gibi bir lehçe özelliği için, UX'i bir pop-up iletişim kutusundan (kesinlikle yaptığı) geliştirirse eklemeye değer.
Dave Dopson

13
Şu anda Flash'a güvenmek, web sitesi geliştiren neredeyse herkes tarafından kabul edilemeyecek bir şey olan site ziyaretçilerinin yüzdesi için işe yaramayacağı anlamına geliyor.
jinglesthula

86

Panoyu bir web sayfasından okumak ve değiştirmek güvenlik ve gizlilikle ilgili endişeleri artırır. Ancak, Internet Explorer'da bunu yapmak mümkündür. Bu örnek snippet'i buldum :

    <script type="text/javascript">
        function select_all(obj) {
            var text_val=eval(obj);
            text_val.focus();
            text_val.select();
            r = text_val.createTextRange();
            if (!r.execCommand) return; // feature detection
            r.execCommand('copy');
        }
    </script>
    <input value="http://www.sajithmr.com"
     onclick="select_all(this)" name="url" type="text" />


7
Basit bir kopyalama işlemi için flaş kullanmak aşırıya kaçmış gibi görünüyor, bunu yapmak için temiz bir JS yolu vardı sevindim. Ve bir şirket env. IE gayet iyi. Teşekkürler Bandi!
Eddie

5
plz açıklamak ne execCommand(\\’copy\\’);IE için panoya kopyalamak değil mi? @mrBorna
RozzA

20
Kullanmayın if(!document.all)ama if(!r.execCommand)başkasının uygulayamaması için! Document.all kesinlikle bununla ilgili değildir.
m93a

1
Adamım, bu basit ve temiz kod hakkında Sevdiğim şey, neredeyse hiç küçük bakım ile çalışır. Bu benim için yaptı, güzel çalışıyor.
Samuel Ramzan

1
en son krom, firefox veya MS Edge'de
çalışmaz

69

Geçenlerde bu çok sorun üzerinde bir teknik blog yazısı yazdım (Lucidchart'ta çalışıyorum ve son zamanlarda panomuzda bir revizyon yaptık).

Düz metni panoya kopyalamak, sistem kopyalama olayı sırasında yapmak istediğinizi varsayarsak (kullanıcı CtrlCtarayıcı menüsüne basar veya kullanır).

var isIe = (navigator.userAgent.toLowerCase().indexOf("msie") != -1 
           || navigator.userAgent.toLowerCase().indexOf("trident") != -1);

document.addEventListener('copy', function(e) {
    var textToPutOnClipboard = "This is some text";
    if (isIe) {
        window.clipboardData.setData('Text', textToPutOnClipboard);    
    } else {
        e.clipboardData.setData('text/plain', textToPutOnClipboard);
    }
    e.preventDefault();
});

Bir sistem kopyalama olayı sırasında panoya metin koymak çok daha zordur. Bu diğer cevapların bazıları, bunu yapmanın tek çapraz tarayıcı yolu olan Flash aracılığıyla yapmanın yollarını referans alıyor gibi görünüyor (anladığım kadarıyla).

Bunun dışında, tarayıcıdan tarayıcıya bazı seçenekler vardır.

Bu, IE'deki en basit olanıdır. ClipboardData nesnesine istediğiniz zaman JavaScript'ten şu yolla erişebilirsiniz:

window.clipboardData

(Ancak bunu bir sistem kesme, kopyalama veya yapıştırma olayının dışında yapmaya çalıştığınızda, IE kullanıcıdan web uygulaması panosu izni vermesini ister.)

Chrome'da, size pano izinleri verecek bir Chrome uzantısı oluşturabilirsiniz (Lucidchart için yaptığımız şey budur). Ardından, uzantınızın yüklü olduğu kullanıcılar için sistem etkinliğini kendiniz başlatmanız yeterlidir:

document.execCommand('copy');

Firefox, kullanıcıların panoya erişmek için belirli sitelere izin vermesine izin veren bazı seçeneklere sahip gibi görünüyor , ancak bunların hiçbirini şahsen denemedim.


2
Blog yazısında belirtilmemiş (yakın gelecekte güncellemeyi umuyorum), execCommand kullanarak kesme ve kopyalama tetikleme yeteneğidir. Bu, IE10 +, Chrome 43+ ve Opera29 + 'da desteklenir. Burada okuyun. updates.html5rocks.com/2015/04/cut-and-copy-commands
Richard Shurtz

Bununla ilgili bir sorun, diğer normal kopya olaylarını ele geçirmesidir.
Brock Adams

NB! Bu tarayıcı koklama KÖTÜ. Koklama özelliği. IE'nin güncellenmesini zorlaştırıyorsunuz.
odinho - Velmont

51

clipboard.js , metin veya HTML verilerinin panoya kopyalanmasına izin veren küçük, Flash olmayan bir yardımcı programdır. Kullanımı çok kolay, sadece .js'yi ekleyin ve böyle bir şey kullanın:

<button id='markup-copy'>Copy Button</button>

<script>
document.getElementById('markup-copy').addEventListener('click', function() {
  clipboard.copy({
    'text/plain': 'Markup text. Paste me into a rich text editor.',
    'text/html': '<i>here</i> is some <b>rich text</b>'
  }).then(
    function(){console.log('success'); },
    function(err){console.log('failure', err);
  });

});
</script>

clipboard.js ayrıca GitHub'da .

Not: Bu artık kullanımdan kaldırıldı. Buraya göç edin .


Bu kütüphane angular.io tarafından Kahraman Turu için kullanılır ve kullanıcının sadece kopyalaması gereken önceden seçilmiş bir metni görüntüleyerek execCommand'ı desteklemeyen tarayıcı için zarif modda geri dönüş sağlar.
John-Philip

1
Clipboard.js değiştirildi veya değiştirildi, ancak yaşıyor gibi görünüyor ve aktif olarak korunuyor npmjs.com/package/clipboard
Joao

35

ZeroClipboard, bulduğum en iyi çapraz tarayıcı çözümüdür:

<div id="copy" data-clipboard-text="Copy Me!">Click to copy</div>    
<script src="ZeroClipboard.js"></script>
<script>
  var clip = new ZeroClipboard( document.getElementById('copy') );
</script>

İOS için flash olmayan desteğe ihtiyacınız varsa bir geri dönüş eklemeniz yeterlidir:

clip.on( 'noflash', function ( client, args ) {
    $("#copy").click(function(){            
        var txt = $(this).attr('data-clipboard-text');
        prompt ("Copy link, then click OK.", txt);
    });
});  

http://zeroclipboard.org/

https://github.com/zeroclipboard/ZeroClipboard


25
Flash ile çapraz tarayıcı? iOS ve Android 4.4'te çalışmıyor
Raptor

1
Güncellenmiş cevaba bakınız. Bu, flash kullanıcıları için daha az adım ve herkes için geri dönüş sağlar.
Justin

8
milyarlarca kod satırı vardır. kesinlikle alay. hiç böyle bir canavarı bir projeye dahil etmekten daha iyi değil
vsync

2
74k sürümünde tüm çan ve ıslıklara sahip olmayan 20K olan gist.github.com/JamesMGreene/8698897 basit bir sürümü var. İkisi de çok büyük değil. Benim tahminim çoğu kullanıcı, 74k veya 20k'lik bir dosyanın indireceği ekstra milisaniye ile iyi olduğunu, böylece kopyala / yapıştırın iki yerine bir tıklama olduğunu.
Justin

@Justin Örnekleri kopyalayıp yapıştırsam bile, yerel olarak çalışmasını sağlayamıyorum (minimum değişiklikler yapıyorum, örneğin srckomut dosyası etiketlerinin değeri ). Belgelerinin güzel ama verimsiz olduğunu hissediyorum.
Gui Imamura

29

2018'de, bunu nasıl başarabileceğiniz aşağıda açıklanmıştır:

async copySomething(text?) {
  try {
    const toCopy = text || location.href;
    await navigator.clipboard.writeText(toCopy);
    console.log('Text or Page URL copied');
  }
  catch (err) {
    console.error('Failed to copy: ', err);
  }
}

Benim Angular 6+ kodumda şöyle kullanılır:

<button mat-menu-item (click)="copySomething()">
    <span>Copy link</span>
</button>

Bir dizeyi geçersem kopyalar. Hiçbir şey yoksa, sayfanın URL'sini kopyalar.

Panoya daha fazla jimnastik de yapılabilir. Daha fazla bilgiyi burada bulabilirsiniz:

Pano Erişimini Engelleme


localhost ile bağlandın
Joe Warner

2
Bunun Safari'de çalışmadığını lütfen unutmayın (sürüm 11.1.2)
arjun27

1
Umarım bir gün Apple yetişir. Bu caniuse.com/#feat=clipboard , kısmen desteklendiğinden bahsettiğiniz yukarıdaki sürümü göstermesine rağmen .
KhoPhi

2
Her iki işlevi de okuyorum readText, writeText Reddedilen durumda bir Promise
ramin

3
Sağlanan bağlantıya göre, "navigator.clipboard yalnızca HTTPS üzerinden sunulan sayfalar için desteklenmektedir"
TimH - Codidact 20:18

26

Üzerinde çalıştığım projelerden birinde, Zero Clipboard kitaplığını kullanan bir jQuery panoya kopyala eklentisi .

Ağır bir jQuery kullanıcısıysanız, yerel Sıfır Pano eklentisinden daha kolay kullanılır.


6
92kb gerçekten o kadar büyük değil, hızlı çalışıyor ve isterseniz text()yerine kullanabilirsiniz innerHTML()..
RozzA

17
@John: innerHTMLtarayıcılar uzun zamandır destekleniyor. Microsoft'un aslında fikri güvence altına almaması, onu güvenilmez veya tescilli yapmaz. Aynı zamanda nihayet resmi spesifikasyona ekleniyor (her büyük tarayıcı satıcısı zaten bunun için destek ekledikten sonra ... iç çek ).
James M. Greene

19
@John Flash kullanan bir cevapta jQuery yeterince JavaScripty olmamasından şikayet ediyorsun;)
Max Nanasy

4
innerHTML çoğu durumda alternatiflerden daha iyidir. Yüksek atından in! Daha hızlı, daha verimlidir ve bir sayfanın yeniden oluşturulmasını gerektirmez.
Yörünge Eden

4
@RozzA 92KBgerçekten büyük. LTE olgunlaşana kadar GPRS , WW mobil veri standardıdır ve başlar 1 KB/s. Matematiği kendiniz yapın.
Tino

23

Chrome 42+ ve Firefox 41+ artık document.execCommand ('copy') komutunu destekliyor . Bu yüzden Tim Down'ın eski yanıtı ve Google Developer'ın cevabının bir kombinasyonunu kullanarak tarayıcılar arası bir panoya kopyalama özelliği için birkaç işlev oluşturdum :

function selectElementContents(el) {
    // Copy textarea, pre, div, etc.
    if (document.body.createTextRange) {
        // IE
        var textRange = document.body.createTextRange();
        textRange.moveToElementText(el);
        textRange.select();
        textRange.execCommand("Copy");
    } 
    else if (window.getSelection && document.createRange) {
        // Non-Internet Explorer
        var range = document.createRange();
        range.selectNodeContents(el);
        var sel = window.getSelection();
        sel.removeAllRanges();
        sel.addRange(range);
        try {
            var successful = document.execCommand('copy');
            var msg = successful ? 'successful' : 'unsuccessful';
            console.log('Copy command was ' + msg);
        }
        catch (err) {
            console.log('Oops, unable to copy');
        }
    }
} // end function selectElementContents(el)

function make_copy_button(el) {
    var copy_btn = document.createElement('input');
    copy_btn.type = "button";
    el.parentNode.insertBefore(copy_btn, el.nextSibling);
    copy_btn.onclick = function() {
        selectElementContents(el);
    };

    if (document.queryCommandSupported("copy") || parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2]) >= 42) {
        // Copy works with Internet Explorer 4+, Chrome 42+, Firefox 41+, Opera 29+
        copy_btn.value = "Copy to Clipboard";
    }
    else {
        // Select only for Safari and older Chrome, Firefox and Opera
        copy_btn.value = "Select All (then press Ctrl + C to Copy)";
    }
}
/* Note: document.queryCommandSupported("copy") should return "true" on browsers that support copy
    but there was a bug in Chrome versions 42 to 47 that makes it return "false".  So in those
    versions of Chrome feature detection does not work!
    See https://code.google.com/p/chromium/issues/detail?id=476508
*/

make_copy_button(document.getElementById("markup"));
<pre id="markup">
  Text that can be copied or selected with cross browser support.
</pre>


Bunu özetlediğiniz için teşekkürler! Kodunuzda küçük hatalar var: "range" değişkenini iki kez tanımladınız (var range = document.createRange ()).
Christian Engel

1
Doğru @ChristianEngel. İkincisini çıkardım. Oraya nasıl girdiğini bilmiyorum.
Jeff Baker

23

Bunu çok başarılı bir şekilde kullanıyorum ( jQuery veya başka bir çerçeve olmadan ).

function copyToClp(txt){
    txt = document.createTextNode(txt);
    var m = document;
    var w = window;
    var b = m.body;
    b.appendChild(txt);
    if (b.createTextRange) {
        var d = b.createTextRange();
        d.moveToElementText(txt);
        d.select();
        m.execCommand('copy');
    } 
    else {
        var d = m.createRange();
        var g = w.getSelection;
        d.selectNodeContents(txt);
        g().removeAllRanges();
        g().addRange(d);
        m.execCommand('copy');
        g().removeAllRanges();
    }
    txt.remove();
}

Uyarı

Sekmeler boşluklara dönüştürülür (en azından Chrome'da).


Bu yaklaşımda boşluklar eksik
Bikram

1
Krom. sekmeler bir boşluğa dönüştürülür
Bikram

22

Aşağıdaki çözümü buldum:

Anahtar aşağı işleyicide "ön" etiketi oluşturur. İçeriği bu etikete kopyalanacak şekilde ayarladıktan sonra bu etikette bir seçim yaparız ve işleyicide true değerini döndürürüz. Bu, standart krom işleyicisini çağırır ve seçilen metni kopyalar.

Ve gerekirse, önceki seçimi geri yükleme işlevi için zaman aşımı ayarlanmış olabilir. Mootools uygulamam:

   function EnybyClipboard() {
     this.saveSelection = false;
     this.callback = false;
     this.pastedText = false;

     this.restoreSelection = function() {
       if (this.saveSelection) {
         window.getSelection().removeAllRanges();
         for (var i = 0; i < this.saveSelection.length; i++) {
           window.getSelection().addRange(this.saveSelection[i]);
         }
         this.saveSelection = false;
       }
     };

     this.copyText = function(text) {
       var div = $('special_copy');
       if (!div) {
         div = new Element('pre', {
           'id': 'special_copy',
           'style': 'opacity: 0;position: absolute;top: -10000px;right: 0;'
         });
         div.injectInside(document.body);
       }
       div.set('text', text);
       if (document.createRange) {
         var rng = document.createRange();
         rng.selectNodeContents(div);
         this.saveSelection = [];
         var selection = window.getSelection();
         for (var i = 0; i < selection.rangeCount; i++) {
           this.saveSelection[i] = selection.getRangeAt(i);
         }
         window.getSelection().removeAllRanges();
         window.getSelection().addRange(rng);
         setTimeout(this.restoreSelection.bind(this), 100);
       } else return alert('Copy not work. :(');
     };

     this.getPastedText = function() {
       if (!this.pastedText) alert('Nothing to paste. :(');
       return this.pastedText;
     };

     this.pasteText = function(callback) {
       var div = $('special_paste');
       if (!div) {
         div = new Element('textarea', {
           'id': 'special_paste',
           'style': 'opacity: 0;position: absolute;top: -10000px;right: 0;'
         });
         div.injectInside(document.body);
         div.addEvent('keyup', function() {
           if (this.callback) {
             this.pastedText = $('special_paste').get('value');
             this.callback.call(null, this.pastedText);
             this.callback = false;
             this.pastedText = false;
             setTimeout(this.restoreSelection.bind(this), 100);
           }
         }.bind(this));
       }
       div.set('value', '');
       if (document.createRange) {
         var rng = document.createRange();
         rng.selectNodeContents(div);
         this.saveSelection = [];
         var selection = window.getSelection();
         for (var i = 0; i < selection.rangeCount; i++) {
           this.saveSelection[i] = selection.getRangeAt(i);
         }
         window.getSelection().removeAllRanges();
         window.getSelection().addRange(rng);
         div.focus();
         this.callback = callback;
       } else return alert('Fail to paste. :(');
     };
   }

Kullanımı:

enyby_clip = new EnybyClipboard(); //init 

enyby_clip.copyText('some_text'); // place this in CTRL+C handler and return true;

enyby_clip.pasteText(function callback(pasted_text) {
        alert(pasted_text);
}); // place this in CTRL+V handler and return true;

Hamur üzerine textarea oluşturur ve aynı şekilde çalışır.

PS olabilir bu çözüm flaş olmadan tamamen çapraz tarayıcı çözümü oluşturmak için kullanılabilir. FF ve Chrome'da çalışır.


2
Bunu deneyen var mı? Gerçekten bir dizi tarayıcıda çalışması durumunda şık bir şey gibi görünüyor!
Michael

1
jsfiddle.net/H2FHC Demo: fiddle.jshell.net/H2FHC/show Lütfen açın ve Ctrl + V veya Ctrl + C tuşlarına basın. FF 19.0'da çatal mükemmel. Chrome'da da 25.0.1364.97 m. Opera 12.14 - Tamam. Windows için Safari 5.1.7 - Tamam. IE - BAŞARISIZ.
Enyby

IE için sayfa içindeki öğeye odaklanmanız gerekir. Bkz. Fiddle.jshell.net/H2FHC/3/show ve fiddle.jshell.net/H2FHC/3 IE'de çalıştı 9/10. IE 6/7, document.createRange desteklenmediğinden, işlemin başka bir şekilde seçim oluşturması gerekir.
Enyby

21

Diğer yöntemler düz metni panoya kopyalar. HTML'yi kopyalamak için (yani, sonuçları bir WSIWYG düzenleyicisine yapıştırabilirsiniz), SADECE IE'de aşağıdakileri yapabilirsiniz . Tarayıcı aslında içeriği görsel olarak seçtiğinden, bu temel olarak diğer yöntemlerden farklıdır.

// create an editable DIV and append the HTML content you want copied
var editableDiv = document.createElement("div");
with (editableDiv) {
    contentEditable = true;
}     
editableDiv.appendChild(someContentElement);          

// select the editable content and copy it to the clipboard
var r = document.body.createTextRange();
r.moveToElementText(editableDiv);
r.select();  
r.execCommand("Copy");

// deselect, so the browser doesn't leave the element visibly selected
r.moveToElementText(someHiddenDiv);
r.select();   

burada daha fazla HTML çözümü görebilirsiniz stackoverflow.com/questions/34191780/…
kofifus

21

En iyisi olduğunu düşündüğüm şeyi bir araya getirdim.

  • Doğrudan stile karşılık Internet Explorer'da istisnaları önlemek için cssText öğesini kullanır.
  • Varsa seçimi geri yükler
  • Klavye mobil cihazlarda görünmeyecek şekilde salt okunur olarak ayarlanır
  • İOS için bir geçici çözümü vardır, böylece normalde execCommand'ı engellediğinden çalışır.

İşte burada:

const copyToClipboard = (function initClipboardText() {
  const textarea = document.createElement('textarea');

  // Move it off screen.
  textarea.style.cssText = 'position: absolute; left: -99999em';

  // Set to readonly to prevent mobile devices opening a keyboard when
  // text is .select()'ed.
  textarea.setAttribute('readonly', true);

  document.body.appendChild(textarea);

  return function setClipboardText(text) {
    textarea.value = text;

    // Check if there is any content selected previously.
    const selected = document.getSelection().rangeCount > 0 ?
      document.getSelection().getRangeAt(0) : false;

    // iOS Safari blocks programmtic execCommand copying normally, without this hack.
    // /programming/34045777/copy-to-clipboard-using-javascript-in-ios
    if (navigator.userAgent.match(/ipad|ipod|iphone/i)) {
      const editable = textarea.contentEditable;
      textarea.contentEditable = true;
      const range = document.createRange();
      range.selectNodeContents(textarea);
      const sel = window.getSelection();
      sel.removeAllRanges();
      sel.addRange(range);
      textarea.setSelectionRange(0, 999999);
      textarea.contentEditable = editable;
    }
    else {
      textarea.select();
    }

    try {
      const result = document.execCommand('copy');

      // Restore previous selection.
      if (selected) {
        document.getSelection().removeAllRanges();
        document.getSelection().addRange(selected);
      }

      return result;
    }
    catch (err) {
      console.error(err);
      return false;
    }
  };
})();

Kullanımı: copyToClipboard('some text')


13

Flash 10'dan itibaren panoya yalnızca eylem bir Flash nesnesiyle kullanıcı etkileşiminden kaynaklanıyorsa kopyalayabilirsiniz. ( Adobe'nin Flash 10 duyurusu ile ilgili bölümü okuyun )

Çözüm, Kopyala düğmesinin üzerinde bir flash nesnesini veya kopyayı başlatan herhangi bir öğeyi aşırı derecede büyütmektir. Sıfır Pano şu anda bu uygulama ile en iyi kütüphanedir. Deneyimli Flash geliştiricileri sadece kendi kütüphanelerini yapmak isteyebilirler.


12

  <!DOCTYPE html>

  <style>
    #t {
      width: 1px
      height: 1px
      border: none
    }
    #t:focus {
      outline: none
    }
  </style>

  <script>
    function copy(text) {
      var t = document.getElementById('t')
      t.innerHTML = text
      t.select()
      try {
        var successful = document.execCommand('copy')
        var msg = successful ? 'successfully' : 'unsuccessfully'
        console.log('text coppied ' + msg)
      } catch (err) {
        console.log('Unable to copy text')
      }
      t.innerHTML = ''
    }
  </script>

  <textarea id=t></textarea>

  <button onclick="copy('hello world')">
    Click me
  </button>


En iyi cevap: D, şu şekilde geliştirebilirsiniz: #t {position: absolute; sol: 0; z-endeksi: -900; genişlik: 0 piksel; yükseklik: 0px; sınır: yok; } Yani tamamen gizli olacak! Ama gerçekten teşekkürler kardeşim!
Federico Navarrete

#t {resize: none;}
SmartManoj

Bir açıklama yapılabilir.
Peter Mortensen

12

Aşağıdaki çözümü buldum:

Gizli bir girişte metin var. Çünkü setSelectionRangegizli girdiler üzerinde çalışmaya değil, ben metne geçici türü değişti metni kopyalanan ve sonra tekrar gizli yaptı. Bir öğeden metni kopyalamak isterseniz, işleve aktarabilir ve içeriğini hedef değişkene kaydedebilirsiniz.

jQuery('#copy').on('click', function () {
    copyToClipboard();
});

function copyToClipboard() {
    var target = jQuery('#hidden_text');

    // Make it visible, so can be focused
    target.attr('type', 'text');
    target.focus();
    // Select all the text
    target[0].setSelectionRange(0, target.val().length);

    // Copy the selection
    var succeed;
    try {
        succeed = document.execCommand("copy");
    }
    catch (e) {
        succeed = false;
    }

    // Hide input again
    target.attr('type', 'hidden');

    return succeed;
}

11

HTML girişinden panoya metin kopyalayın:

 function myFunction() {
   /* Get the text field */
   var copyText = document.getElementById("myInput");

   /* Select the text field */
   copyText.select();

   /* Copy the text inside the text field */
   document.execCommand("Copy");

   /* Alert the copied text */
   alert("Copied the text: " + copyText.value);
 }
 <!-- The text field -->
 <input type="text" value="Hello Friend" id="myInput">

 <!-- The button used to copy the text -->
<button onclick="myFunction()">Copy text</button>

Not: yöntem Internet Explorer 9'da desteklenen ve daha önce değil.document.execCommand()

Kaynak : W3Schools - Metni Panoya Kopyala


11

Zaten birçok cevap var, ancak bir tane eklemek ister (jQuery). Herhangi bir tarayıcıda, mobil tarayıcılarda da harika çalışır (yani güvenlik hakkında bilgi ister, ancak kabul ettiğinizde iyi çalışır).

function appCopyToClipBoard(sText)
{
    var oText = false,
        bResult = false;
    try
    {
        oText = document.createElement("textarea");
        $(oText).addClass('clipboardCopier').val(sText).insertAfter('body').focus();
        oText.select();
        document.execCommand("Copy");
        bResult = true;
    }
    catch(e) {
    }

    $(oText).remove();
    return bResult;
}

Kodunuzda:

if (!appCopyToClipBoard('Hai there! This is copied to the clipboard.'))
{
    alert('Sorry, copy to clipboard failed.');
}

9

Bu, diğer cevaplar arasında bir kombinasyon.

var copyToClipboard = function(textToCopy){
    $("body")
        .append($('<textarea name="fname" class="textToCopyInput"/>' )
        .val(textToCopy))
        .find(".textToCopyInput")
        .select();
      try {
        var successful = document.execCommand('copy');
        var msg = successful ? 'successful' : 'unsuccessful';
        alert('Text copied to clipboard!');
      } catch (err) {
          window.prompt("To copy the text to clipboard: Ctrl+C, Enter", textToCopy);
      }
     $(".textToCopyInput").remove();
}

JQuery kullanır, ancak elbette olması gerekmez. İsterseniz bunu değiştirebilirsiniz. Sadece elime jQuery vardı. Girişin gösterilmediğinden emin olmak için biraz CSS de ekleyebilirsiniz. Örneğin:

.textToCopyInput{opacity: 0; position: absolute;}

Ya da tabii ki bazı satır içi stiller de yapabilirsiniz

.append($('<textarea name="fname" style="opacity: 0;  position: absolute;" class="textToCopyInput"/>' )

Değişken verilerden doğrudan kopyalama .ie: var str = "word"; ?

Değişken msg kullanılmıyor
Voyager

Kullanmak daha iyi '<textarea class = "textToCopyInput" /> </textarea>' textToCopyiçeriyorsa\n
Voyager


8

(Excel gibi bir şey) özel bir ızgara düzenleme oluşturma ve Excel ile uyumluluk aynı sorun vardı. Birden fazla hücre seçmeyi, kopyalamayı ve yapışmayı desteklemem gerekiyordu.

Çözüm: Kullanıcının kopyalaması için veri ekleyeceğiniz bir metin alanı oluşturun (kullanıcı hücreleri seçerken benim için), odağı ayarlayın (örneğin, kullanıcı bastığında Ctrl) ve metnin tamamını seçin.

Böylece, kullanıcı Ctrl+ tuşuna bastığında Ckopyaladığı hücreleri seçer. Test ettikten sonra sadece bir piksele textarea yeniden boyutlandırma (Ben ekranda çalışıp çalışmadığını test etmedi: yok). Tüm tarayıcılarda güzel çalışır ve kullanıcı için şeffaftır.

Yapıştırma - bunun gibi yapabilirsiniz (hedefinize göre değişir) - textarea'ya odaklanın ve onpaste kullanarak macun olaylarını yakalayın (projemde düzenlemek için hücrelerde textareas kullanıyorum).

Bir örnek (ticari proje) yapıştıramıyorum, ama fikri anladınız.



6

Bu, @ Chase'in cevabının bir genişlemesidir, avantajı, yalnızca IE9'daki DIV'ler için değil, IMAGE ve TABLE öğeleri için çalışacaktır.

if (document.createRange) {
    // IE9 and modern browsers
    var r = document.createRange();
    r.setStartBefore(to_copy);
    r.setEndAfter(to_copy);
    r.selectNode(to_copy);
    var sel = window.getSelection();
    sel.addRange(r);
    document.execCommand('Copy');  // does nothing on FF
} else {
    // IE 8 and earlier.  This stuff won't work on IE9.
    // (unless forced into a backward compatibility mode,
    // or selecting plain divs, not img or table). 
    var r = document.body.createTextRange();
    r.moveToElementText(to_copy);
    r.select()
    r.execCommand('Copy');
}

5

Soruyu yanlış okudum, ancak referans için, bir dizi DOM'yi (panoya değil; tüm modern tarayıcılarla uyumlu) ayıklayabilir ve pano davranışı almak için onkopi ve onpaste ve onbeforepaste olaylarıyla birleştirebilirsiniz. İşte bunu başarmak için kod:

function clipBoard(sCommand) {
  var oRange=contentDocument.createRange();
  oRange.setStart(startNode, startOffset);
  oRange.setEnd(endNode, endOffset);
/* This is where the actual selection happens.
in the above, startNode and endNode are dom nodes defining the beginning 
and end of the "selection" respectively. startOffset and endOffset are 
constants that are defined as follows:

END_TO_END: 2
END_TO_START: 3
NODE_AFTER: 1
NODE_BEFORE: 0
NODE_BEFORE_AND_AFTER: 2
NODE_INSIDE: 3
START_TO_END: 1
START_TO_START: 0

and would be used like oRange.START_TO_END */
      switch(sCommand) {
    case "cut":
          this.oFragment=oRange.extractContents();
      oRange.collapse();
      break;
    case "copy":
      this.oFragment=oRange.cloneContents();
      break;
    case "paste":
      oRange.deleteContents();
      var cloneFragment=this.oFragment.cloneNode(true)
      oRange.insertNode(cloneFragment);
      oRange.collapse();
      break;
  }
}

1
aslında kodu düzelttim. Tüm tarayıcılarda çalışır ancak panoya kopyalanmaz. Sadece değişkenler aracılığıyla içeriği çıkarır (keser), klonlar (kopyalar). Görünüşe göre kullanımı unuttum.
mrBorna

5

Benim hatam. Bu yalnızca IE'de çalışır.

İşte metni kopyalamanın başka bir yolu:

<p>
    <a onclick="window.clipboardData.setData('text', document.getElementById('Test').innerText);">Copy</a>
</p>

9
Bu, mevcut Chrome (V31) veya FireFox'ta (v25) çalışmaz. Hata, window.clipboardData öğesinin tanımsız olmasıdır. Artı tarafta, IE9'da çalışır. Bu nedenle, iyi tarayıcıları önemsemediğiniz ve sitenizi kötü amaçlı tarayıcılar kullanarak kilitlemek istediğiniz sürece, bunu yapmanın yolu budur!
Anthony

2
neden bu kadar aptalca cevaplar anlamıyorum. w3schools.com/howto/tryit.asp?filename=tryhow_js_copy_clipboard
Martian2049

5

İnternette çeşitli yollara baktıktan sonra, şimdiye kadar çalıştığım tek şey buydu. Bu dağınık bir konu. Orada dünya çapında yayınlanan pek çok çözüm vardır ve bunların çoğu yapmak değil işi. Bu benim için çalıştı:

NOT: Bu kod yalnızca 'onClick' yöntemi gibi bir şeye doğrudan senkron kod olarak yürütüldüğünde çalışır. Ajax'a eşzamansız bir yanıtla veya başka bir eşzamansız şekilde çağırırsanız çalışmaz.

copyToClipboard(text) {
    var copyText = document.createElement("input");
    copyText.type = "text";
    document.body.appendChild(copyText);
    copyText.style = "display: inline; width: 1px;";
    copyText.value = text;
    copyText.focus();
    document.execCommand("SelectAll");
    document.execCommand("Copy");
    copyText.remove();
}

Bu kod bir milisaniye ekranda 1 piksel genişliğinde bir bileşen görünür bir şekilde gösterecektir, ama bu konuda endişelenmemeye karar verdi, ki bu gerçek bir sorun varsa başkaları ele olabilir bir şeydir.


5

Seçilen bir metni ('Kopyalanacak Metin') panonuza kopyalamak için bir Yer İşareti (JavaScript çalıştıran tarayıcı yer işareti) oluşturun ve yürütün (üzerine tıklayın). Geçici bir metin oluşturur.

GitHub'dan kod:

https://gist.github.com/stefanmaric/2abf96c740191cda3bc7a8b0fc905a7d

(function (text) {
  var node = document.createElement('textarea');
  var selection = document.getSelection();

  node.textContent = text;
  document.body.appendChild(node);

  selection.removeAllRanges();
  node.select();
  document.execCommand('copy');

  selection.removeAllRanges();
  document.body.removeChild(node);
})('Text To Copy');
Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.