Byla vydána (Mastodon, 𝕏) nová stabilní verze 2.10.38 svobodné aplikace pro úpravu a vytváření rastrové grafiky GIMP (GNU Image Manipulation Program). Přehled novinek v oznámení o vydání a v souboru NEWS na GitLabu. Nový GIMP je již k dispozici také na Flathubu.
Google zveřejnil seznam 1220 projektů od 195 organizací (Debian, GNU, openSUSE, Linux Foundation, Haiku, Python, …) přijatých do letošního, již dvacátého, Google Summer of Code.
Na základě DMCA požadavku bylo na konci dubna z GitHubu odstraněno 8535 repozitářů se zdrojovými kódy open source emulátoru přenosné herní konzole Nintendo Switch yuzu.
Webový prohlížeč Dillo (Wikipedie) byl vydán ve verzi 3.1.0. Po devíti letech od vydání předchozí verze 3.0.5. Doména dillo.org již nepatří vývojářům Dilla.
O víkendu probíhá v Bostonu, a také virtuálně, konference LibrePlanet 2024 organizovaná nadací Free Software Foundation (FSF).
Nová vývojová verze Wine 9.8 řeší mimo jiné chybu #3689 při instalaci Microsoft Office 97 nahlášenou v roce 2005.
Coppwr, tj. GUI nástroj pro nízkoúrovňové ovládání PipeWire, byl vydán v nové verzi 1.6.0. Zdrojové kódy jsou k dispozici na GitHubu. Instalovat lze také z Flathubu.
Byla vydána dubnová aktualizace aneb nová verze 1.89 editoru zdrojových kódů Visual Studio Code (Wikipedie). Přehled novinek i s náhledy a animovanými gify v poznámkách k vydání. Vypíchnout lze, že v terminálu lze nově povolit vkládání kopírovaného textu stisknutím středního tlačítka myši. Ve verzi 1.89 vyjde také VSCodium, tj. komunitní sestavení Visual Studia Code bez telemetrie a licenčních podmínek Microsoftu.
Proton, tj. fork Wine integrovaný v Steam Play a umožňující v Linuxu přímo ze Steamu hrát hry určené pouze pro Windows, byl vydán ve verzi 9.0-1 (𝕏). Přehled novinek se seznamem nově podporovaných her na GitHubu. Aktuální přehled her pro Windows běžících díky Protonu také na Linuxu na stránkách ProtonDB.
Byla vydána verze 1.78.0 programovacího jazyka Rust (Wikipedie). Podrobnosti v poznámkách k vydání na GitHubu. Vyzkoušet Rust lze například na stránce Rust by Example.
O co jde. Uživatel celý život tvořil literární díla na mechanickém psacím stroji a návrhům na užití elektrického stroje či počítače tvrdohlavě odolával. Jenže slábnoucí zrak už má problém s malými nekontrastními písmeny na papíře. Řešením by mohla být rozumně velká obrazovka s klávesnicí, ale v módu co nejvíce emulujícím psací stroj (typewriter, proto TWedit ...).
Takže co musí textový editor umět:
Mám představu, že by to běželo na upraveném raspbianu a z hlediska uživatele by se to prostě zaplo do zdi, co nejrychleji nabootovalo a psalo se. Ukládání po wifi a jako záloha lokálně (fleška ? datový oddíl). Ideální by bylo, kdyby se to dalo vypnout prostým vytažením ze zdi, což řeší i možné výpadky proudu atd. Rychlý sync jednoho souboru si dovedu po wifi představit i častěji než za vteřinu. Jen si nejsem jistý, jestli nevyjít z něčeho prefabrikovaného (QT4 ? nějaká pythoní zhůvěřilost ?), grafiku si "řešit sám" anebo to pojmout jako terminálový program ?. Dalo by se to psát i v C/C++ ale pro tento účel je i rpi "dělo", takže bych to klidně spatlal v pythonu.
Díky za tipy, nápady a připomínky (sorry, asi by to patřilo spíš do poradny než do blogu, ale myslím, že by to tam zapadlo)
Tiskni Sdílej:
No jako hračka je to pěkný. Reálně jsem na mechanickém psacím stroji začínal a musím říct, že se nechová zcela stejně. Korekční myši se stále prodávají, stejně tak páska (i když červenočervenou jsem už dlouho nekupoval, tak nevím). Spíš bych později implementoval funkci "přepiš slovo" (na jeden čudlík).
Textadept není v debianních repozitáříchStiahni si binárku.
import java.awt.Dimension; import java.awt.HeadlessException; import javax.swing.JFrame; import javax.swing.JTextPane; public class ParodyTextEditor extends JFrame { public ParodyTextEditor() throws HeadlessException { JTextPane tp = new JTextPane(); setMinimumSize(new Dimension(400, 300)); setContentPane(tp); } public static void main(String[] args) { ParodyTextEditor te = new ParodyTextEditor(); te.pack(); te.setVisible(true); } }
import java.awt.Frame; import java.awt.HeadlessException; import java.awt.event.WindowAdapter; import java.awt.event.WindowEvent; import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.FileReader; import java.io.FileWriter; import java.io.StringReader; import javax.swing.JFrame; import javax.swing.JTextPane; public class ParodyTextEditor extends JFrame { private class WindowAdapterSave extends WindowAdapter { @Override public void windowClosing(WindowEvent e) { saveFile(); } } private String fileName; JTextPane tp = new JTextPane(); public ParodyTextEditor(String fileName) throws HeadlessException { this.fileName = fileName; setContentPane(tp); addWindowListener(new WindowAdapterSave()); openFile(); } public void openFile() { try { BufferedReader fr = new BufferedReader(new FileReader(fileName)); String s = null; StringBuilder sb = new StringBuilder(); String nr = System.getProperty("line.separator"); while ((s = fr.readLine()) != null) { sb.append(s); sb.append(nr); } fr.close(); tp.setText(sb.toString()); } catch (Exception e) { e.printStackTrace(); } } public void saveFile() { BufferedReader br = new BufferedReader(new StringReader(tp.getText())); try { BufferedWriter bw = new BufferedWriter(new FileWriter(fileName)); String s = null; while ((s = br.readLine()) != null) { bw.write(s); bw.newLine(); } bw.flush(); bw.close(); } catch (Exception e) { System.err.println("Do souboru se nepovedlo zapsat."); } } public static void main(String[] args) { String fn; if (args.length == 0) fn = System.getProperty("user.home") + System.getProperty("file.separator") + "test.txt"; else fn = args[0]; ParodyTextEditor te = new ParodyTextEditor(fn); te.pack(); if (te.getExtendedState() != Frame.MAXIMIZED_BOTH) te.setExtendedState(Frame.MAXIMIZED_BOTH); te.setResizable(false); te.setVisible(true); } }
e.printStackTrace();
te.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);Takže s každým spuštěním zůstane po zavření okna aktivní instance Java Virtual Machine.
Kódovaní je implicitní Javy, kde ho nastavovat ?No právě, implicitní = závislé na platformě a jazyku/locale. Takže pokud někdo má angličtinu a otevře UTF-8 soubor, tak si užije spoustu zajímavých otazníčků (o tom že by tam byly nějaké bukvy nebo katakana ani nemluvě).
Máme 21. století, textové soubory mají být v UTF-8!
Vždy používat Files.newBufferedReader (pro 6 a nižší kombianci FileInputStream + InputStreamReader) s explicitně určeným kódováním. FileReader měl být IMCO Deprecated už v 1.2Za hrubku považuji zapomenutýJo, tohle... to si nejsem jist jak tohle vadí na serverech, tam se většinou GUI nespouštíte.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);Takže s každým spuštěním zůstane po zavření okna aktivní instance Java Virtual Machine.
Jo, tohle... to si nejsem jist jak tohle vadí na serverech, tam se většinou GUI nespouštíMe to taky zmatlo. Proto jsem zminoval to handlovani vyjimky.
FileReader určitě není Deprecated ani v Javě 1.8, tím jsem si jist.Ne, ale měl by. Spousta lidí si totiž myslí toto:
Implicitní kódování Javy je UTF8, takže by mělo vyhovět.A to právě není pravda - implicitní kódování je určeno podle nastavení uživatele (závisí na platformě: locale, user settings, lokalizace OS, atd). Ukázka (groovy, aby to byl oneliner):
augur:~ % (export LC_ALL=cs_CZ ; echo "ďábelský kůň и буквы" | groovy -e 'println new FileReader(new FileDescriptor(0)).readLine()') ďábelský kůň и буквы augur:~ % (export LC_ALL=C ; echo "ďábelský kůň и буквы" | groovy -e 'println new FileReader(new FileDescriptor(0)).readLine()') ????belsk?? k???? ?? ??????????Jasně, v případě jednoúčelového psacího stroje to problém není, ale u aplikací běžících na serveru (kde se často věci spouští bez jakéhokoliv locale) nebo na růzých mutaích Windows to bývá docela chyták.
public class JTextPaneWdthHighRow extends JTextPane { private static final long serialVersionUID = 802326126658140029L; public JTextPaneWdthHighRow() { init(); } public JTextPaneWdthHighRow(StyledDocument doc) { super(doc); init(); } private void init(){ setOpaque(false); } @Override protected void paintComponent(Graphics g) { g.setColor(getBackground()); g.fillRect(0, 0, getWidth(), getHeight()); try { Rectangle rect = modelToView(getCaretPosition()); if (rect != null) { g.setColor(Color.CYAN); g.fillRect(0, rect.y, getWidth(), rect.height); } } catch (BadLocationException e) { } super.paintComponent(g); } @Override public void repaint(long tm, int x, int y, int width, int height) { // This forces repaints to repaint the entire TextPane. super.repaint(tm, 0, 0, getWidth(), getHeight()); } }
import java.awt.Frame; import java.awt.HeadlessException; import java.awt.event.WindowAdapter; import java.awt.event.WindowEvent; import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.FileReader; import java.io.FileWriter; import java.io.StringReader; import javax.swing.JFrame; import javax.swing.JTextPane; import javax.swing.text.AbstractDocument; import javax.swing.text.AttributeSet; import javax.swing.text.BadLocationException; import javax.swing.text.DefaultStyledDocument; import javax.swing.text.DocumentFilter; import javax.swing.text.StyledDocument; public class ParodyTextEditor extends JFrame { private static final String N_L = "\n"; private class BeepDocListener extends DocumentFilter { @Override public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attrs) throws BadLocationException { if (N_L.equals(text)) { java.awt.Toolkit.getDefaultToolkit().beep(); //System.out.println("Beep"); } super.replace(fb, offset, length, text, attrs); } @Override public void insertString(FilterBypass fb, int offset, String string, AttributeSet attr) throws BadLocationException { if (N_L.equals(string)) { java.awt.Toolkit.getDefaultToolkit().beep(); //System.out.println("Beep"); } super.insertString(fb, offset, string, attr); } } private static final long serialVersionUID = -8546298818112441622L; private class WindowAdapterSave extends WindowAdapter { @Override public void windowClosing(WindowEvent e) { saveFile(); } } private String fileName; private JTextPane tp; private StyledDocument styledDocument; public ParodyTextEditor(String fileName) throws HeadlessException { this.fileName = fileName; setContentPane(getTp()); addWindowListener(new WindowAdapterSave()); openFile(); } public void openFile() { try { BufferedReader fr = new BufferedReader(new FileReader(fileName)); String s = null; StringBuilder sb = new StringBuilder(); String nr = System.getProperty("line.separator"); while ((s = fr.readLine()) != null) { sb.append(s); sb.append(nr); } fr.close(); getTp().setText(sb.toString()); } catch (Exception e) { System.err.println("Nepovedlo se nacist soubor."); e.printStackTrace(System.err); } } public void saveFile() { BufferedReader br = new BufferedReader(new StringReader(getTp().getText())); try { BufferedWriter bw = new BufferedWriter(new FileWriter(fileName)); String s = null; while ((s = br.readLine()) != null) { bw.write(s); bw.newLine(); } bw.flush(); bw.close(); } catch (Exception e) { System.err.println("Do souboru se nepovedlo zapsat."); e.printStackTrace(System.err); } } public static void main(String[] args) { String fn; if (args.length == 0) fn = System.getProperty("user.home") + System.getProperty("file.separator") + "test.txt"; else fn = args[0]; ParodyTextEditor te = new ParodyTextEditor(fn); te.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); te.pack(); if (te.getExtendedState() != Frame.MAXIMIZED_BOTH) te.setExtendedState(Frame.MAXIMIZED_BOTH); te.setResizable(false); te.setVisible(true); } protected JTextPane getTp() { if (tp == null) { tp = new JTextPane(getStyledDocument()); ((AbstractDocument) tp.getDocument()).setDocumentFilter(new BeepDocListener()); } return tp; } protected StyledDocument getStyledDocument() { if (styledDocument == null) { styledDocument = new DefaultStyledDocument(); } return styledDocument; } }